cog/Frameworks/libsidplay/sidplay-residfp-code/.svn/pristine/25/25937d3c98ac1956babada5bfdb...

219 lines
5.6 KiB
Plaintext

/*
* This file is part of libsidplayfp, a SID player engine.
*
* Copyright 2013 Leandro Nini <drfiemost@users.sourceforge.net>
* Copyright 2007-2010 Antti Lankila
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*/
#ifndef PARAMETERS_H
#define PARAMETERS_H
#include <vector>
#include <string>
#include <sstream>
class Parameters
{
public:
enum
{
BIAS,
PULSESTRENGTH,
TOPBIT,
DISTANCE,
STMIX
};
public:
float bias, pulsestrength, topbit, distance, stmix;
public:
Parameters() { reset(); }
void reset()
{
bias = 0.f;
pulsestrength = 0.f;
topbit = 0.f;
distance = 0.f;
stmix = 0.f;
}
float GetValue(int i)
{
switch (i)
{
case BIAS: return bias;
case PULSESTRENGTH: return pulsestrength;
case TOPBIT: return topbit;
case DISTANCE: return distance;
case STMIX: return stmix;
}
}
void SetValue(int i, float v)
{
switch (i)
{
case BIAS: bias = v; break;
case PULSESTRENGTH: pulsestrength = v; break;
case TOPBIT: topbit = v; break;
case DISTANCE: distance = v; break;
case STMIX: stmix = v; break;
}
}
std::string toString()
{
std::ostringstream ss;
ss << "bias = " << bias << std::endl;
ss << "pulsestrength = " << pulsestrength << std::endl;
ss << "topbit = " << topbit << std::endl;
ss << "distance = " << distance << std::endl;
ss << "stmix = " << stmix << std::endl;
return ss.str();
}
private:
void SimulateMix(float bitarray[12], float wa[], bool HasPulse)
{
float tmp[12];
for (int sb = 0; sb < 12; sb ++)
{
float n = 0.f;
float avg = 0.f;
for (int cb = 0; cb < 12; cb ++)
{
const float weight = wa[sb - cb + 12];
avg += bitarray[cb] * weight;
n += weight;
}
if (HasPulse)
{
const float weight = wa[sb - 12 + 12];
avg += pulsestrength * weight;
n += weight;
}
tmp[sb] = (bitarray[sb] + avg / n) * 0.5f;
}
for (int i = 0; i < 12; i ++)
bitarray[i] = tmp[i];
}
int GetScore8(float bitarray[12])
{
int result = 0;
for (int cb = 0; cb < 8; cb ++)
{
if (bitarray[4+cb] > bias)
result |= 1 << cb;
}
return result;
}
static int ScoreResult(int a, int b)
{
// audible error
int v = a ^ b;
return v;
/* int c = 0;
while (v != 0)
{
v &= v - 1;
c ++;
}
return c;
*/
}
public:
int Score(int wave, const std::vector<int> &reference, bool print, int bestscore)
{
int score = 0;
float wa[12 + 12 + 1];
for (int i = 0; i <= 12; i ++)
{
wa[12-i] = wa[12+i] = 1.0f / (1.0f + i * i * distance);
}
for (int j = 4095; j >= 0; j --)
{
/* S */
float bitarray[12];
for (int i = 0; i < 12; i ++)
bitarray[i] = (j & (1 << i)) != 0 ? 1.f : 0.f;
/* T */
if ((wave & 3) == 1)
{
const bool top = (j & 2048) != 0;
for (int i = 11; i > 0; i --)
{
bitarray[i] = top ? 1.f - bitarray[i-1] : bitarray[i-1];
}
bitarray[0] = 0.f;
}
/* ST */
if ((wave & 3) == 3)
{
bitarray[0] *= stmix;
for (int i = 1; i < 12; i ++)
{
bitarray[i] = bitarray[i-1] * (1.f - stmix) + bitarray[i] * stmix;
}
}
bitarray[11] *= topbit;
SimulateMix(bitarray, wa, wave > 4);
const int simval = GetScore8(bitarray);
const int refval = reference[j];
score += ScoreResult(simval, refval);
if (print)
{
float analogval = 0.f;
for (int i = 0; i < 12; i ++)
{
float val = (bitarray[i] - bias) * 512 + 0.5f;
if (val < 0.f)
val = 0.f;
else if (val > 1.f)
val = 1.f;
analogval += val * (1 << i);
}
analogval /= 16.f;
std::cout << j << " "
<< refval << " "
<< simval << " "
<< analogval << " "
<< std::endl;
}
if (score > bestscore)
{
return score;
}
}
return score;
}
};
#endif