2024-06-16 05:31:13 -04:00
|
|
|
/* amd.c
|
|
|
|
|
|
|
|
This file is part of a program that implements a Software-Defined Radio.
|
|
|
|
|
|
|
|
Copyright (C) 2012, 2013 Warren Pratt, NR0V
|
|
|
|
Copyright (C) 2024 Edouard Griffiths, F4EXB Adapted to SDRangel
|
|
|
|
|
|
|
|
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
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <cmath>
|
|
|
|
|
|
|
|
#include "comm.hpp"
|
|
|
|
#include "amd.hpp"
|
|
|
|
#include "anf.hpp"
|
|
|
|
#include "emnr.hpp"
|
|
|
|
#include "anr.hpp"
|
2024-07-02 18:52:16 -04:00
|
|
|
#include "snba.hpp"
|
2024-06-16 05:31:13 -04:00
|
|
|
|
|
|
|
namespace WDSP {
|
|
|
|
|
2024-07-26 11:52:34 -04:00
|
|
|
AMD::AMD
|
2024-07-05 21:25:41 -04:00
|
|
|
(
|
2024-07-26 11:52:34 -04:00
|
|
|
int _run,
|
|
|
|
int _buff_size,
|
|
|
|
float *_in_buff,
|
|
|
|
float *_out_buff,
|
|
|
|
int _mode,
|
|
|
|
int _levelfade,
|
|
|
|
int _sbmode,
|
|
|
|
int _sample_rate,
|
|
|
|
double _fmin,
|
|
|
|
double _fmax,
|
|
|
|
double _zeta,
|
|
|
|
double _omegaN,
|
|
|
|
double _tauR,
|
|
|
|
double _tauI
|
2024-07-05 21:25:41 -04:00
|
|
|
)
|
2024-06-16 05:31:13 -04:00
|
|
|
{
|
2024-07-26 11:52:34 -04:00
|
|
|
run = _run;
|
|
|
|
buff_size = _buff_size;
|
|
|
|
in_buff = _in_buff;
|
|
|
|
out_buff = _out_buff;
|
|
|
|
mode = _mode;
|
|
|
|
levelfade = _levelfade;
|
|
|
|
sbmode = _sbmode;
|
|
|
|
sample_rate = (double) _sample_rate;
|
|
|
|
fmin = _fmin;
|
|
|
|
fmax = _fmax;
|
|
|
|
zeta = _zeta;
|
|
|
|
omegaN = _omegaN;
|
|
|
|
tauR = _tauR;
|
|
|
|
tauI = _tauI;
|
|
|
|
init();
|
2024-06-16 05:31:13 -04:00
|
|
|
}
|
|
|
|
|
2024-07-26 11:52:34 -04:00
|
|
|
void AMD::init()
|
2024-06-16 05:31:13 -04:00
|
|
|
{
|
|
|
|
//pll
|
2024-07-26 11:52:34 -04:00
|
|
|
omega_min = 2 * M_PI * fmin / sample_rate;
|
|
|
|
omega_max = 2 * M_PI * fmax / sample_rate;
|
|
|
|
g1 = 1.0 - std::exp(-2.0 * omegaN * zeta / sample_rate);
|
|
|
|
g2 = -g1 + 2.0 * (1 - exp(-omegaN * zeta / sample_rate) * cos(omegaN / sample_rate * sqrt(1.0 - zeta * zeta)));
|
|
|
|
phs = 0.0;
|
|
|
|
fil_out = 0.0;
|
|
|
|
omega = 0.0;
|
2024-06-16 05:31:13 -04:00
|
|
|
|
|
|
|
//fade leveler
|
2024-07-26 11:52:34 -04:00
|
|
|
dc = 0.0;
|
|
|
|
dc_insert = 0.0;
|
|
|
|
mtauR = exp(-1.0 / (sample_rate * tauR));
|
|
|
|
onem_mtauR = 1.0 - mtauR;
|
|
|
|
mtauI = exp(-1.0 / (sample_rate * tauI));
|
|
|
|
onem_mtauI = 1.0 - mtauI;
|
2024-06-16 05:31:13 -04:00
|
|
|
|
|
|
|
//sideband separation
|
2024-07-26 11:52:34 -04:00
|
|
|
c0[0] = -0.328201924180698;
|
|
|
|
c0[1] = -0.744171491539427;
|
|
|
|
c0[2] = -0.923022915444215;
|
|
|
|
c0[3] = -0.978490468768238;
|
|
|
|
c0[4] = -0.994128272402075;
|
|
|
|
c0[5] = -0.998458978159551;
|
|
|
|
c0[6] = -0.999790306259206;
|
|
|
|
|
|
|
|
c1[0] = -0.0991227952747244;
|
|
|
|
c1[1] = -0.565619728761389;
|
|
|
|
c1[2] = -0.857467122550052;
|
|
|
|
c1[3] = -0.959123933111275;
|
|
|
|
c1[4] = -0.988739372718090;
|
|
|
|
c1[5] = -0.996959189310611;
|
|
|
|
c1[6] = -0.999282492800792;
|
2024-06-16 05:31:13 -04:00
|
|
|
}
|
|
|
|
|
2024-07-26 11:52:34 -04:00
|
|
|
void AMD::flush()
|
2024-06-16 05:31:13 -04:00
|
|
|
{
|
2024-07-26 11:52:34 -04:00
|
|
|
dc = 0.0;
|
|
|
|
dc_insert = 0.0;
|
2024-06-16 05:31:13 -04:00
|
|
|
}
|
|
|
|
|
2024-07-26 11:52:34 -04:00
|
|
|
void AMD::execute()
|
2024-06-16 05:31:13 -04:00
|
|
|
{
|
|
|
|
int i;
|
2024-07-05 21:25:41 -04:00
|
|
|
double audio;
|
|
|
|
double vco[2];
|
|
|
|
double corr[2];
|
|
|
|
double det;
|
|
|
|
double del_out;
|
|
|
|
double ai, bi, aq, bq;
|
|
|
|
double ai_ps, bi_ps, aq_ps, bq_ps;
|
2024-06-16 05:31:13 -04:00
|
|
|
int j, k;
|
2024-07-17 20:08:05 -04:00
|
|
|
|
2024-07-26 11:52:34 -04:00
|
|
|
if (run)
|
2024-06-16 05:31:13 -04:00
|
|
|
{
|
2024-07-26 11:52:34 -04:00
|
|
|
switch (mode)
|
2024-06-16 05:31:13 -04:00
|
|
|
{
|
|
|
|
|
|
|
|
case 0: //AM Demodulator
|
|
|
|
{
|
2024-07-26 11:52:34 -04:00
|
|
|
for (i = 0; i < buff_size; i++)
|
2024-06-16 05:31:13 -04:00
|
|
|
{
|
2024-07-26 11:52:34 -04:00
|
|
|
double xr = in_buff[2 * i + 0];
|
|
|
|
double xi = in_buff[2 * i + 1];
|
2024-07-17 20:08:05 -04:00
|
|
|
audio = sqrt(xr*xr + xi*xi);
|
|
|
|
|
2024-07-26 11:52:34 -04:00
|
|
|
if (levelfade)
|
2024-06-16 05:31:13 -04:00
|
|
|
{
|
2024-07-26 11:52:34 -04:00
|
|
|
dc = mtauR * dc + onem_mtauR * audio;
|
|
|
|
dc_insert = mtauI * dc_insert + onem_mtauI * audio;
|
|
|
|
audio += dc_insert - dc;
|
2024-06-16 05:31:13 -04:00
|
|
|
}
|
2024-07-17 20:08:05 -04:00
|
|
|
|
2024-07-26 11:52:34 -04:00
|
|
|
out_buff[2 * i + 0] = audio;
|
|
|
|
out_buff[2 * i + 1] = audio;
|
2024-06-16 05:31:13 -04:00
|
|
|
}
|
2024-07-17 20:08:05 -04:00
|
|
|
|
2024-06-16 05:31:13 -04:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case 1: //Synchronous AM Demodulator with Sideband Separation
|
|
|
|
{
|
2024-07-26 11:52:34 -04:00
|
|
|
for (i = 0; i < buff_size; i++)
|
2024-06-16 05:31:13 -04:00
|
|
|
{
|
2024-07-26 11:52:34 -04:00
|
|
|
vco[0] = cos(phs);
|
|
|
|
vco[1] = sin(phs);
|
2024-06-16 05:31:13 -04:00
|
|
|
|
2024-07-26 11:52:34 -04:00
|
|
|
ai = in_buff[2 * i + 0] * vco[0];
|
|
|
|
bi = in_buff[2 * i + 0] * vco[1];
|
|
|
|
aq = in_buff[2 * i + 1] * vco[0];
|
|
|
|
bq = in_buff[2 * i + 1] * vco[1];
|
2024-06-16 05:31:13 -04:00
|
|
|
|
2024-07-26 11:52:34 -04:00
|
|
|
if (sbmode != 0)
|
2024-06-16 05:31:13 -04:00
|
|
|
{
|
2024-07-26 11:52:34 -04:00
|
|
|
a[0] = dsI;
|
|
|
|
b[0] = bi;
|
|
|
|
c[0] = dsQ;
|
|
|
|
d[0] = aq;
|
|
|
|
dsI = ai;
|
|
|
|
dsQ = bq;
|
2024-06-16 05:31:13 -04:00
|
|
|
|
|
|
|
for (j = 0; j < STAGES; j++)
|
|
|
|
{
|
|
|
|
k = 3 * j;
|
2024-07-26 11:52:34 -04:00
|
|
|
a[k + 3] = c0[j] * (a[k] - a[k + 5]) + a[k + 2];
|
|
|
|
b[k + 3] = c1[j] * (b[k] - b[k + 5]) + b[k + 2];
|
|
|
|
c[k + 3] = c0[j] * (c[k] - c[k + 5]) + c[k + 2];
|
|
|
|
d[k + 3] = c1[j] * (d[k] - d[k + 5]) + d[k + 2];
|
2024-06-16 05:31:13 -04:00
|
|
|
}
|
2024-07-17 20:08:05 -04:00
|
|
|
|
2024-07-26 11:52:34 -04:00
|
|
|
ai_ps = a[OUT_IDX];
|
|
|
|
bi_ps = b[OUT_IDX];
|
|
|
|
bq_ps = c[OUT_IDX];
|
|
|
|
aq_ps = d[OUT_IDX];
|
2024-06-16 05:31:13 -04:00
|
|
|
|
|
|
|
for (j = OUT_IDX + 2; j > 0; j--)
|
|
|
|
{
|
2024-07-26 11:52:34 -04:00
|
|
|
a[j] = a[j - 1];
|
|
|
|
b[j] = b[j - 1];
|
|
|
|
c[j] = c[j - 1];
|
|
|
|
d[j] = d[j - 1];
|
2024-06-16 05:31:13 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
corr[0] = +ai + bq;
|
|
|
|
corr[1] = -bi + aq;
|
|
|
|
|
2024-07-26 11:52:34 -04:00
|
|
|
switch(sbmode)
|
2024-06-16 05:31:13 -04:00
|
|
|
{
|
|
|
|
case 0: //both sidebands
|
|
|
|
{
|
|
|
|
audio = corr[0];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 1: //LSB
|
|
|
|
{
|
|
|
|
audio = (ai_ps - bi_ps) + (aq_ps + bq_ps);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 2: //USB
|
|
|
|
{
|
|
|
|
audio = (ai_ps + bi_ps) - (aq_ps - bq_ps);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-07-26 11:52:34 -04:00
|
|
|
if (levelfade)
|
2024-06-16 05:31:13 -04:00
|
|
|
{
|
2024-07-26 11:52:34 -04:00
|
|
|
dc = mtauR * dc + onem_mtauR * audio;
|
|
|
|
dc_insert = mtauI * dc_insert + onem_mtauI * corr[0];
|
|
|
|
audio += dc_insert - dc;
|
2024-06-16 05:31:13 -04:00
|
|
|
}
|
2024-07-17 20:08:05 -04:00
|
|
|
|
2024-07-26 11:52:34 -04:00
|
|
|
out_buff[2 * i + 0] = audio;
|
|
|
|
out_buff[2 * i + 1] = audio;
|
2024-06-16 05:31:13 -04:00
|
|
|
|
2024-07-17 20:08:05 -04:00
|
|
|
if ((corr[0] == 0.0) && (corr[1] == 0.0))
|
|
|
|
corr[0] = 1.0;
|
|
|
|
|
2024-06-16 05:31:13 -04:00
|
|
|
det = atan2(corr[1], corr[0]);
|
2024-07-26 11:52:34 -04:00
|
|
|
del_out = fil_out;
|
|
|
|
omega += g2 * det;
|
2024-07-17 20:08:05 -04:00
|
|
|
|
2024-07-26 11:52:34 -04:00
|
|
|
if (omega < omega_min)
|
|
|
|
omega = omega_min;
|
2024-07-17 20:08:05 -04:00
|
|
|
|
2024-07-26 11:52:34 -04:00
|
|
|
if (omega > omega_max)
|
|
|
|
omega = omega_max;
|
2024-07-17 20:08:05 -04:00
|
|
|
|
2024-07-26 11:52:34 -04:00
|
|
|
fil_out = g1 * det + omega;
|
|
|
|
phs += del_out;
|
2024-07-17 20:08:05 -04:00
|
|
|
|
2024-07-26 11:52:34 -04:00
|
|
|
while (phs >= 2 * M_PI)
|
|
|
|
phs -= 2 * M_PI;
|
2024-07-17 20:08:05 -04:00
|
|
|
|
2024-07-26 11:52:34 -04:00
|
|
|
while (phs < 0.0)
|
|
|
|
phs += 2 * M_PI;
|
2024-06-16 05:31:13 -04:00
|
|
|
}
|
2024-07-17 20:08:05 -04:00
|
|
|
|
2024-06-16 05:31:13 -04:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2024-07-26 11:52:34 -04:00
|
|
|
else if (in_buff != out_buff)
|
2024-06-16 05:31:13 -04:00
|
|
|
{
|
2024-07-26 11:52:34 -04:00
|
|
|
std::copy (in_buff, in_buff + buff_size * 2, out_buff);
|
2024-06-16 05:31:13 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-07-26 11:52:34 -04:00
|
|
|
void AMD::setBuffers(float* in, float* out)
|
2024-06-16 05:31:13 -04:00
|
|
|
{
|
2024-07-26 11:52:34 -04:00
|
|
|
in_buff = in;
|
|
|
|
out_buff = out;
|
2024-06-16 05:31:13 -04:00
|
|
|
}
|
|
|
|
|
2024-07-26 11:52:34 -04:00
|
|
|
void AMD::setSamplerate(int rate)
|
2024-06-16 05:31:13 -04:00
|
|
|
{
|
2024-07-26 11:52:34 -04:00
|
|
|
sample_rate = rate;
|
|
|
|
init();
|
2024-06-16 05:31:13 -04:00
|
|
|
}
|
|
|
|
|
2024-07-26 11:52:34 -04:00
|
|
|
void AMD::setSize(int size)
|
2024-06-16 05:31:13 -04:00
|
|
|
{
|
2024-07-26 11:52:34 -04:00
|
|
|
buff_size = size;
|
2024-06-16 05:31:13 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/********************************************************************************************************
|
|
|
|
* *
|
2024-07-26 11:52:34 -04:00
|
|
|
* Public Properties *
|
2024-06-16 05:31:13 -04:00
|
|
|
* *
|
|
|
|
********************************************************************************************************/
|
|
|
|
|
2024-07-26 11:52:34 -04:00
|
|
|
void AMD::setSBMode(int _sbmode)
|
2024-06-16 05:31:13 -04:00
|
|
|
{
|
2024-07-26 11:52:34 -04:00
|
|
|
sbmode = _sbmode;
|
2024-06-16 05:31:13 -04:00
|
|
|
}
|
|
|
|
|
2024-07-26 11:52:34 -04:00
|
|
|
void AMD::setFadeLevel(int _levelfade)
|
2024-06-16 05:31:13 -04:00
|
|
|
{
|
2024-07-26 11:52:34 -04:00
|
|
|
levelfade = _levelfade;
|
2024-06-16 05:31:13 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namesoace WDSP
|