mirror of
https://github.com/f4exb/sdrangel.git
synced 2024-12-23 10:05:46 -05:00
1252 lines
37 KiB
C++
1252 lines
37 KiB
C++
/*---------------------------------------------------------------------------*\
|
|
|
|
FILE........: fsk.c
|
|
AUTHOR......: Brady O'Brien
|
|
DATE CREATED: 7 January 2016
|
|
|
|
C Implementation of 2/4FSK modulator/demodulator, based on octave/fsk_horus.m
|
|
|
|
\*---------------------------------------------------------------------------*/
|
|
|
|
/*
|
|
Copyright (C) 2016 David Rowe
|
|
|
|
All rights reserved.
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU Lesser General Public License version 2.1, as
|
|
published by the Free Software Foundation. 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 Lesser General Public License
|
|
along with this program; if not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
/*---------------------------------------------------------------------------*\
|
|
|
|
DEFINES
|
|
|
|
\*---------------------------------------------------------------------------*/
|
|
|
|
/* P oversampling rate constant -- should probably be init-time configurable */
|
|
#define horus_P 8
|
|
|
|
/* Define this to enable EbNodB estimate */
|
|
/* This needs square roots, may take more cpu time than it's worth */
|
|
#define EST_EBNO
|
|
|
|
/* This is a flag for the freq. estimator to use a precomputed/rt computed hann window table
|
|
On platforms with slow cosf, this will produce a substantial speedup at the cost of a small
|
|
amount of memory
|
|
*/
|
|
#define USE_HANN_TABLE
|
|
|
|
/* This flag turns on run-time hann table generation. If USE_HANN_TABLE is unset,
|
|
this flag has no effect. If USE_HANN_TABLE is set and this flag is set, the
|
|
hann table will be allocated and generated when fsk_init or fsk_init_hbr is
|
|
called. If this flag is not set, a hann function table of size fsk->Ndft MUST
|
|
be provided. On small platforms, this can be used with a precomputed table to
|
|
save memory at the cost of flash space.
|
|
*/
|
|
#define GENERATE_HANN_TABLE_RUNTIME
|
|
|
|
/* Turn off table generation if on cortex M4 to save memory */
|
|
#ifdef CORTEX_M4
|
|
#undef USE_HANN_TABLE
|
|
#endif
|
|
|
|
/*---------------------------------------------------------------------------*\
|
|
|
|
INCLUDES
|
|
|
|
\*---------------------------------------------------------------------------*/
|
|
|
|
#include <assert.h>
|
|
#include <stdlib.h>
|
|
#include <stdint.h>
|
|
#include <math.h>
|
|
|
|
#include "fsk.h"
|
|
#include "comp_prim.h"
|
|
#include "kiss_fftr.h"
|
|
#include "modem_probe.h"
|
|
|
|
namespace FreeDV
|
|
{
|
|
|
|
/*---------------------------------------------------------------------------*\
|
|
|
|
FUNCTIONS
|
|
|
|
\*---------------------------------------------------------------------------*/
|
|
|
|
static void stats_init(struct FSK *fsk);
|
|
|
|
#ifdef USE_HANN_TABLE
|
|
/*
|
|
This is used by fsk_create and fsk_create_hbr to generate a hann function
|
|
table
|
|
*/
|
|
static void fsk_generate_hann_table(struct FSK* fsk){
|
|
int Ndft = fsk->Ndft;
|
|
int i;
|
|
|
|
/* Set up complex oscilator to calculate hann function */
|
|
COMP dphi = comp_exp_j((2*M_PI)/((float)Ndft-1));
|
|
COMP rphi = {.5,0};
|
|
|
|
rphi = cmult(cconj(dphi),rphi);
|
|
|
|
for (i = 0; i < Ndft; i++)
|
|
{
|
|
rphi = cmult(dphi,rphi);
|
|
float hannc = .5-rphi.real;
|
|
//float hann = .5-(.5*cosf((2*M_PI*(float)(i))/((float)Ndft-1)));
|
|
fsk->hann_table[i] = hannc;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
|
|
|
|
/*---------------------------------------------------------------------------*\
|
|
|
|
FUNCTION....: fsk_create_hbr
|
|
AUTHOR......: Brady O'Brien
|
|
DATE CREATED: 11 February 2016
|
|
|
|
Create and initialize an instance of the FSK modem. Returns a pointer
|
|
to the modem state/config struct. One modem config struct may be used
|
|
for both mod and demod. returns NULL on failure.
|
|
|
|
\*---------------------------------------------------------------------------*/
|
|
|
|
struct FSK * fsk_create_hbr(int Fs, int Rs,int P,int M, int tx_f1, int tx_fs)
|
|
{
|
|
struct FSK *fsk;
|
|
int i;
|
|
int memold;
|
|
int Ndft = 0;
|
|
/* Number of symbols in a processing frame */
|
|
int nsyms = 48;
|
|
/* Check configuration validity */
|
|
assert(Fs > 0 );
|
|
assert(Rs > 0 );
|
|
assert(tx_f1 > 0);
|
|
assert(tx_fs > 0);
|
|
assert(P > 0);
|
|
/* Ts (Fs/Rs) must be an integer */
|
|
assert( (Fs%Rs) == 0 );
|
|
/* Ts/P (Fs/Rs/P) must be an integer */
|
|
assert( ((Fs/Rs)%P) == 0 );
|
|
assert( M==2 || M==4);
|
|
|
|
fsk = (struct FSK*) malloc(sizeof(struct FSK));
|
|
if(fsk == NULL) return NULL;
|
|
|
|
|
|
/* Set constant config parameters */
|
|
fsk->Fs = Fs;
|
|
fsk->Rs = Rs;
|
|
fsk->Ts = Fs/Rs;
|
|
fsk->burst_mode = 0;
|
|
fsk->N = fsk->Ts*nsyms;
|
|
fsk->P = P;
|
|
fsk->Nsym = nsyms;
|
|
fsk->Nmem = fsk->N+(2*fsk->Ts);
|
|
fsk->f1_tx = tx_f1;
|
|
fsk->fs_tx = tx_fs;
|
|
fsk->nin = fsk->N;
|
|
fsk->mode = M==2 ? MODE_2FSK : MODE_4FSK;
|
|
fsk->Nbits = M==2 ? fsk->Nsym : fsk->Nsym*2;
|
|
|
|
/* Find smallest 2^N value that fits Fs for efficient FFT */
|
|
/* It would probably be better to use KISS-FFt's routine here */
|
|
for(i=1; i; i<<=1)
|
|
if((fsk->N)&i)
|
|
Ndft = i;
|
|
|
|
fsk->Ndft = Ndft;
|
|
|
|
fsk->est_min = Rs/4;
|
|
if(fsk->est_min<0) fsk->est_min = 0;
|
|
|
|
fsk->est_max = (Fs/2)-Rs/4;
|
|
|
|
fsk->est_space = Rs-(Rs/5);
|
|
|
|
/* Set up rx state */
|
|
|
|
for( i=0; i<M; i++)
|
|
fsk->phi_c[i] = comp_exp_j(0);
|
|
|
|
memold = (4*fsk->Ts);
|
|
|
|
fsk->nstash = memold;
|
|
fsk->samp_old = (COMP*) malloc(sizeof(COMP)*memold);
|
|
if(fsk->samp_old == NULL){
|
|
free(fsk);
|
|
return NULL;
|
|
}
|
|
|
|
for(i=0;i<memold;i++) {
|
|
fsk->samp_old[i].real = 0;
|
|
fsk->samp_old[i].imag = 0;
|
|
}
|
|
|
|
fsk->fft_cfg = kiss_fft_alloc(fsk->Ndft,0,NULL,NULL);
|
|
if(fsk->fft_cfg == NULL){
|
|
free(fsk->samp_old);
|
|
free(fsk);
|
|
return NULL;
|
|
}
|
|
|
|
fsk->fft_est = (float*)malloc(sizeof(float)*fsk->Ndft/2);
|
|
if(fsk->fft_est == NULL){
|
|
free(fsk->samp_old);
|
|
free(fsk->fft_cfg);
|
|
free(fsk);
|
|
return NULL;
|
|
}
|
|
|
|
#ifdef USE_HANN_TABLE
|
|
#ifdef GENERATE_HANN_TABLE_RUNTIME
|
|
fsk->hann_table = (float*)malloc(sizeof(float)*fsk->Ndft);
|
|
if(fsk->hann_table == NULL){
|
|
free(fsk->fft_est);
|
|
free(fsk->samp_old);
|
|
free(fsk->fft_cfg);
|
|
free(fsk);
|
|
return NULL;
|
|
}
|
|
fsk_generate_hann_table(fsk);
|
|
#else
|
|
fsk->hann_table = NULL;
|
|
#endif
|
|
#endif
|
|
|
|
for(i=0;i<fsk->Ndft/2;i++)fsk->fft_est[i] = 0;
|
|
|
|
fsk->norm_rx_timing = 0;
|
|
|
|
/* Set up tx state */
|
|
fsk->tx_phase_c = comp_exp_j(0);
|
|
|
|
/* Set up demod stats */
|
|
fsk->EbNodB = 0;
|
|
|
|
for( i=0; i<M; i++)
|
|
fsk->f_est[i] = 0;
|
|
|
|
fsk->ppm = 0;
|
|
|
|
fsk->stats = (struct MODEM_STATS*)malloc(sizeof(struct MODEM_STATS));
|
|
if(fsk->stats == NULL){
|
|
free(fsk->fft_est);
|
|
free(fsk->samp_old);
|
|
free(fsk->fft_cfg);
|
|
free(fsk);
|
|
return NULL;
|
|
}
|
|
stats_init(fsk);
|
|
fsk->normalise_eye = 1;
|
|
|
|
return fsk;
|
|
}
|
|
|
|
|
|
#define HORUS_MIN 800
|
|
#define HORUS_MAX 2500
|
|
#define HORUS_MIN_SPACING 100
|
|
|
|
/*---------------------------------------------------------------------------*\
|
|
|
|
FUNCTION....: fsk_create
|
|
AUTHOR......: Brady O'Brien
|
|
DATE CREATED: 7 January 2016
|
|
|
|
Create and initialize an instance of the FSK modem. Returns a pointer
|
|
to the modem state/config struct. One modem config struct may be used
|
|
for both mod and demod. returns NULL on failure.
|
|
|
|
\*---------------------------------------------------------------------------*/
|
|
|
|
struct FSK * fsk_create(int Fs, int Rs,int M, int tx_f1, int tx_fs)
|
|
{
|
|
struct FSK *fsk;
|
|
int i;
|
|
int Ndft = 0;
|
|
int memold;
|
|
|
|
/* Check configuration validity */
|
|
assert(Fs > 0 );
|
|
assert(Rs > 0 );
|
|
assert(tx_f1 > 0);
|
|
assert(tx_fs > 0);
|
|
assert(horus_P > 0);
|
|
/* Ts (Fs/Rs) must be an integer */
|
|
assert( (Fs%Rs) == 0 );
|
|
/* Ts/P (Fs/Rs/P) must be an integer */
|
|
assert( ((Fs/Rs)%horus_P) == 0 );
|
|
assert( M==2 || M==4);
|
|
|
|
fsk = (struct FSK*) malloc(sizeof(struct FSK));
|
|
if(fsk == NULL) return NULL;
|
|
|
|
Ndft = 1024;
|
|
|
|
/* Set constant config parameters */
|
|
fsk->Fs = Fs;
|
|
fsk->Rs = Rs;
|
|
fsk->Ts = Fs/Rs;
|
|
fsk->N = Fs;
|
|
fsk->burst_mode = 0;
|
|
fsk->P = horus_P;
|
|
fsk->Nsym = fsk->N/fsk->Ts;
|
|
fsk->Ndft = Ndft;
|
|
fsk->Nmem = fsk->N+(2*fsk->Ts);
|
|
fsk->f1_tx = tx_f1;
|
|
fsk->fs_tx = tx_fs;
|
|
fsk->nin = fsk->N;
|
|
fsk->mode = M==2 ? MODE_2FSK : MODE_4FSK;
|
|
fsk->Nbits = M==2 ? fsk->Nsym : fsk->Nsym*2;
|
|
fsk->est_min = HORUS_MIN;
|
|
fsk->est_max = HORUS_MAX;
|
|
fsk->est_space = HORUS_MIN_SPACING;
|
|
|
|
/* Set up rx state */
|
|
for( i=0; i<M; i++)
|
|
fsk->phi_c[i] = comp_exp_j(0);
|
|
|
|
memold = (4*fsk->Ts);
|
|
|
|
fsk->nstash = memold;
|
|
fsk->samp_old = (COMP*) malloc(sizeof(COMP)*memold);
|
|
if(fsk->samp_old == NULL){
|
|
free(fsk);
|
|
return NULL;
|
|
}
|
|
|
|
for(i=0;i<memold;i++) {
|
|
fsk->samp_old[i].real = 0.0;
|
|
fsk->samp_old[i].imag = 0.0;
|
|
}
|
|
|
|
fsk->fft_cfg = kiss_fft_alloc(Ndft,0,NULL,NULL);
|
|
if(fsk->fft_cfg == NULL){
|
|
free(fsk->samp_old);
|
|
free(fsk);
|
|
return NULL;
|
|
}
|
|
|
|
fsk->fft_est = (float*)malloc(sizeof(float)*fsk->Ndft/2);
|
|
if(fsk->fft_est == NULL){
|
|
free(fsk->samp_old);
|
|
free(fsk->fft_cfg);
|
|
free(fsk);
|
|
return NULL;
|
|
}
|
|
|
|
#ifdef USE_HANN_TABLE
|
|
#ifdef GENERATE_HANN_TABLE_RUNTIME
|
|
fsk->hann_table = (float*)malloc(sizeof(float)*fsk->Ndft);
|
|
if(fsk->hann_table == NULL){
|
|
free(fsk->fft_est);
|
|
free(fsk->samp_old);
|
|
free(fsk->fft_cfg);
|
|
free(fsk);
|
|
return NULL;
|
|
}
|
|
fsk_generate_hann_table(fsk);
|
|
#else
|
|
fsk->hann_table = NULL;
|
|
#endif
|
|
#endif
|
|
|
|
for(i=0;i<Ndft/2;i++)fsk->fft_est[i] = 0;
|
|
|
|
fsk->norm_rx_timing = 0;
|
|
|
|
/* Set up tx state */
|
|
fsk->tx_phase_c = comp_exp_j(0);
|
|
|
|
/* Set up demod stats */
|
|
fsk->EbNodB = 0;
|
|
|
|
for( i=0; i<M; i++)
|
|
fsk->f_est[i] = 0;
|
|
|
|
fsk->ppm = 0;
|
|
|
|
fsk->stats = (struct MODEM_STATS*)malloc(sizeof(struct MODEM_STATS));
|
|
|
|
if(fsk->stats == NULL){
|
|
free(fsk->fft_est);
|
|
free(fsk->samp_old);
|
|
free(fsk->fft_cfg);
|
|
free(fsk);
|
|
return NULL;
|
|
}
|
|
stats_init(fsk);
|
|
fsk->normalise_eye = 1;
|
|
|
|
return fsk;
|
|
}
|
|
|
|
/* make sure stats have known values in case monitoring process reads stats before they are set */
|
|
|
|
static void stats_init(struct FSK *fsk) {
|
|
/* Take a sample for the eye diagrams */
|
|
int i,j,m;
|
|
int P = fsk->P;
|
|
int M = fsk->mode;
|
|
|
|
/* due to oversample rate P, we have too many samples for eye
|
|
trace. So lets output a decimated version */
|
|
|
|
/* asserts below as we found some problems over-running eye matrix */
|
|
|
|
/* TODO: refactor eye tracing code here and in fsk_demod */
|
|
|
|
int neyesamp_dec = ceil(((float)P*2)/MODEM_STATS_EYE_IND_MAX);
|
|
int neyesamp = (P*2)/neyesamp_dec;
|
|
assert(neyesamp <= MODEM_STATS_EYE_IND_MAX);
|
|
fsk->stats->neyesamp = neyesamp;
|
|
|
|
int eye_traces = MODEM_STATS_ET_MAX/M;
|
|
|
|
fsk->stats->neyetr = fsk->mode*eye_traces;
|
|
for(i=0; i<eye_traces; i++) {
|
|
for (m=0; m<M; m++){
|
|
for(j=0; j<neyesamp; j++) {
|
|
assert((i*M+m) < MODEM_STATS_ET_MAX);
|
|
fsk->stats->rx_eye[i*M+m][j] = 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
fsk->stats->rx_timing = fsk->stats->snr_est = 0;
|
|
|
|
}
|
|
|
|
|
|
void fsk_set_nsym(struct FSK *fsk,int nsyms){
|
|
assert(nsyms>0);
|
|
int Ndft,i;
|
|
Ndft = 0;
|
|
|
|
/* Set constant config parameters */
|
|
fsk->N = fsk->Ts*nsyms;
|
|
fsk->Nsym = nsyms;
|
|
fsk->Nmem = fsk->N+(2*fsk->Ts);
|
|
fsk->nin = fsk->N;
|
|
fsk->Nbits = fsk->mode==2 ? fsk->Nsym : fsk->Nsym*2;
|
|
|
|
/* Find smallest 2^N value that fits Fs for efficient FFT */
|
|
/* It would probably be better to use KISS-FFt's routine here */
|
|
for(i=1; i; i<<=1)
|
|
if((fsk->N)&i)
|
|
Ndft = i;
|
|
|
|
fsk->Ndft = Ndft;
|
|
|
|
free(fsk->fft_cfg);
|
|
free(fsk->fft_est);
|
|
|
|
fsk->fft_cfg = kiss_fft_alloc(Ndft,0,NULL,NULL);
|
|
fsk->fft_est = (float*)malloc(sizeof(float)*fsk->Ndft/2);
|
|
|
|
for(i=0;i<Ndft/2;i++)fsk->fft_est[i] = 0;
|
|
|
|
}
|
|
|
|
/* Set the FSK modem into burst demod mode */
|
|
|
|
void fsk_enable_burst_mode(struct FSK *fsk,int nsyms){
|
|
fsk_set_nsym(fsk,nsyms);
|
|
fsk->nin = fsk->N;
|
|
fsk->burst_mode = 1;
|
|
}
|
|
|
|
void fsk_clear_estimators(struct FSK *fsk){
|
|
int i;
|
|
/* Clear freq estimator state */
|
|
for(i=0; i < (fsk->Ndft/2); i++){
|
|
fsk->fft_est[i] = 0;
|
|
}
|
|
/* Reset timing diff correction */
|
|
fsk->nin = fsk->N;
|
|
}
|
|
|
|
uint32_t fsk_nin(struct FSK *fsk){
|
|
return (uint32_t)fsk->nin;
|
|
}
|
|
|
|
void fsk_destroy(struct FSK *fsk){
|
|
free(fsk->fft_cfg);
|
|
free(fsk->samp_old);
|
|
free(fsk->stats);
|
|
free(fsk);
|
|
}
|
|
|
|
void fsk_get_demod_stats(struct FSK *fsk, struct MODEM_STATS *stats){
|
|
/* copy from internal stats, note we can't overwrite stats completely
|
|
as it has other states rqd by caller, also we want a consistent
|
|
interface across modem types for the freedv_api.
|
|
*/
|
|
|
|
stats->clock_offset = fsk->stats->clock_offset;
|
|
stats->snr_est = fsk->stats->snr_est; // TODO: make this SNR not Eb/No
|
|
stats->rx_timing = fsk->stats->rx_timing;
|
|
stats->foff = fsk->stats->foff;
|
|
|
|
stats->neyesamp = fsk->stats->neyesamp;
|
|
stats->neyetr = fsk->stats->neyetr;
|
|
memcpy(stats->rx_eye, fsk->stats->rx_eye, sizeof(stats->rx_eye));
|
|
memcpy(stats->f_est, fsk->stats->f_est, fsk->mode*sizeof(float));
|
|
|
|
/* these fields not used for FSK so set to something sensible */
|
|
|
|
stats->sync = 0;
|
|
stats->nr = fsk->stats->nr;
|
|
stats->Nc = fsk->stats->Nc;
|
|
}
|
|
|
|
/*
|
|
* Set the minimum and maximum frequencies at which the freq. estimator can find tones
|
|
*/
|
|
void fsk_set_est_limits(struct FSK *fsk,int est_min, int est_max){
|
|
|
|
fsk->est_min = est_min;
|
|
if(fsk->est_min<0) fsk->est_min = 0;
|
|
|
|
fsk->est_max = est_max;
|
|
}
|
|
|
|
/*
|
|
* Internal function to estimate the frequencies of the two tones within a block of samples.
|
|
* This is split off because it is fairly complicated, needs a bunch of memory, and probably
|
|
* takes more cycles than the rest of the demod.
|
|
* Parameters:
|
|
* fsk - FSK struct from demod containing FSK config
|
|
* fsk_in - block of samples in this demod cycles, must be nin long
|
|
* freqs - Array for the estimated frequencies
|
|
* M - number of frequency peaks to find
|
|
*/
|
|
void fsk_demod_freq_est(struct FSK *fsk, COMP fsk_in[],float *freqs,int M){
|
|
int Ndft = fsk->Ndft;
|
|
int Fs = fsk->Fs;
|
|
int nin = fsk->nin;
|
|
int i,j;
|
|
float hann;
|
|
float max;
|
|
float tc;
|
|
int imax;
|
|
kiss_fft_cfg fft_cfg = fsk->fft_cfg;
|
|
int *freqi = new int[M];
|
|
int f_min,f_max,f_zero;
|
|
|
|
/* Array to do complex FFT from using kiss_fft */
|
|
kiss_fft_cpx *fftin = (kiss_fft_cpx*)malloc(sizeof(kiss_fft_cpx)*Ndft);
|
|
kiss_fft_cpx *fftout = (kiss_fft_cpx*)malloc(sizeof(kiss_fft_cpx)*Ndft);
|
|
|
|
#ifndef USE_HANN_TABLE
|
|
COMP dphi = comp_exp_j((2*M_PI)/((float)Ndft-1));
|
|
COMP rphi = {.5,0};
|
|
rphi = cmult(cconj(dphi),rphi);
|
|
#endif
|
|
|
|
f_min = (fsk->est_min*Ndft)/Fs;
|
|
f_max = (fsk->est_max*Ndft)/Fs;
|
|
f_zero = (fsk->est_space*Ndft)/Fs;
|
|
|
|
/* scale averaging time constant based on number of samples */
|
|
tc = 0.95*Ndft/Fs;
|
|
|
|
int samps;
|
|
int fft_samps;
|
|
int fft_loops = nin / Ndft;
|
|
|
|
for (j = 0; j < fft_loops; j++)
|
|
{
|
|
/* 48000 sample rate (for example) will have a spare */
|
|
/* 896 samples besides the 46 "Ndft" samples, so adjust */
|
|
|
|
samps = (nin - ((j + 1) * Ndft));
|
|
fft_samps = (samps >= Ndft) ? Ndft : samps;
|
|
|
|
/* Copy FSK buffer into reals of FFT buffer and apply a hann window */
|
|
for(i=0; i<fft_samps; i++){
|
|
#ifdef USE_HANN_TABLE
|
|
hann = fsk->hann_table[i];
|
|
#else
|
|
//hann = 1-cosf((2*M_PI*(float)(i))/((float)fft_samps-1));
|
|
rphi = cmult(dphi,rphi);
|
|
hann = .5-rphi.real;
|
|
#endif
|
|
fftin[i].r = hann*fsk_in[i+Ndft*j].real;
|
|
fftin[i].i = hann*fsk_in[i+Ndft*j].imag;
|
|
}
|
|
|
|
/* Zero out the remaining slots on spare samples */
|
|
for(; i<Ndft;i++){
|
|
fftin[i].r = 0;
|
|
fftin[i].i = 0;
|
|
}
|
|
|
|
/* Do the FFT */
|
|
kiss_fft(fft_cfg,fftin,fftout);
|
|
|
|
/* Find the magnitude^2 of each freq slot and stash away in the real
|
|
* value, so this only has to be done once. Since we're only comparing
|
|
* these values and only need the mag of 2 points, we don't need to do
|
|
* a sqrt to each value */
|
|
for(i=0; i<Ndft/2; i++){
|
|
fftout[i].r = (fftout[i].r*fftout[i].r) + (fftout[i].i*fftout[i].i) ;
|
|
}
|
|
|
|
/* Zero out the minimum and maximum ends */
|
|
for(i=0; i<f_min; i++){
|
|
fftout[i].r = 0;
|
|
}
|
|
for(i=f_max-1; i<Ndft/2; i++){
|
|
fftout[i].r = 0;
|
|
}
|
|
/* Mix back in with the previous fft block */
|
|
/* Copy new fft est into imag of fftout for frequency divination below */
|
|
for(i=0; i<Ndft/2; i++){
|
|
fsk->fft_est[i] = (fsk->fft_est[i]*(1-tc)) + (sqrtf(fftout[i].r)*tc);
|
|
fftout[i].i = fsk->fft_est[i];
|
|
}
|
|
}
|
|
|
|
modem_probe_samp_f("t_fft_est", fsk->fft_est, Ndft/2);
|
|
|
|
max = 0;
|
|
/* Find the M frequency peaks here */
|
|
for(i=0; i<M; i++){
|
|
imax = 0;
|
|
max = 0;
|
|
for(j=0;j<Ndft/2;j++){
|
|
if(fftout[j].i > max){
|
|
max = fftout[j].i;
|
|
imax = j;
|
|
}
|
|
}
|
|
/* Blank out FMax +/-Fspace/2 */
|
|
f_min = imax - f_zero;
|
|
f_min = f_min < 0 ? 0 : f_min;
|
|
f_max = imax + f_zero;
|
|
f_max = f_max > Ndft ? Ndft : f_max;
|
|
for(j=f_min; j<f_max; j++)
|
|
fftout[j].i = 0;
|
|
|
|
/* Stick the freq index on the list */
|
|
freqi[i] = imax;
|
|
}
|
|
|
|
/* Gnome sort the freq list */
|
|
/* My favorite sort of sort*/
|
|
i = 1;
|
|
while(i<M){
|
|
if(freqi[i] >= freqi[i-1]) i++;
|
|
else{
|
|
j = freqi[i];
|
|
freqi[i] = freqi[i-1];
|
|
freqi[i-1] = j;
|
|
if(i>1) i--;
|
|
}
|
|
}
|
|
|
|
/* Convert freqs from indices to frequencies */
|
|
for(i=0; i<M; i++){
|
|
freqs[i] = (float)(freqi[i])*((float)Fs/(float)Ndft);
|
|
}
|
|
|
|
delete[] freqi;
|
|
free(fftin);
|
|
free(fftout);
|
|
}
|
|
|
|
void fsk2_demod(struct FSK *fsk, uint8_t rx_bits[], float rx_sd[], COMP fsk_in[]){
|
|
int N = fsk->N;
|
|
int Ts = fsk->Ts;
|
|
int Rs = fsk->Rs;
|
|
int Fs = fsk->Fs;
|
|
int nsym = fsk->Nsym;
|
|
int nin = fsk->nin;
|
|
int P = fsk->P;
|
|
int Nmem = fsk->Nmem;
|
|
int M = fsk->mode;
|
|
int i, j, m, dc_i, cbuf_i;
|
|
float ft1;
|
|
int nstash = fsk->nstash;
|
|
|
|
COMP* *f_int = new COMP*[M]; /* Filtered and downsampled symbol tones */
|
|
COMP *t = new COMP[M]; /* complex number temps */
|
|
COMP t_c; /* another complex temp */
|
|
COMP *phi_c = new COMP[M];
|
|
COMP phi_ft;
|
|
int nold = Nmem-nin;
|
|
|
|
COMP *dphi = new COMP[M];
|
|
COMP dphift;
|
|
float rx_timing,norm_rx_timing,old_norm_rx_timing,d_norm_rx_timing,appm;
|
|
int using_old_samps;
|
|
|
|
COMP* sample_src;
|
|
COMP* f_intbuf_m;
|
|
|
|
float *f_est = new float[M];
|
|
float fc_avg, fc_tx;
|
|
float meanebno,stdebno,eye_max;
|
|
int neyesamp,neyeoffset;
|
|
|
|
#ifdef MODEMPROBE_ENABLE
|
|
char mp_name_tmp[20]; /* Temporary string for modem probe trace names */
|
|
#endif
|
|
|
|
//for(size_t jj = 0; jj<nin; jj++){
|
|
// fprintf(stderr,"%f,j%f,",fsk_in[jj].real,fsk_in[jj].imag);
|
|
//}
|
|
|
|
/* Load up demod phases from struct */
|
|
for(m = 0; m < M; m++) {
|
|
phi_c[m] = fsk->phi_c[m];
|
|
}
|
|
|
|
/* Estimate tone frequencies */
|
|
fsk_demod_freq_est(fsk,fsk_in,f_est,M);
|
|
modem_probe_samp_f("t_f_est",f_est,M);
|
|
|
|
|
|
/* Allocate circular buffer for integration */
|
|
f_intbuf_m = (COMP*) malloc(sizeof(COMP)*Ts);
|
|
|
|
/* allocate memory for the integrated samples */
|
|
for( m=0; m<M; m++){
|
|
/* allocate memory for the integrated samples */
|
|
/* Note: This must be kept after fsk_demod_freq_est for memory usage reasons */
|
|
f_int[m] = (COMP*) malloc(sizeof(COMP)*(nsym+1)*P);
|
|
}
|
|
|
|
/* If this is the first run, we won't have any valid f_est */
|
|
/* TODO: add first_run flag to FSK to make negative freqs possible */
|
|
if(fsk->f_est[0]<1){
|
|
for( m=0; m<M; m++)
|
|
fsk->f_est[m] = f_est[m];
|
|
}
|
|
|
|
/* Initalize downmixers for each symbol tone */
|
|
for( m=0; m<M; m++){
|
|
/* Back the stored phase off to account for re-integraton of old samples */
|
|
dphi[m] = comp_exp_j(-2*(Nmem-nin-(Ts/P))*M_PI*((fsk->f_est[m])/(float)(Fs)));
|
|
phi_c[m] = cmult(dphi[m],phi_c[m]);
|
|
//fprintf(stderr,"F%d = %f",m,fsk->f_est[m]);
|
|
|
|
/* Figure out how much to nudge each sample downmixer for every sample */
|
|
dphi[m] = comp_exp_j(2*M_PI*((fsk->f_est[m])/(float)(Fs)));
|
|
}
|
|
|
|
/* Integrate and downsample for symbol tones */
|
|
for(m=0; m<M; m++){
|
|
/* Copy buffer pointers in to avoid second buffer indirection */
|
|
float f_est_m = f_est[m];
|
|
COMP* f_int_m = &(f_int[m][0]);
|
|
COMP dphi_m = dphi[m];
|
|
|
|
dc_i = 0;
|
|
cbuf_i = 0;
|
|
sample_src = &(fsk->samp_old[nstash-nold]);
|
|
using_old_samps = 1;
|
|
|
|
/* Pre-fill integration buffer */
|
|
for(dc_i=0; dc_i<Ts-(Ts/P); dc_i++){
|
|
/* Switch sample source to new samples when we run out of old ones */
|
|
if(dc_i>=nold && using_old_samps){
|
|
sample_src = &fsk_in[0];
|
|
dc_i = 0;
|
|
using_old_samps = 0;
|
|
|
|
/* Recalculate delta-phi after switching to new sample source */
|
|
phi_c[m] = comp_normalize(phi_c[m]);
|
|
dphi_m = comp_exp_j(2*M_PI*((f_est_m)/(float)(Fs)));
|
|
}
|
|
/* Downconvert and place into integration buffer */
|
|
f_intbuf_m[dc_i]=cmult(sample_src[dc_i],cconj(phi_c[m]));
|
|
|
|
#ifdef MODEMPROBE_ENABLE
|
|
snprintf(mp_name_tmp,19,"t_f%zd_dc",m+1);
|
|
modem_probe_samp_c(mp_name_tmp,&f_intbuf_m[dc_i],1);
|
|
#endif
|
|
/* Spin downconversion phases */
|
|
phi_c[m] = cmult(phi_c[m],dphi_m);
|
|
}
|
|
cbuf_i = dc_i;
|
|
|
|
/* Integrate over Ts at offsets of Ts/P */
|
|
for(i=0; i<(nsym+1)*P; i++){
|
|
/* Downconvert and Place Ts/P samples in the integration buffers */
|
|
for(j=0; j<(Ts/P); j++,dc_i++){
|
|
/* Switch sample source to new samples when we run out of old ones */
|
|
if(dc_i>=nold && using_old_samps){
|
|
sample_src = &fsk_in[0];
|
|
dc_i = 0;
|
|
using_old_samps = 0;
|
|
|
|
/* Recalculate delta-phi after switching to new sample source */
|
|
phi_c[m] = comp_normalize(phi_c[m]);
|
|
dphi_m = comp_exp_j(2*M_PI*((f_est_m)/(float)(Fs)));
|
|
}
|
|
/* Downconvert and place into integration buffer */
|
|
f_intbuf_m[cbuf_i+j]=cmult(sample_src[dc_i],cconj(phi_c[m]));
|
|
|
|
#ifdef MODEMPROBE_ENABLE
|
|
snprintf(mp_name_tmp,19,"t_f%zd_dc",m+1);
|
|
modem_probe_samp_c(mp_name_tmp,&f_intbuf_m[cbuf_i+j],1);
|
|
#endif
|
|
/* Spin downconversion phases */
|
|
phi_c[m] = cmult(phi_c[m],dphi_m);
|
|
|
|
}
|
|
|
|
/* Dump internal samples */
|
|
cbuf_i += Ts/P;
|
|
if(cbuf_i>=Ts) cbuf_i = 0;
|
|
|
|
/* Integrate over the integration buffers, save samples */
|
|
float it_r = 0;
|
|
float it_i = 0;
|
|
for(j=0; j<Ts; j++){
|
|
it_r += f_intbuf_m[j].real;
|
|
it_i += f_intbuf_m[j].imag;
|
|
}
|
|
f_int_m[i].real = it_r;
|
|
f_int_m[i].imag = it_i;
|
|
}
|
|
}
|
|
|
|
/* Save phases back into FSK struct */
|
|
for(m=0; m<M; m++){
|
|
fsk->phi_c[m] = phi_c[m];
|
|
fsk->f_est[m] = f_est[m];
|
|
}
|
|
|
|
/* Stash samples away in the old sample buffer for the next round of bit getting */
|
|
memcpy((void*)&(fsk->samp_old[0]),(void*)&(fsk_in[nin-nstash]),sizeof(COMP)*nstash);
|
|
|
|
/* Fine Timing Estimation */
|
|
/* Apply magic nonlinearity to f1_int and f2_int, shift down to 0,
|
|
* extract angle */
|
|
|
|
/* Figure out how much to spin the oscillator to extract magic spectral line */
|
|
dphift = comp_exp_j(2*M_PI*((float)(Rs)/(float)(P*Rs)));
|
|
phi_ft.real = 1;
|
|
phi_ft.imag = 0;
|
|
t_c=comp0();
|
|
for(i=0; i<(nsym+1)*P; i++){
|
|
/* Get abs^2 of fx_int[i], and add 'em */
|
|
ft1 = 0;
|
|
for( m=0; m<M; m++){
|
|
ft1 += (f_int[m][i].real*f_int[m][i].real) + (f_int[m][i].imag*f_int[m][i].imag);
|
|
}
|
|
|
|
/* Down shift and accumulate magic line */
|
|
t_c = cadd(t_c,fcmult(ft1,phi_ft));
|
|
|
|
/* Spin the oscillator for the magic line shift */
|
|
phi_ft = cmult(phi_ft,dphift);
|
|
}
|
|
|
|
/* Check for NaNs in the fine timing estimate, return if found */
|
|
/* otherwise segfaults happen */
|
|
if( isnan(t_c.real) || isnan(t_c.imag)){
|
|
return;
|
|
}
|
|
|
|
/* Get the magic angle */
|
|
norm_rx_timing = atan2f(t_c.imag,t_c.real)/(2*M_PI);
|
|
rx_timing = norm_rx_timing*(float)P;
|
|
|
|
old_norm_rx_timing = fsk->norm_rx_timing;
|
|
fsk->norm_rx_timing = norm_rx_timing;
|
|
|
|
/* Estimate sample clock offset */
|
|
d_norm_rx_timing = norm_rx_timing - old_norm_rx_timing;
|
|
|
|
/* Filter out big jumps in due to nin change */
|
|
if(fabsf(d_norm_rx_timing) < .2){
|
|
appm = 1e6*d_norm_rx_timing/(float)nsym;
|
|
fsk->ppm = .9*fsk->ppm + .1*appm;
|
|
}
|
|
|
|
/* Figure out how many samples are needed the next modem cycle */
|
|
/* Unless we're in burst mode */
|
|
if(!fsk->burst_mode){
|
|
if(norm_rx_timing > 0.25)
|
|
fsk->nin = N+Ts/2;
|
|
else if(norm_rx_timing < -0.25)
|
|
fsk->nin = N-Ts/2;
|
|
else
|
|
fsk->nin = N;
|
|
}
|
|
|
|
modem_probe_samp_f("t_norm_rx_timing",&(norm_rx_timing),1);
|
|
modem_probe_samp_i("t_nin",&(fsk->nin),1);
|
|
|
|
/* Re-sample the integrators with linear interpolation magic */
|
|
int low_sample = (int)floorf(rx_timing);
|
|
float fract = rx_timing - (float)low_sample;
|
|
int high_sample = (int)ceilf(rx_timing);
|
|
|
|
/* Vars for finding the max-of-4 for each bit */
|
|
float *tmax = new float[M];
|
|
|
|
#ifdef EST_EBNO
|
|
meanebno = 0;
|
|
stdebno = 0;
|
|
#endif
|
|
|
|
/* FINALLY, THE BITS */
|
|
/* also, resample fx_int */
|
|
for(i = 0; i < nsym; i++)
|
|
{
|
|
int st = (i+1)*P;
|
|
for( m=0; m<M; m++){
|
|
t[m] = fcmult(1-fract,f_int[m][st+ low_sample]);
|
|
t[m] = cadd(t[m],fcmult( fract,f_int[m][st+high_sample]));
|
|
/* Figure mag^2 of each resampled fx_int */
|
|
tmax[m] = (t[m].real*t[m].real) + (t[m].imag*t[m].imag);
|
|
}
|
|
|
|
float max = tmax[0]; /* Maximum for figuring correct symbol */
|
|
float min = tmax[0];
|
|
int sym = 0; /* Index of maximum */
|
|
for( m=0; m<M; m++){
|
|
if(tmax[m]>max){
|
|
max = tmax[m];
|
|
sym = m;
|
|
}
|
|
if(tmax[m]<min){
|
|
min = tmax[m];
|
|
}
|
|
}
|
|
|
|
/* Get the actual bit */
|
|
if(rx_bits != NULL){
|
|
/* Get bits for 2FSK and 4FSK */
|
|
/* TODO: Replace this with something more generic maybe */
|
|
if(M==2){
|
|
rx_bits[i] = sym==1; /* 2FSK. 1 bit per symbol */
|
|
}else if(M==4){
|
|
rx_bits[(i*2)+1] = (sym&0x1); /* 4FSK. 2 bits per symbol */
|
|
rx_bits[(i*2) ] = (sym&0x2)>>1;
|
|
}
|
|
}
|
|
|
|
/* Produce soft decision symbols */
|
|
if(rx_sd != NULL){
|
|
/* Convert symbols from max^2 into max */
|
|
for( m=0; m<M; m++)
|
|
tmax[m] = sqrtf(tmax[m]);
|
|
|
|
if(M==2){
|
|
rx_sd[i] = tmax[0] - tmax[1];
|
|
}else if(M==4){
|
|
/* TODO: Find a soft-decision mode that works for 4FSK */
|
|
min = sqrtf(min);
|
|
rx_sd[(i*2)+1] = - tmax[0] ; /* Bits=00 */
|
|
rx_sd[(i*2) ] = - tmax[0] ;
|
|
rx_sd[(i*2)+1]+= tmax[1] ; /* Bits=01 */
|
|
rx_sd[(i*2) ]+= - tmax[1] ;
|
|
rx_sd[(i*2)+1]+= - tmax[2] ; /* Bits=10 */
|
|
rx_sd[(i*2) ]+= tmax[2] ;
|
|
rx_sd[(i*2)+1]+= tmax[3] ; /* Bits=11 */
|
|
rx_sd[(i*2) ]+= tmax[3] ;
|
|
}
|
|
}
|
|
/* Accumulate resampled int magnitude for EbNodB estimation */
|
|
/* Standard deviation is calculated by algorithm devised by crafty soviets */
|
|
#ifdef EST_EBNO
|
|
/* Accumulate the square of the sampled value */
|
|
ft1 = max;
|
|
stdebno += ft1;
|
|
|
|
/* Figure the abs value of the max tone */
|
|
meanebno += sqrtf(ft1);
|
|
#endif
|
|
/* Soft output goes here */
|
|
}
|
|
|
|
delete[] tmax;
|
|
|
|
#ifdef EST_EBNO
|
|
|
|
/* Calculate mean for EbNodB estimation */
|
|
meanebno = meanebno/(float)nsym;
|
|
|
|
/* Calculate the std. dev for EbNodB estimate */
|
|
stdebno = (stdebno/(float)nsym) - (meanebno*meanebno);
|
|
/* trap any negative numbers to avoid NANs flowing through */
|
|
if (stdebno > 0.0) {
|
|
stdebno = sqrt(stdebno);
|
|
} else {
|
|
stdebno = 0.0;
|
|
}
|
|
|
|
fsk->EbNodB = -6+(20*log10f((1e-6+meanebno)/(1e-6+stdebno)));
|
|
#else
|
|
fsk->EbNodB = 1;
|
|
#endif
|
|
|
|
/* Write some statistics to the stats struct */
|
|
|
|
/* Save clock offset in ppm */
|
|
fsk->stats->clock_offset = fsk->ppm;
|
|
|
|
/* Calculate and save SNR from EbNodB estimate */
|
|
|
|
fsk->stats->snr_est = .5*fsk->stats->snr_est + .5*fsk->EbNodB;//+ 10*log10f(((float)Rs)/((float)Rs*M));
|
|
|
|
/* Save rx timing */
|
|
fsk->stats->rx_timing = (float)rx_timing;
|
|
|
|
/* Estimate and save frequency offset */
|
|
fc_avg = (f_est[0]+f_est[1])/2;
|
|
fc_tx = (fsk->f1_tx+fsk->f1_tx+fsk->fs_tx)/2;
|
|
fsk->stats->foff = fc_tx-fc_avg;
|
|
|
|
/* Take a sample for the eye diagrams ---------------------------------- */
|
|
|
|
/* due to oversample rate P, we have too many samples for eye
|
|
trace. So lets output a decimated version. We use 2P
|
|
as we want two symbols worth of samples in trace */
|
|
|
|
int neyesamp_dec = ceil(((float)P*2)/MODEM_STATS_EYE_IND_MAX);
|
|
neyesamp = (P*2)/neyesamp_dec;
|
|
assert(neyesamp <= MODEM_STATS_EYE_IND_MAX);
|
|
fsk->stats->neyesamp = neyesamp;
|
|
|
|
#ifdef I_DONT_UNDERSTAND
|
|
neyeoffset = high_sample+1+(P*28); /* WTF this line? Where does "28" come from ? */
|
|
#endif /* ifdef-ed out as I am afraid it will index out of memory as P changes */
|
|
neyeoffset = high_sample+1;
|
|
|
|
int eye_traces = MODEM_STATS_ET_MAX/M;
|
|
int ind;
|
|
|
|
fsk->stats->neyetr = fsk->mode*eye_traces;
|
|
for( i=0; i<eye_traces; i++){
|
|
for ( m=0; m<M; m++){
|
|
for(j=0; j<neyesamp; j++) {
|
|
/*
|
|
2*P*i...........: advance two symbols for next trace
|
|
neyeoffset......: centre trace on ideal timing offset, peak eye opening
|
|
j*neweyesamp_dec: For 2*P>MODEM_STATS_EYE_IND_MAX advance through integrated
|
|
samples newamp_dec at a time so we dont overflow rx_eye[][]
|
|
*/
|
|
ind = 2*P*i + neyeoffset + j*neyesamp_dec;
|
|
assert((i*M+m) < MODEM_STATS_ET_MAX);
|
|
assert(ind < (nsym+1)*P);
|
|
fsk->stats->rx_eye[i*M+m][j] = cabsolute(f_int[m][ind]);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (fsk->normalise_eye) {
|
|
eye_max = 0;
|
|
/* Normalize eye to +/- 1 */
|
|
for(i=0; i<M*eye_traces; i++)
|
|
for(j=0; j<neyesamp; j++)
|
|
if(fabsf(fsk->stats->rx_eye[i][j])>eye_max)
|
|
eye_max = fabsf(fsk->stats->rx_eye[i][j]);
|
|
|
|
for(i=0; i<M*eye_traces; i++)
|
|
for(j=0; j<neyesamp; j++)
|
|
fsk->stats->rx_eye[i][j] = fsk->stats->rx_eye[i][j]/eye_max;
|
|
}
|
|
|
|
fsk->stats->nr = 0;
|
|
fsk->stats->Nc = 0;
|
|
|
|
for(i=0; i<M; i++) {
|
|
fsk->stats->f_est[i] = f_est[i];
|
|
}
|
|
|
|
/* Dump some internal samples */
|
|
modem_probe_samp_f("t_EbNodB",&(fsk->EbNodB),1);
|
|
modem_probe_samp_f("t_ppm",&(fsk->ppm),1);
|
|
modem_probe_samp_f("t_rx_timing",&(rx_timing),1);
|
|
|
|
#ifdef MODEMPROBE_ENABLE
|
|
for( m=0; m<M; m++){
|
|
snprintf(mp_name_tmp,19,"t_f%zd_int",m+1);
|
|
modem_probe_samp_c(mp_name_tmp,f_int[m],(nsym+1)*P);
|
|
snprintf(mp_name_tmp,19,"t_f%zd",m+1);
|
|
modem_probe_samp_f(mp_name_tmp,&f_est[m],1);
|
|
}
|
|
#endif
|
|
|
|
for( m=0; m<M; m++){
|
|
free(f_int[m]);
|
|
}
|
|
free(f_intbuf_m);
|
|
|
|
delete[] f_est;
|
|
delete[] dphi;
|
|
delete[] phi_c;
|
|
delete[] t;
|
|
delete[] f_int;
|
|
}
|
|
|
|
void fsk_demod(struct FSK *fsk, uint8_t rx_bits[], COMP fsk_in[]){
|
|
fsk2_demod(fsk,rx_bits,NULL,fsk_in);
|
|
}
|
|
|
|
void fsk_demod_sd(struct FSK *fsk, float rx_sd[], COMP fsk_in[]){
|
|
fsk2_demod(fsk,NULL,rx_sd,fsk_in);
|
|
}
|
|
|
|
void fsk_mod(struct FSK *fsk,float fsk_out[],uint8_t tx_bits[]){
|
|
COMP tx_phase_c = fsk->tx_phase_c; /* Current complex TX phase */
|
|
int f1_tx = fsk->f1_tx; /* '0' frequency */
|
|
int fs_tx = fsk->fs_tx; /* space between frequencies */
|
|
int Ts = fsk->Ts; /* samples-per-symbol */
|
|
int Fs = fsk->Fs; /* sample freq */
|
|
int M = fsk->mode;
|
|
COMP *dosc_f = new COMP[M]; /* phase shift per sample */
|
|
COMP dph; /* phase shift of current bit */
|
|
int i, j, m, bit_i, sym;
|
|
|
|
/* Init the per sample phase shift complex numbers */
|
|
for( m=0; m<M; m++){
|
|
dosc_f[m] = comp_exp_j(2*M_PI*((float)(f1_tx+(fs_tx*m))/(float)(Fs)));
|
|
}
|
|
|
|
bit_i = 0;
|
|
for( i=0; i<fsk->Nsym; i++){
|
|
sym = 0;
|
|
/* Pack the symbol number from the bit stream */
|
|
for( m=M; m>>=1; ){
|
|
uint8_t bit = tx_bits[bit_i];
|
|
bit = (bit==1)?1:0;
|
|
sym = (sym<<1)|bit;
|
|
bit_i++;
|
|
}
|
|
/* Look up symbol phase shift */
|
|
dph = dosc_f[sym];
|
|
/* Spin the oscillator for a symbol period */
|
|
for(j=0; j<Ts; j++){
|
|
tx_phase_c = cmult(tx_phase_c,dph);
|
|
fsk_out[i*Ts+j] = 2*tx_phase_c.real;
|
|
}
|
|
|
|
}
|
|
|
|
/* Normalize TX phase to prevent drift */
|
|
tx_phase_c = comp_normalize(tx_phase_c);
|
|
|
|
/* save TX phase */
|
|
fsk->tx_phase_c = tx_phase_c;
|
|
|
|
delete[] dosc_f;
|
|
}
|
|
|
|
void fsk_mod_c(struct FSK *fsk,COMP fsk_out[],uint8_t tx_bits[]){
|
|
COMP tx_phase_c = fsk->tx_phase_c; /* Current complex TX phase */
|
|
int f1_tx = fsk->f1_tx; /* '0' frequency */
|
|
int fs_tx = fsk->fs_tx; /* space between frequencies */
|
|
int Ts = fsk->Ts; /* samples-per-symbol */
|
|
int Fs = fsk->Fs; /* sample freq */
|
|
int M = fsk->mode;
|
|
COMP *dosc_f = new COMP[M]; /* phase shift per sample */
|
|
COMP dph; /* phase shift of current bit */
|
|
int i, j, bit_i, sym;
|
|
int m;
|
|
|
|
/* Init the per sample phase shift complex numbers */
|
|
for( m=0; m<M; m++){
|
|
dosc_f[m] = comp_exp_j(2*M_PI*((float)(f1_tx+(fs_tx*m))/(float)(Fs)));
|
|
}
|
|
|
|
bit_i = 0;
|
|
for( i=0; i<fsk->Nsym; i++){
|
|
sym = 0;
|
|
/* Pack the symbol number from the bit stream */
|
|
for( m=M; m>>=1; ){
|
|
uint8_t bit = tx_bits[bit_i];
|
|
bit = (bit==1)?1:0;
|
|
sym = (sym<<1)|bit;
|
|
bit_i++;
|
|
}
|
|
/* Look up symbol phase shift */
|
|
dph = dosc_f[sym];
|
|
/* Spin the oscillator for a symbol period */
|
|
for(j=0; j<Ts; j++){
|
|
tx_phase_c = cmult(tx_phase_c,dph);
|
|
fsk_out[i*Ts+j] = fcmult(2,tx_phase_c);
|
|
}
|
|
}
|
|
|
|
/* Normalize TX phase to prevent drift */
|
|
tx_phase_c = comp_normalize(tx_phase_c);
|
|
|
|
/* save TX phase */
|
|
fsk->tx_phase_c = tx_phase_c;
|
|
|
|
delete[] dosc_f;
|
|
}
|
|
|
|
|
|
/* Modulator that assume an external VCO. The output is a voltage
|
|
that changes for each symbol */
|
|
|
|
void fsk_mod_ext_vco(struct FSK *fsk, float vco_out[], uint8_t tx_bits[]) {
|
|
int f1_tx = fsk->f1_tx; /* '0' frequency */
|
|
int fs_tx = fsk->fs_tx; /* space between frequencies */
|
|
int Ts = fsk->Ts; /* samples-per-symbol */
|
|
int M = fsk->mode;
|
|
int i, j, m, sym, bit_i;
|
|
|
|
bit_i = 0;
|
|
for(i=0; i<fsk->Nsym; i++) {
|
|
/* generate the symbol number from the bit stream,
|
|
e.g. 0,1 for 2FSK, 0,1,2,3 for 4FSK */
|
|
|
|
sym = 0;
|
|
|
|
/* unpack the symbol number from the bit stream */
|
|
|
|
for( m=M; m>>=1; ){
|
|
uint8_t bit = tx_bits[bit_i];
|
|
bit = (bit==1)?1:0;
|
|
sym = (sym<<1)|bit;
|
|
bit_i++;
|
|
}
|
|
|
|
/*
|
|
Map 'sym' to VCO frequency
|
|
Note: drive is inverted, a higher tone drives VCO voltage lower
|
|
*/
|
|
|
|
//fprintf(stderr, "i: %d sym: %d freq: %f\n", i, sym, f1_tx + fs_tx*(float)sym);
|
|
for(j=0; j<Ts; j++) {
|
|
vco_out[i*Ts+j] = f1_tx + fs_tx*(float)sym;
|
|
}
|
|
}
|
|
}
|
|
|
|
void fsk_stats_normalise_eye(struct FSK *fsk, int normalise_enable) {
|
|
fsk->normalise_eye = normalise_enable;
|
|
}
|
|
|
|
} // freeDV
|
|
|
|
|
|
|
|
|
|
|