2012-02-25 01:47:23 +01:00
|
|
|
/* SSNES - A Super Nintendo Entertainment System (SNES) Emulator frontend for libsnes.
|
|
|
|
* Copyright (C) 2010-2012 - Hans-Kristian Arntzen
|
|
|
|
*
|
|
|
|
* Some code herein may be based on code found in BSNES.
|
|
|
|
*
|
|
|
|
* SSNES 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 Found-
|
|
|
|
* ation, either version 3 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* SSNES 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 SSNES.
|
|
|
|
* If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "../resampler.h"
|
|
|
|
#include "../utils.h"
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <math.h>
|
2012-02-27 19:49:00 +01:00
|
|
|
#include <complex.h>
|
2012-02-25 01:47:23 +01:00
|
|
|
#include <assert.h>
|
2012-02-27 19:49:00 +01:00
|
|
|
#include <stdbool.h>
|
2012-02-25 01:47:23 +01:00
|
|
|
|
2012-02-25 21:17:48 +01:00
|
|
|
static void gen_signal(float *out, double omega, double bias_samples, size_t samples)
|
2012-02-25 01:47:23 +01:00
|
|
|
{
|
|
|
|
for (size_t i = 0; i < samples; i += 2)
|
|
|
|
{
|
2012-02-25 14:02:56 +01:00
|
|
|
out[i + 0] = cos(((i >> 1) + bias_samples) * omega);
|
2012-02-25 01:47:23 +01:00
|
|
|
out[i + 1] = out[i + 0];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-02-27 19:49:00 +01:00
|
|
|
struct snr_result
|
2012-02-25 01:47:23 +01:00
|
|
|
{
|
2012-02-27 19:49:00 +01:00
|
|
|
double snr;
|
|
|
|
double gain;
|
|
|
|
};
|
2012-02-25 01:47:23 +01:00
|
|
|
|
2012-02-27 19:49:00 +01:00
|
|
|
static unsigned bitrange(unsigned len)
|
|
|
|
{
|
|
|
|
unsigned ret = 0;
|
|
|
|
while ((len >>= 1))
|
|
|
|
ret++;
|
2012-02-25 01:47:23 +01:00
|
|
|
|
2012-02-27 19:49:00 +01:00
|
|
|
return ret;
|
2012-02-25 01:47:23 +01:00
|
|
|
}
|
|
|
|
|
2012-02-27 19:49:00 +01:00
|
|
|
static unsigned bitswap(unsigned i, unsigned range)
|
2012-02-25 21:17:48 +01:00
|
|
|
{
|
2012-02-27 19:49:00 +01:00
|
|
|
unsigned ret = 0;
|
|
|
|
for (unsigned shifts = 0; shifts < range; shifts++)
|
|
|
|
ret |= i & (1 << (range - shifts - 1)) ? (1 << shifts) : 0;
|
2012-02-25 21:17:48 +01:00
|
|
|
|
2012-02-27 19:49:00 +01:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
// When interleaving the butterfly buffer, addressing puts bits in reverse.
|
|
|
|
// [0, 1, 2, 3, 4, 5, 6, 7] => [0, 4, 2, 6, 1, 5, 3, 7]
|
|
|
|
static void interleave(complex double *butterfly_buf, size_t samples)
|
|
|
|
{
|
|
|
|
unsigned range = bitrange(samples);
|
|
|
|
for (unsigned i = 0; i < samples; i++)
|
2012-02-25 21:17:48 +01:00
|
|
|
{
|
2012-02-27 19:49:00 +01:00
|
|
|
unsigned target = bitswap(i, range);
|
|
|
|
if (target > i)
|
|
|
|
{
|
|
|
|
complex double tmp = butterfly_buf[target];
|
|
|
|
butterfly_buf[target] = butterfly_buf[i];
|
|
|
|
butterfly_buf[i] = tmp;
|
|
|
|
}
|
2012-02-25 21:17:48 +01:00
|
|
|
}
|
2012-02-27 19:49:00 +01:00
|
|
|
}
|
2012-02-25 21:17:48 +01:00
|
|
|
|
2012-02-27 19:49:00 +01:00
|
|
|
static complex double gen_phase(double index)
|
|
|
|
{
|
|
|
|
return cexp(-M_PI * I * index);
|
|
|
|
}
|
2012-02-25 21:17:48 +01:00
|
|
|
|
2012-02-27 19:49:00 +01:00
|
|
|
static void butterfly(complex double *a, complex double *b, complex double mod)
|
|
|
|
{
|
|
|
|
mod *= *b;
|
|
|
|
complex double a_ = *a + mod;
|
|
|
|
complex double b_ = *a - mod;
|
|
|
|
*a = a_;
|
|
|
|
*b = b_;
|
2012-02-25 21:17:48 +01:00
|
|
|
}
|
|
|
|
|
2012-02-27 19:49:00 +01:00
|
|
|
static void butterflies(complex double *butterfly_buf, size_t step_size, size_t samples)
|
2012-02-25 01:47:23 +01:00
|
|
|
{
|
2012-02-27 19:49:00 +01:00
|
|
|
for (unsigned i = 0; i < samples; i += 2 * step_size)
|
|
|
|
for (unsigned j = i; j < i + step_size; j++)
|
|
|
|
butterfly(&butterfly_buf[j], &butterfly_buf[j + step_size], gen_phase((double)(j - i) / step_size));
|
|
|
|
}
|
2012-02-25 01:47:23 +01:00
|
|
|
|
2012-02-27 19:49:00 +01:00
|
|
|
static void calculate_fft(const float *data, complex double *butterfly_buf, size_t samples)
|
2012-02-25 14:02:56 +01:00
|
|
|
{
|
2012-02-27 19:49:00 +01:00
|
|
|
// Enforce POT.
|
|
|
|
assert((samples & (samples - 1)) == 0);
|
2012-02-25 01:47:23 +01:00
|
|
|
|
2012-02-27 19:49:00 +01:00
|
|
|
for (unsigned i = 0; i < samples; i++)
|
|
|
|
butterfly_buf[i] = data[2 * i];
|
2012-02-25 21:17:48 +01:00
|
|
|
|
2012-02-27 19:49:00 +01:00
|
|
|
// Interleave buffer to work with FFT.
|
|
|
|
interleave(butterfly_buf, samples);
|
2012-02-25 01:47:23 +01:00
|
|
|
|
2012-02-27 19:49:00 +01:00
|
|
|
// Fly, lovely butterflies! :D
|
|
|
|
for (unsigned step_size = 1; step_size < samples; step_size *= 2)
|
|
|
|
butterflies(butterfly_buf, step_size, samples);
|
2012-02-25 01:47:23 +01:00
|
|
|
|
2012-02-27 19:49:00 +01:00
|
|
|
// We only have real data.
|
|
|
|
for (unsigned i = 1; i < samples / 2; i++)
|
2012-02-28 00:29:45 +01:00
|
|
|
butterfly_buf[i] += conj(butterfly_buf[samples - i]);
|
2012-02-25 01:47:23 +01:00
|
|
|
|
2012-02-27 19:49:00 +01:00
|
|
|
// Normalize amplitudes.
|
|
|
|
for (unsigned i = 0; i < samples / 2; i++)
|
|
|
|
butterfly_buf[i] /= (double)samples;
|
2012-02-25 01:47:23 +01:00
|
|
|
}
|
|
|
|
|
2012-02-27 19:49:00 +01:00
|
|
|
static void test_fft(void)
|
2012-02-25 01:47:23 +01:00
|
|
|
{
|
2012-02-27 19:49:00 +01:00
|
|
|
fprintf(stderr, "Sanity checking FFT ...\n");
|
|
|
|
float signal[32];
|
|
|
|
complex double butterfly_buf[16];
|
2012-02-25 01:47:23 +01:00
|
|
|
|
2012-02-28 00:29:45 +01:00
|
|
|
const float cos_freqs[] = {
|
2012-02-27 19:49:00 +01:00
|
|
|
1.0, 4.0, 6.0,
|
|
|
|
};
|
|
|
|
|
2012-02-28 00:29:45 +01:00
|
|
|
const float sin_freqs[] = {
|
|
|
|
-2.0, 5.0, 7.0,
|
|
|
|
};
|
|
|
|
|
2012-02-27 19:49:00 +01:00
|
|
|
for (unsigned i = 0; i < 16; i++)
|
2012-02-25 01:47:23 +01:00
|
|
|
{
|
2012-02-27 19:49:00 +01:00
|
|
|
signal[2 * i] = 0.0;
|
2012-02-28 00:29:45 +01:00
|
|
|
for (unsigned j = 0; j < sizeof(cos_freqs) / sizeof(cos_freqs[0]); j++)
|
|
|
|
signal[2 * i] += cos(2.0 * M_PI * i * cos_freqs[j] / 16.0);
|
|
|
|
for (unsigned j = 0; j < sizeof(sin_freqs) / sizeof(sin_freqs[0]); j++)
|
|
|
|
signal[2 * i] += sin(2.0 * M_PI * i * sin_freqs[j] / 16.0);
|
2012-02-25 01:47:23 +01:00
|
|
|
}
|
|
|
|
|
2012-02-27 19:49:00 +01:00
|
|
|
calculate_fft(signal, butterfly_buf, 16);
|
|
|
|
|
|
|
|
printf("FFT: { ");
|
|
|
|
for (unsigned i = 0; i < 7; i++)
|
2012-02-28 00:29:45 +01:00
|
|
|
printf("(%4.2lf, %4.2lf), ", creal(butterfly_buf[i]), cimag(butterfly_buf[i]));
|
|
|
|
printf("(%4.2lf, %4.2lf) }\n", creal(butterfly_buf[7]), cimag(butterfly_buf[7]));
|
2012-02-27 19:49:00 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// This doesn't yet take account for slight phase distortions,
|
|
|
|
// so reported SNR is lower than reality.
|
|
|
|
static void calculate_snr(struct snr_result *res,
|
|
|
|
unsigned in_rate,
|
|
|
|
const float *resamp, complex double *butterfly_buf, size_t samples)
|
|
|
|
{
|
|
|
|
samples >>= 1;
|
|
|
|
calculate_fft(resamp, butterfly_buf, samples);
|
|
|
|
|
2012-02-27 21:13:27 +01:00
|
|
|
double signal = cabs(butterfly_buf[in_rate] * butterfly_buf[in_rate]);
|
2012-02-27 19:49:00 +01:00
|
|
|
butterfly_buf[in_rate] = 0.0;
|
|
|
|
|
|
|
|
double noise = 0.0;
|
|
|
|
for (unsigned i = 0; i < samples / 2; i++)
|
|
|
|
noise += cabs(butterfly_buf[i] * butterfly_buf[i]);
|
2012-02-25 01:47:23 +01:00
|
|
|
|
2012-02-27 19:49:00 +01:00
|
|
|
res->snr = 10.0 * log10(signal / noise);
|
|
|
|
res->gain = 10.0 * log10(signal);
|
|
|
|
}
|
|
|
|
|
|
|
|
int main(int argc, char *argv[])
|
|
|
|
{
|
|
|
|
if (argc != 2)
|
2012-02-25 01:47:23 +01:00
|
|
|
{
|
2012-02-27 19:49:00 +01:00
|
|
|
fprintf(stderr, "Usage: %s <ratio> (out-rate is fixed for FFT).\n", argv[0]);
|
2012-02-25 01:47:23 +01:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2012-02-27 19:49:00 +01:00
|
|
|
double ratio = strtod(argv[1], NULL);
|
|
|
|
|
|
|
|
const unsigned fft_samples = 1024 * 128;
|
|
|
|
unsigned out_rate = fft_samples;
|
|
|
|
unsigned in_rate = out_rate / ratio;
|
|
|
|
ratio = (double)out_rate / in_rate;
|
|
|
|
|
|
|
|
if (ratio <= 1.0)
|
2012-02-25 01:47:23 +01:00
|
|
|
{
|
|
|
|
fprintf(stderr, "Ratio too low ...\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2012-02-27 21:34:31 +01:00
|
|
|
static const float freq_list[] = {
|
|
|
|
0.001, 0.002, 0.003, 0.004, 0.005, 0.008,
|
|
|
|
0.010, 0.015, 0.020, 0.025, 0.030, 0.035, 0.040, 0.045, 0.050,
|
|
|
|
0.10, 0.15, 0.20, 0.25, 0.30, 0.35, 0.40, 0.45,
|
|
|
|
0.46, 0.47, 0.48, 0.49, 0.495,
|
2012-02-25 01:47:23 +01:00
|
|
|
};
|
|
|
|
|
2012-02-25 14:02:56 +01:00
|
|
|
unsigned samples = in_rate * 2;
|
2012-02-27 19:49:00 +01:00
|
|
|
float *input = calloc(sizeof(float), samples);
|
|
|
|
float *output = calloc(sizeof(float), (fft_samples + 1) * 2);
|
|
|
|
complex double *butterfly_buf = calloc(sizeof(complex double), fft_samples);
|
|
|
|
bool warned = false;
|
2012-02-25 14:02:56 +01:00
|
|
|
assert(input);
|
|
|
|
assert(output);
|
|
|
|
|
|
|
|
ssnes_resampler_t *re = resampler_new();
|
|
|
|
assert(re);
|
|
|
|
|
2012-02-27 19:49:00 +01:00
|
|
|
test_fft();
|
|
|
|
|
2012-02-27 21:34:31 +01:00
|
|
|
for (unsigned i = 0; i < sizeof(freq_list) / sizeof(freq_list[0]); i++)
|
2012-02-25 01:47:23 +01:00
|
|
|
{
|
2012-02-27 21:34:31 +01:00
|
|
|
unsigned freq = freq_list[i] * in_rate;
|
|
|
|
double omega = 2.0 * M_PI * freq / in_rate;
|
2012-02-25 14:31:36 +01:00
|
|
|
double sample_offset;
|
2012-02-25 14:02:56 +01:00
|
|
|
resampler_preinit(re, omega, &sample_offset);
|
2012-02-25 21:17:48 +01:00
|
|
|
gen_signal(input, omega, sample_offset, samples);
|
2012-02-25 01:47:23 +01:00
|
|
|
|
|
|
|
struct resampler_data data = {
|
|
|
|
.data_in = input,
|
|
|
|
.data_out = output,
|
2012-02-25 14:02:56 +01:00
|
|
|
.input_frames = in_rate,
|
2012-02-25 01:47:23 +01:00
|
|
|
.ratio = ratio,
|
|
|
|
};
|
|
|
|
|
|
|
|
resampler_process(re, &data);
|
|
|
|
|
2012-02-25 14:02:56 +01:00
|
|
|
unsigned out_samples = data.output_frames * 2;
|
2012-02-25 01:47:23 +01:00
|
|
|
|
2012-02-27 19:49:00 +01:00
|
|
|
if (out_samples != fft_samples * 2 && !warned)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "Out samples != fft_samples ... %u / %u\n", out_samples, fft_samples * 2);
|
|
|
|
warned = true;
|
|
|
|
}
|
|
|
|
|
2012-02-25 14:02:56 +01:00
|
|
|
struct snr_result res;
|
2012-02-27 21:34:31 +01:00
|
|
|
calculate_snr(&res, freq, output, butterfly_buf, fft_samples * 2);
|
2012-02-25 01:47:23 +01:00
|
|
|
|
2012-02-27 21:34:31 +01:00
|
|
|
printf("SNR @ w = %5.3f : %6.2lf dB, Gain: %6.1lf dB\n",
|
2012-02-27 21:13:27 +01:00
|
|
|
freq_list[i], res.snr, res.gain);
|
2012-02-25 01:47:23 +01:00
|
|
|
}
|
2012-02-25 14:02:56 +01:00
|
|
|
|
|
|
|
resampler_free(re);
|
|
|
|
free(input);
|
|
|
|
free(output);
|
2012-02-27 19:49:00 +01:00
|
|
|
free(butterfly_buf);
|
2012-02-25 01:47:23 +01:00
|
|
|
}
|
|
|
|
|