mirror of
https://github.com/libretro/RetroArch
synced 2025-03-21 04:21:13 +00:00
Improve SNR testing routines.
This commit is contained in:
parent
5f330cbfa5
commit
cd54cc6822
@ -19,6 +19,7 @@
|
|||||||
|
|
||||||
#include "resampler.h"
|
#include "resampler.h"
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
|
#include <math.h>
|
||||||
#include "../boolean.h"
|
#include "../boolean.h"
|
||||||
|
|
||||||
#define CHANNELS 2
|
#define CHANNELS 2
|
||||||
@ -29,6 +30,18 @@ struct ssnes_resampler
|
|||||||
double r_frac;
|
double r_frac;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
void resampler_preinit(ssnes_resampler_t *re, double omega, unsigned *samples_offset)
|
||||||
|
{
|
||||||
|
*samples_offset = 4;
|
||||||
|
for (unsigned i = 0; i < 4; i++)
|
||||||
|
{
|
||||||
|
re->chan_data[0][i] = cos(i * omega);
|
||||||
|
re->chan_data[1][i] = re->chan_data[0][i];
|
||||||
|
}
|
||||||
|
|
||||||
|
re->r_frac = 0.0;
|
||||||
|
}
|
||||||
|
|
||||||
static inline float hermite_kernel(float mu1, float a, float b, float c, float d)
|
static inline float hermite_kernel(float mu1, float a, float b, float c, float d)
|
||||||
{
|
{
|
||||||
float mu2, mu3, m0, m1, a0, a1, a2, a3;
|
float mu2, mu3, m0, m1, a0, a1, a2, a3;
|
||||||
|
@ -20,6 +20,12 @@
|
|||||||
#define __SSNES_RESAMPLER_H
|
#define __SSNES_RESAMPLER_H
|
||||||
|
|
||||||
#include <stddef.h>
|
#include <stddef.h>
|
||||||
|
#include <math.h>
|
||||||
|
|
||||||
|
// M_PI is left out of ISO C99 :(
|
||||||
|
#ifndef M_PI
|
||||||
|
#define M_PI 3.14159265358979323846264338327
|
||||||
|
#endif
|
||||||
|
|
||||||
typedef struct ssnes_resampler ssnes_resampler_t;
|
typedef struct ssnes_resampler ssnes_resampler_t;
|
||||||
|
|
||||||
@ -38,5 +44,8 @@ ssnes_resampler_t *resampler_new(void);
|
|||||||
void resampler_process(ssnes_resampler_t *re, struct resampler_data *data);
|
void resampler_process(ssnes_resampler_t *re, struct resampler_data *data);
|
||||||
void resampler_free(ssnes_resampler_t *re);
|
void resampler_free(ssnes_resampler_t *re);
|
||||||
|
|
||||||
|
// Generate a starting cosine pulse with given frequency for testing (SNR, etc) purposes.
|
||||||
|
void resampler_preinit(ssnes_resampler_t *re, double omega, unsigned *samples_offset);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
17
audio/sinc.c
17
audio/sinc.c
@ -31,11 +31,6 @@
|
|||||||
#define SSNES_LOG(...)
|
#define SSNES_LOG(...)
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
// M_PI is left out of ISO C99 :(
|
|
||||||
#ifndef M_PI
|
|
||||||
#define M_PI 3.14159265358979323846264338327
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#if __SSE__
|
#if __SSE__
|
||||||
#include <xmmintrin.h>
|
#include <xmmintrin.h>
|
||||||
#endif
|
#endif
|
||||||
@ -70,6 +65,18 @@ struct ssnes_resampler
|
|||||||
uint32_t time;
|
uint32_t time;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
void resampler_preinit(ssnes_resampler_t *re, double omega, unsigned *samples_offset)
|
||||||
|
{
|
||||||
|
*samples_offset = SIDELOBES + 1;
|
||||||
|
for (int i = 0; i < 2 * SIDELOBES; i++)
|
||||||
|
{
|
||||||
|
re->buffer_l[i] = cos((i - (SIDELOBES - 1)) * omega);
|
||||||
|
re->buffer_r[i] = re->buffer_l[i];
|
||||||
|
}
|
||||||
|
|
||||||
|
re->time = 0;
|
||||||
|
}
|
||||||
|
|
||||||
static inline double sinc(double val)
|
static inline double sinc(double val)
|
||||||
{
|
{
|
||||||
if (fabs(val) < 0.00001)
|
if (fabs(val) < 0.00001)
|
||||||
|
193
audio/test/snr.c
193
audio/test/snr.c
@ -22,96 +22,64 @@
|
|||||||
#include <math.h>
|
#include <math.h>
|
||||||
#include <assert.h>
|
#include <assert.h>
|
||||||
|
|
||||||
static void gen_signal(float *out, double freq, double sample_rate, double bias_phase, size_t samples)
|
static void gen_signal(float *out, double freq, double sample_rate, unsigned bias_samples, size_t samples)
|
||||||
{
|
{
|
||||||
|
double omega = 2.0 * M_PI * freq / sample_rate;
|
||||||
|
|
||||||
for (size_t i = 0; i < samples; i += 2)
|
for (size_t i = 0; i < samples; i += 2)
|
||||||
{
|
{
|
||||||
out[i + 0] = cos((2.0 * M_PI * freq * ((i >> 1) + bias_phase)) / sample_rate);
|
out[i + 0] = cos(((i >> 1) + bias_samples) * omega);
|
||||||
out[i + 1] = out[i + 0];
|
out[i + 1] = out[i + 0];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static double calculate_snr(const float *orig, const float *resamp, size_t samples)
|
static double calculate_gain(const float *orig, const float *resamp, size_t samples)
|
||||||
|
{
|
||||||
|
double orig_power = 0.0;
|
||||||
|
double resamp_power = 0.0;
|
||||||
|
|
||||||
|
for (size_t i = 0; i < samples; i += 2)
|
||||||
|
orig_power += orig[i] * orig[i];
|
||||||
|
|
||||||
|
for (size_t i = 0; i < samples; i += 2)
|
||||||
|
resamp_power += resamp[i] * resamp[i];
|
||||||
|
|
||||||
|
return sqrt(resamp_power / orig_power);
|
||||||
|
}
|
||||||
|
|
||||||
|
struct snr_result
|
||||||
|
{
|
||||||
|
double snr;
|
||||||
|
double gain;
|
||||||
|
};
|
||||||
|
|
||||||
|
static void calculate_snr(struct snr_result *res, const float *orig, const float *resamp, size_t samples)
|
||||||
{
|
{
|
||||||
double noise = 0.0;
|
double noise = 0.0;
|
||||||
double signal = 0.0;
|
double signal = 0.0;
|
||||||
|
|
||||||
|
// Account for gain losses at higher frequencies as it's not really noise.
|
||||||
|
double filter_gain = calculate_gain(orig, resamp, samples);
|
||||||
|
double makeup_gain = 1.0 / filter_gain;
|
||||||
|
|
||||||
for (size_t i = 0; i < samples; i += 2)
|
for (size_t i = 0; i < samples; i += 2)
|
||||||
signal += orig[i] * orig[i];
|
signal += orig[i] * orig[i];
|
||||||
|
|
||||||
for (size_t i = 0; i < samples; i += 2)
|
for (size_t i = 0; i < samples; i += 2)
|
||||||
{
|
{
|
||||||
double diff = resamp[i] - orig[i];
|
double diff = makeup_gain * resamp[i] - orig[i];
|
||||||
noise += diff * diff;
|
noise += diff * diff;
|
||||||
}
|
}
|
||||||
|
|
||||||
double snr = 10 * log10(signal / noise);
|
res->snr = 10 * log10(signal / noise);
|
||||||
|
res->gain = 20.0 * log10(filter_gain);
|
||||||
return snr;
|
|
||||||
}
|
|
||||||
|
|
||||||
#define SAMPLES 0x100000
|
|
||||||
|
|
||||||
// This approach is kinda stupid.
|
|
||||||
// There should be a good way to directly (and accurately) determine phase after correlating
|
|
||||||
// the two signals.
|
|
||||||
double find_best_snr(const float *output,
|
|
||||||
size_t samples,
|
|
||||||
double freq,
|
|
||||||
double out_rate,
|
|
||||||
uint64_t *first_offset, uint64_t *last_offset,
|
|
||||||
uint64_t *first_subphase, uint64_t *last_subphase, uint64_t *subphases)
|
|
||||||
{
|
|
||||||
static float output_expected[SAMPLES];
|
|
||||||
|
|
||||||
double max_snr = -100.0;
|
|
||||||
uint64_t best_offset = *first_offset;
|
|
||||||
uint64_t best_subphase = *first_subphase;
|
|
||||||
|
|
||||||
for (uint64_t offset = *first_offset; offset <= *last_offset; offset += 2)
|
|
||||||
{
|
|
||||||
for (uint64_t subphase = *first_subphase; subphase <= *last_subphase; subphase++)
|
|
||||||
{
|
|
||||||
gen_signal(output_expected, freq, out_rate, (double)subphase / *subphases, samples);
|
|
||||||
double snr = calculate_snr(output_expected, output + offset, samples);
|
|
||||||
if (snr > max_snr)
|
|
||||||
{
|
|
||||||
max_snr = snr;
|
|
||||||
best_offset = offset;
|
|
||||||
best_subphase = subphase;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Narrow down the search area.
|
|
||||||
uint64_t left_offset = *first_offset;
|
|
||||||
uint64_t right_offset = *last_offset;
|
|
||||||
if (best_offset > left_offset)
|
|
||||||
left_offset = best_offset - 1;
|
|
||||||
if (best_offset < right_offset)
|
|
||||||
right_offset = best_offset + 1;
|
|
||||||
|
|
||||||
*first_offset = left_offset;
|
|
||||||
*last_offset = right_offset;
|
|
||||||
|
|
||||||
*subphases *= 2;
|
|
||||||
best_subphase *= 2;
|
|
||||||
|
|
||||||
uint64_t left_subphase = best_subphase - 2;
|
|
||||||
uint64_t right_subphase = best_subphase + 2;
|
|
||||||
if (best_subphase < 2)
|
|
||||||
left_subphase = 0;
|
|
||||||
|
|
||||||
*first_subphase = left_subphase;
|
|
||||||
*last_subphase = right_subphase;
|
|
||||||
|
|
||||||
return max_snr;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int main(int argc, char *argv[])
|
int main(int argc, char *argv[])
|
||||||
{
|
{
|
||||||
static float input[SAMPLES];
|
float *input;
|
||||||
static float output[SAMPLES * 8];
|
float *output;
|
||||||
|
float *output_expected;
|
||||||
|
|
||||||
if (argc != 3)
|
if (argc != 3)
|
||||||
{
|
{
|
||||||
@ -119,10 +87,10 @@ int main(int argc, char *argv[])
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
double in_rate = strtod(argv[1], NULL);
|
unsigned in_rate = strtoul(argv[1], NULL, 0);
|
||||||
double out_rate = strtod(argv[2], NULL);
|
unsigned out_rate = strtoul(argv[2], NULL, 0);
|
||||||
|
|
||||||
double ratio = out_rate / in_rate;
|
double ratio = (double)out_rate / in_rate;
|
||||||
if (ratio >= 7.99)
|
if (ratio >= 7.99)
|
||||||
{
|
{
|
||||||
fprintf(stderr, "Ratio is too high ...\n");
|
fprintf(stderr, "Ratio is too high ...\n");
|
||||||
@ -136,47 +104,78 @@ int main(int argc, char *argv[])
|
|||||||
}
|
}
|
||||||
|
|
||||||
static const float freq_list[] = {
|
static const float freq_list[] = {
|
||||||
100, 200, 400, 600, 800, 1000,
|
30, 50,
|
||||||
2000, 3000, 5000, 8000, 10000, 12000, 15000, 18000, 20000,
|
100, 150,
|
||||||
|
200, 250,
|
||||||
|
300, 350,
|
||||||
|
400, 450,
|
||||||
|
500, 550,
|
||||||
|
600, 650,
|
||||||
|
700, 800,
|
||||||
|
900, 1000,
|
||||||
|
1100, 1200,
|
||||||
|
1300, 1500,
|
||||||
|
1600, 1700,
|
||||||
|
1800, 1900,
|
||||||
|
2000, 2100,
|
||||||
|
2200, 2300,
|
||||||
|
2500, 3000,
|
||||||
|
3500, 4000,
|
||||||
|
4500, 5000,
|
||||||
|
5500, 6000,
|
||||||
|
6500, 7000,
|
||||||
|
7500, 8000,
|
||||||
|
9000, 9500,
|
||||||
|
10000, 11000,
|
||||||
|
12000, 13000,
|
||||||
|
14000, 15000,
|
||||||
|
16000, 17000,
|
||||||
|
18000, 19000,
|
||||||
|
20000, 21000,
|
||||||
|
22000,
|
||||||
};
|
};
|
||||||
|
|
||||||
for (unsigned i = 0; i < sizeof(freq_list) / sizeof(freq_list[0]); i++)
|
unsigned samples = in_rate * 2;
|
||||||
|
input = calloc(sizeof(float), samples);
|
||||||
|
output = calloc(sizeof(float), samples * 8);
|
||||||
|
output_expected = calloc(sizeof(float), samples * 8);
|
||||||
|
assert(input);
|
||||||
|
assert(output);
|
||||||
|
assert(output_expected);
|
||||||
|
|
||||||
|
ssnes_resampler_t *re = resampler_new();
|
||||||
|
assert(re);
|
||||||
|
|
||||||
|
for (unsigned i = 0; i < sizeof(freq_list) / sizeof(freq_list[0]) && freq_list[i] < 0.5f * in_rate; i++)
|
||||||
{
|
{
|
||||||
gen_signal(input, freq_list[i], in_rate, 0.0, SAMPLES);
|
double omega = 2.0 * M_PI * freq_list[i] / in_rate;
|
||||||
|
unsigned sample_offset;
|
||||||
|
resampler_preinit(re, omega, &sample_offset);
|
||||||
|
|
||||||
|
gen_signal(input, freq_list[i], in_rate, sample_offset, samples);
|
||||||
|
|
||||||
struct resampler_data data = {
|
struct resampler_data data = {
|
||||||
.data_in = input,
|
.data_in = input,
|
||||||
.data_out = output,
|
.data_out = output,
|
||||||
.input_frames = SAMPLES / 2,
|
.input_frames = in_rate,
|
||||||
.ratio = ratio,
|
.ratio = ratio,
|
||||||
};
|
};
|
||||||
|
|
||||||
ssnes_resampler_t *re = resampler_new();
|
|
||||||
assert(re);
|
|
||||||
resampler_process(re, &data);
|
resampler_process(re, &data);
|
||||||
resampler_free(re);
|
|
||||||
|
|
||||||
#define MAX_OFFSET 128
|
unsigned out_samples = data.output_frames * 2;
|
||||||
uint64_t first_offset = 0;
|
gen_signal(output_expected, freq_list[i], out_rate, 0, out_samples);
|
||||||
uint64_t last_offset = MAX_OFFSET - 2;
|
|
||||||
uint64_t first_subphase = 0;
|
|
||||||
uint64_t last_subphase = 1;
|
|
||||||
uint64_t subphases = 2;
|
|
||||||
|
|
||||||
double max_snr = -100.0;
|
struct snr_result res;
|
||||||
|
calculate_snr(&res, output_expected, output, out_samples);
|
||||||
|
|
||||||
// Iteratively find the correct SNR value.
|
printf("SNR @ %7.1f Hz: %6.2lf dB, Gain: %6.1f dB\n",
|
||||||
for (unsigned j = 0; j < 48; j++)
|
freq_list[i], res.snr, res.gain);
|
||||||
{
|
|
||||||
double snr = find_best_snr(output, SAMPLES - MAX_OFFSET, freq_list[i], out_rate,
|
|
||||||
&first_offset, &last_offset,
|
|
||||||
&first_subphase, &last_subphase, &subphases);
|
|
||||||
|
|
||||||
if (snr > max_snr)
|
|
||||||
max_snr = snr;
|
|
||||||
}
|
|
||||||
|
|
||||||
printf("SNR @ %.0f Hz: %lf dB\n", freq_list[i], max_snr);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
resampler_free(re);
|
||||||
|
free(input);
|
||||||
|
free(output);
|
||||||
|
free(output_expected);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user