wdsp/utilities.c
Uladzimir Karpenka 89c8a0e2b5 first commit
2026-06-01 15:58:45 +03:00

640 lines
18 KiB
C

/* utilities.c
This file is part of a program that implements a Software-Defined Radio.
Copyright (C) 2013, 2019, 2024 Warren Pratt, NR0V
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.
The author can be reached by email at
warren@wpratt.com
*/
#define _CRT_SECURE_NO_WARNINGS
#include "comm.h"
/********************************************************************************************************
* *
* Required Utilities *
* *
********************************************************************************************************/
PORT
void *malloc0 (int size)
{
int alignment = 16;
void* p = _aligned_malloc (size, alignment);
if (p != 0) memset (p, 0, size);
return p;
}
#if !defined(linux) && !defined(__APPLE__)
// Exported calls
PORT void
*NewCriticalSection()
{ // used by VAC
LPCRITICAL_SECTION cs_ptr;
cs_ptr = (LPCRITICAL_SECTION)calloc (1,sizeof (CRITICAL_SECTION));
return (void *)cs_ptr;
}
PORT void
DestroyCriticalSection (LPCRITICAL_SECTION cs_ptr)
{ // used by VAC
free ((char *)cs_ptr);
}
#endif
/********************************************************************************************************
* *
* Test & Debug Utilities *
* *
********************************************************************************************************/
void print_impulse (const char* filename, int N, double* impulse, int rtype, int pr_mode)
{
int i;
FILE* file;
if (pr_mode == 0)
file = fopen (filename, "w");
else
file = fopen (filename, "a");
if (file)
{
if (rtype == 0)
for (i = 0; i < N; i++)
fprintf(file, "%.17e\n", impulse[i]);
else
for (i = 0; i < N; i++)
fprintf(file, "%.17e\t%.17e\n", impulse[2 * i + 0], impulse[2 * i + 1]);
fprintf(file, "\n\n\n\n");
fflush(file);
fclose(file);
}
}
PORT
void analyze_bandpass_filter (int N, double f_low, double f_high, double samplerate, int wintype, int rtype, double scale)
{
double* linphase_imp;
double* minphase_imp = (double *) malloc0 (N * sizeof (complex));
linphase_imp = fir_bandpass (N, f_low, f_high, samplerate, wintype, rtype, scale);
mp_imp (N, linphase_imp, minphase_imp, 16, 0);
print_impulse ("linear_phase_impulse.txt", N, linphase_imp, 1, 0);
print_impulse ("minimum_phase_impulse.txt", N, minphase_imp, 1, 0);
_aligned_free (minphase_imp);
_aligned_free (linphase_imp);
}
void print_peak_val (const char* filename, int N, double* buff, double thresh)
{
int i;
static unsigned int seqnum;
double peak = 0.0;
FILE* file;
for (i = 0; i < N; i++)
if (buff[i] > peak) peak = buff[i];
if (peak >= thresh)
{
if (file = fopen(filename, "a"))
{
fprintf(file, "%d\t\t%.17e\n", seqnum, peak);
fflush(file);
fclose(file);
}
}
seqnum++;
}
void print_peak_env (const char* filename, int N, double* buff, double thresh)
{
int i;
static unsigned int seqnum;
double peak = 0.0;
double new_peak;
FILE* file;
for (i = 0; i < N; i++)
{
new_peak = sqrt (buff[2 * i + 0] * buff[2 * i + 0] + buff[2 * i + 1] * buff[2 * i + 1]);
if (new_peak > peak) peak = new_peak;
}
if (peak >= thresh)
{
if (file = fopen(filename, "a"))
{
fprintf(file, "%d\t\t%.17e\n", seqnum, peak);
fflush(file);
fclose(file);
}
}
seqnum++;
}
void print_peak_env_f2 (const char* filename, int N, float* Ibuff, float* Qbuff)
{
int i;
double peak = 0.0;
double new_peak;
FILE* file;
if (file = fopen(filename, "a"))
{
for (i = 0; i < N; i++)
{
new_peak = sqrt (Ibuff[i] * Ibuff[i] + Qbuff[i] * Qbuff[i]);
if (new_peak > peak) peak = new_peak;
}
fprintf(file, "%.17e\n", peak);
fflush(file);
fclose(file);
}
}
void print_iqc_values (const char* filename, int state, double env_in, double I, double Q, double ym, double yc, double ys, double thresh)
{
static unsigned int seqnum;
double env_out;
FILE* file;
env_out = sqrt (I * I + Q * Q);
if (env_out > thresh)
{
if (file = fopen(filename, "a"))
{
if (seqnum == 0)
fprintf(file, "seqnum\tstate\tenv_in\t\tenv_out\t\tym\t\tyc\t\tys\n");
fprintf(file, "%d\t%d\t%f\t%f\t%f\t%f\t%f\n", seqnum, state, env_in, env_out, ym, yc, ys);
fflush(file);
fclose(file);
seqnum++;
}
}
}
PORT
void print_buffer_parameters (const char* filename, int channel)
{
IOB a = ch[channel].iob.pc;
FILE* file;
if (file = fopen(filename, "a"))
{
fprintf(file, "channel = %d\n", channel);
fprintf(file, "in_size = %d\n", a->in_size);
fprintf(file, "r1_outsize = %d\n", a->r1_outsize);
fprintf(file, "r1_size = %d\n", a->r1_size);
fprintf(file, "r2_size = %d\n", a->r2_size);
fprintf(file, "out_size = %d\n", a->out_size);
fprintf(file, "r2_insize = %d\n", a->r2_insize);
fprintf(file, "r1_active_buffsize = %d\n", a->r1_active_buffsize);
fprintf(file, "f2_active_buffsize = %d\n", a->r2_active_buffsize);
fprintf(file, "r1_inidx = %d\n", a->r1_inidx);
fprintf(file, "r1_outidx = %d\n", a->r1_outidx);
fprintf(file, "r1_unqueuedsamps = %d\n", a->r1_unqueuedsamps);
fprintf(file, "r2_inidx = %d\n", a->r2_inidx);
fprintf(file, "r2_outidx = %d\n", a->r2_outidx);
fprintf(file, "r2_havesamps = %d\n", a->r2_havesamps);
fprintf(file, "in_rate = %d\n", ch[channel].in_rate);
fprintf(file, "dsp_rate = %d\n", ch[channel].dsp_rate);
fprintf(file, "out_rate = %d\n", ch[channel].out_rate);
fprintf(file, "\n");
fflush(file);
fclose(file);
}
}
void print_meter (const char* filename, double* meter, int enum_av, int enum_pk, int enum_gain)
{
FILE* file;
if (file = fopen(filename, "a"))
{
if (enum_gain >= 0)
fprintf(file, "%.4e\t%.4e\t%.4e\n", meter[enum_av], meter[enum_pk], meter[enum_gain]);
else
fprintf(file, "%.4e\t%.4e\n", meter[enum_av], meter[enum_pk]);
fflush(file);
fclose(file);
}
}
void print_message (const char* filename, const char* message, int p0, int p1, int p2)
{
FILE* file;
if (file = fopen(filename, "a"))
{
const char* msg = message;
fprintf(file, "%s %d %d %d\n", msg, p0, p1, p2);
fflush(file);
fclose(file);
}
}
void print_window_gain (const char* filename, int wintype, double inv_coherent_gain, double inherent_power_gain)
{
FILE* file;
if (file = fopen(filename, "a"))
{
double enb = inherent_power_gain * inv_coherent_gain * inv_coherent_gain;
switch (wintype)
{
case 0:
fprintf(file, "Rectangular %.4f\t%.4f\t%.4f\n", inv_coherent_gain, inherent_power_gain, enb);
break;
case 1:
fprintf(file, "Blackman-Harris 4-term %.4f\t%.4f\t%.4f\n", inv_coherent_gain, inherent_power_gain, enb);
break;
case 2:
fprintf(file, "Hann %.4f\t%.4f\t%.4f\n", inv_coherent_gain, inherent_power_gain, enb);
break;
case 3:
fprintf(file, "Flat Top %.4f\t%.4f\t%.4f\n", inv_coherent_gain, inherent_power_gain, enb);
break;
case 4:
fprintf(file, "Hamming %.4f\t%.4f\t%.4f\n", inv_coherent_gain, inherent_power_gain, enb);
break;
case 5:
fprintf(file, "Kaiser %.4f\t%.4f\t%.4f\n", inv_coherent_gain, inherent_power_gain, enb);
break;
case 6:
fprintf(file, "Blackman-Harris 7-term %.4f\t%.4f\t%.4f\n", inv_coherent_gain, inherent_power_gain, enb);
break;
default:
fprintf(file, "Specified Window Type is Invalid\n");
break;
}
fflush(file);
fclose(file);
}
}
void print_deviation (const char* filename, double dpmax, double rate)
{
FILE* file;
if (file = fopen(filename, "a"))
{
double peak = dpmax * rate / TWOPI;
fprintf(file, "Peak Dev = %.4f\n", peak);
fflush(file);
fclose(file);
}
}
void __cdecl CalccPrintSamples (void *pargs)
{
int i;
double env_tx, env_rx;
int channel = (int)(uintptr_t)pargs;
CALCC a = txa[channel].calcc.p;
FILE* file;
if (file = fopen("samples.txt", "w"))
{
fprintf(file, "\n");
for (i = 0; i < a->nsamps; i++)
{
env_tx = sqrt(a->txs[2 * i + 0] * a->txs[2 * i + 0] + a->txs[2 * i + 1] * a->txs[2 * i + 1]);
env_rx = sqrt(a->rxs[2 * i + 0] * a->rxs[2 * i + 0] + a->rxs[2 * i + 1] * a->rxs[2 * i + 1]);
fprintf(file, "%.12f %.12f %.12f %.12f %.12f %.12f\n",
a->txs[2 * i + 0], a->txs[2 * i + 1], env_tx,
a->rxs[2 * i + 0], a->rxs[2 * i + 1], env_rx);
}
fflush(file);
fclose(file);
}
_endthread();
}
void doCalccPrintSamples(int channel)
{ // no sample buffering - use in single cal mode
_beginthread(CalccPrintSamples, 0, (void *)(uintptr_t)channel);
}
void print_anb_parms (const char* filename, ANB a)
{
FILE* file;
if (file = fopen(filename, "a"))
{
fprintf(file, "Run = %d\n", a->run);
fprintf(file, "Buffer Size = %d\n", a->buffsize);
fprintf(file, "Sample Rate = %d\n", (int)a->samplerate);
fprintf(file, "Threshold = %.6f\n", a->threshold);
fprintf(file, "BackTau = %.6f\n", a->backtau);
fprintf(file, "BackMult = %.6f\n", a->backmult);
fprintf(file, "Tau = %.6f\n", a->tau);
fflush(file);
fclose(file);
}
}
#if !defined(linux) && !defined(__APPLE__)
// Audacity: Import Raw Data, Signed 32-bit PCM, Little-endian, Mono/Stereo per mode selection, 48K rate
int audiocount = 0;
int* data = 0;
int ready = 0;
int done = 0;
void WriteAudioFile(void* arg)
{
byte* dat = (byte *)arg;
FILE* file;
// reverse bits of each byte (possibly needed on some platforms)
// int i;
// byte b;
// for (i = 0; i < 4 * audiocount; i++)
// {
// b = dat[i];
// b = ((b >> 1) & 0x55) | ((b << 1) & 0xaa);
// b = ((b >> 2) & 0x33) | ((b << 2) & 0xcc);
// b = ((b >> 4) & 0x0f) | ((b << 4) & 0xf0);
// dat[i] = b;
// }
if (file = fopen("AudioFile", "wb"))
{
fwrite((int*)dat, sizeof(int), audiocount, file);
fflush(file);
fclose(file);
}
_aligned_free(data);
data = 0;
_endthread();
}
void WriteAudioWDSP (double seconds, int rate, int size, double* indata, int mode, double gain)
{
// seconds - number of seconds of audio to record
// rate - sample rate
// size - number of complex samples
// indata - pointer to data
static int n;
int i;
const double conv = 2147483647.0 * gain;
if (!ready)
{
if (mode != 3)
n = (int)(seconds * rate);
else
n = 2 * (int)(seconds * rate);
ready = 1;
}
if (!data) data = (int*)malloc0(n * sizeof(int));
for (i = 0; i < size; i++)
{
if (audiocount < n)
{
switch (mode)
{
case 0: // I only (mono)
data[audiocount++] = (int)(conv * indata[2 * i + 0]);
break;
case 1: // Q only (mono)
data[audiocount++] = (int)(conv * indata[2 * i + 1]);
break;
case 2: // envelope (mono)
data[audiocount++] = (int)(conv * sqrt(indata[2 * i + 0] * indata[2 * i + 0] + indata[2 * i + 1] * indata[2 * i + 1]));
break;
case 3: // complex samples (stereo)
data[audiocount++] = (int)(conv * indata[2 * i + 0]);
data[audiocount++] = (int)(conv * indata[2 * i + 1]);
break;
case 4: // double samples (mono)
data[audiocount++] = (int)(conv * indata[i]);
break;
default: // invalid mode/datatype
data[audiocount++] = 0;
break;
}
}
}
if ((audiocount == n) && !done)
{
done = 1;
_beginthread(WriteAudioFile, 0, (void *)data);
}
}
void WriteScaledAudioFile (void* arg)
{
typedef struct
{
int n;
double* ddata;
} *dstr;
dstr dstruct = (dstr)arg;
FILE* file;
int i;
double max = 0.0;
double abs_val;
const double conv = 2147483647.0;
int *idata = (int *) malloc0 (dstruct->n * sizeof (int));
for (i = 0; i < dstruct->n; i++)
{
abs_val = fabs (dstruct->ddata[i]);
if (abs_val > max)
max = abs_val;
}
for (i = 0; i < dstruct->n; i++)
idata[i] = dstruct->ddata[i] >= 0.0 ? (int)floor(conv * dstruct->ddata[i] / max + 0.5) : (int)ceil(conv * dstruct->ddata[i] / max - 0.5);
if (file = fopen("AudioFile", "wb"))
{
fwrite(idata, sizeof(int), dstruct->n, file);
fflush(file);
fclose(file);
}
_aligned_free (dstruct->ddata);
_aligned_free (dstruct);
_aligned_free (idata);
_endthread();
}
void WriteScaledAudio (
double seconds, // number of seconds of audio to record
int rate, // sample rate
int size, // incoming buffer size
double* indata ) // pointer to incoming data buffer
{
static int ready;
typedef struct
{
int n;
double* ddata;
} dstr, *DSTR;
static DSTR dstruct;
static int count, complete;
int i;
if (!ready)
{
dstruct = (DSTR) malloc0 (sizeof (dstr));
dstruct->n = 2 * (int)(seconds * rate);
dstruct->ddata = (double *) malloc0 (dstruct->n * sizeof (double));
ready = 1;
}
for (i = 0; i < size; i++)
{
if (count < dstruct->n)
{
dstruct->ddata[count++] = indata[2 * i + 0];
dstruct->ddata[count++] = indata[2 * i + 1];
}
}
if ((count >= dstruct->n) && !complete)
{
complete = 1;
_beginthread (WriteScaledAudioFile, 0, (void *)dstruct);
}
}
#endif
/********************************************************************************************************
* *
* Bandpass Filter Characterization Utility *
* *
********************************************************************************************************/
double* model_bandpass(int nc, double f_low, double f_high, double rate, int wtype, int points)
{
double* h = fir_bandpass(nc, f_low, f_high, rate, wtype, 1, 1.0 / (double)nc);
double* in = (double*)malloc0(points * sizeof(complex));
double* out = (double*)malloc0(points * sizeof(complex));
memcpy(in, h, nc * sizeof(complex));
fftw_plan p = fftw_plan_dft_1d(points, (fftw_complex*)in, (fftw_complex*)out, FFTW_FORWARD, FFTW_PATIENT);
fftw_execute(p);
fftw_destroy_plan(p);
double* mag = (double*)malloc0(points * sizeof(double));
double mult = 1.0/sqrt(out[0] * out[0] + out[1] * out[1]);
for (int i = 0; i < points; i++)
{
mag[i] = mult * sqrt(out[2 * i + 0] * out[2 * i + 0] + out[2 * i + 1] * out[2 * i + 1]);
if (mag[i] > 1.0e-300)
mag[i] = 20.0 * log10(mag[i]);
else
mag[i] = -200.0;
}
// reverse normal-order
double* magrev = (double*)malloc0(points * sizeof(double));
memcpy(magrev, &mag[points / 2], points / 2 * sizeof(double));
memcpy(&magrev[points / 2], mag, points / 2 * sizeof(double));
_aligned_free(mag);
_aligned_free(out);
_aligned_free(in);
_aligned_free(h);
return magrev;
}
void print_bandpass_response (const char* filename, int points, double* response)
{
int i;
FILE* file;
if (file = fopen(filename, "w"))
{
for (i = 0; i < points; i++)
fprintf(file, "%.17e\n", response[i]);
fflush(file);
fclose(file);
}
}
BFCU pbfcu[4];
PORT
int create_bfcu(int id, int min_size, int max_size, double rate, double corner, int points)
{
// id - from 0 through 3
// min_size = minimum impulse response size for filters (power of two)
// max_size = maximum impulse response size for filters (power of two)
// rate = sample-rate at which filters run
// corner = -6dB corner frequency; bandpass filter is symmetrical about zero
// two corners, one at +corner and the other at -corner
// points = number of points to generate for each filter response (power of two; >= max_size)
if (max_size > points) return -1;
BFCU a = (BFCU)malloc0(sizeof(bfcu));
int nc, i_wtype, i_dataset;
int i_corner_offset;
a->id = id;
a->min_size = min_size;
a->max_size = max_size;
a->rate = rate;
a->corner = corner;
a->points = points;
nc = a->min_size;
while (nc <= a->max_size)
{
for (i_wtype = 0; i_wtype < 2; i_wtype++)
{
i_dataset = 2 * (int)log2(nc / a->min_size) + i_wtype;
a->dataset[i_dataset] = model_bandpass(nc, -a->corner, +a->corner, a->rate, i_wtype, a->points);
}
nc *= 2;
}
i_corner_offset = (int)round(a->corner / a->rate * a->points);
a->i_lower_corner = (a->points / 2) - i_corner_offset;
a->i_upper_corner = (a->points / 2) + i_corner_offset;
pbfcu[a->id] = a;
return 0;
}
PORT
void destroy_bfcu(int id)
{
BFCU a = pbfcu[id];
int nc = a->min_size;
while (nc <= a->max_size)
{
int i_dataset = 2 * (int)log2(nc / a->min_size) + 0;
_aligned_free(a->dataset[i_dataset]);
_aligned_free(a->dataset[i_dataset + 1]);
nc *= 2;
}
_aligned_free(a);
}
PORT
void getFilterCorners(int id, int* lower_index, int* upper_index)
{
// stores the index of the lower corner and the index of the upper corner at the pointers given.
BFCU a = pbfcu[id];
*lower_index = a->i_lower_corner;
*upper_index = a->i_upper_corner;
}
PORT
void getFilterCurve(int id, int size, int w_type, int index_low, int index_high, double* segment)
{
// size = filter_size
// w_type = window_type (0 -> bh4; 1->bh7)
// index_low = lower index of the segment you want (range is 0 through points-1)
// index_high = upper index of the segment you want (range is 0 through points-1)
// segment = pointer to location where you want the result stored
BFCU a = pbfcu[id];
int i_dataset = 2 * (int)log2(size / a->min_size) + w_type;
memcpy(segment, &a->dataset[i_dataset][index_low], (index_high - index_low + 1) * sizeof(double));
}
void test_bfcu()
{
create_bfcu(0, 1024, 16384, 48000.0, 1000.0, 16384);
int lower_corner, upper_corner;
getFilterCorners(0, &lower_corner, &upper_corner);
double* segment = (double*)malloc0(1025 * sizeof(double));
getFilterCurve(0, 4096, 1, upper_corner - 512, upper_corner + 512, segment);
print_bandpass_response("response", 1025, segment);
_aligned_free(segment);
destroy_bfcu(0);
}