Skip to content

Commit 0661e23

Browse files
committed
Issue 271 - DCD dropping at wrong time.
1 parent 74a5c34 commit 0661e23

12 files changed

+219
-162
lines changed

src/atest.c

+43-2
Original file line numberDiff line numberDiff line change
@@ -82,7 +82,7 @@
8282
#include "ptt.h"
8383
#include "dtime_now.h"
8484
#include "fx25.h"
85-
85+
#include "hdlc_rec.h"
8686

8787

8888
#if 0 /* Typical but not flexible enough. */
@@ -188,6 +188,9 @@ static int J_opt = 0; /* 2400 bps PSK compatible MFJ-2400 and maybe others. */
188188
static int h_opt = 0; // Hexadecimal display of received packet.
189189
static char P_opt[16] = ""; // Demodulator profiles.
190190
static int d_x_opt = 1; // FX.25 debug.
191+
static int d_o_opt = 0; // "-d o" option for DCD output control. */
192+
static int dcd_count = 0;
193+
static int dcd_missing_errors = 0;
191194

192195

193196
int main (int argc, char *argv[])
@@ -382,6 +385,7 @@ int main (int argc, char *argv[])
382385
for (char *p=optarg; *p!='\0'; p++) {
383386
switch (*p) {
384387
case 'x': d_x_opt++; break; // FX.25
388+
case 'o': d_o_opt++; break; // DCD output control
385389
default: break;
386390
}
387391
}
@@ -686,6 +690,10 @@ int main (int argc, char *argv[])
686690
elapsed = dtime_now() - start_time;
687691

688692
dw_printf ("%d packets decoded in %.3f seconds. %.1f x realtime\n", packets_decoded_total, elapsed, total_filetime/elapsed);
693+
if (d_o_opt) {
694+
dw_printf ("DCD count = %d\n", dcd_count);
695+
dw_printf ("DCD missing erors = %d\n", dcd_missing_errors);
696+
}
689697

690698
if (error_if_less_than != -1 && packets_decoded_total < error_if_less_than) {
691699
text_color_set(DW_COLOR_ERROR);
@@ -744,6 +752,7 @@ void dlq_rec_frame (int chan, int subchan, int slice, packet_t pp, alevel_t alev
744752
char alevel_text[AX25_ALEVEL_TO_TEXT_SIZE];
745753

746754
packets_decoded_one++;
755+
if ( ! hdlc_rec_data_detect_any(chan)) dcd_missing_errors++;
747756

748757
ax25_format_addrs (pp, stemp);
749758

@@ -777,7 +786,7 @@ void dlq_rec_frame (int chan, int subchan, int slice, packet_t pp, alevel_t alev
777786
int min = (int)(sec / 60.);
778787
sec -= min * 60;
779788

780-
dw_printf ("%d:%07.4f ", min, sec);
789+
dw_printf ("%d:%06.3f ", min, sec);
781790

782791
if (h != AX25_SOURCE) {
783792
dw_printf ("Digipeater ");
@@ -876,6 +885,38 @@ void dlq_rec_frame (int chan, int subchan, int slice, packet_t pp, alevel_t alev
876885

877886
void ptt_set (int ot, int chan, int ptt_signal)
878887
{
888+
// Should only get here for DCD output control.
889+
static double dcd_start_time[MAX_CHANS];
890+
891+
if (d_o_opt) {
892+
double t = (double)sample_number / my_audio_config.adev[0].samples_per_sec;
893+
double sec1, sec2;
894+
int min1, min2;
895+
896+
text_color_set(DW_COLOR_INFO);
897+
898+
if (ptt_signal) {
899+
//sec1 = t;
900+
//min1 = (int)(sec1 / 60.);
901+
//sec1 -= min1 * 60;
902+
//dw_printf ("DCD[%d] = ON %d:%06.3f\n", chan, min1, sec1);
903+
dcd_count++;
904+
dcd_start_time[chan] = t;
905+
}
906+
else {
907+
//dw_printf ("DCD[%d] = off %d:%06.3f %3.0f\n", chan, min, sec, (t - dcd_start_time[chan]) * 1000.);
908+
909+
sec1 = dcd_start_time[chan];
910+
min1 = (int)(sec1 / 60.);
911+
sec1 -= min1 * 60;
912+
913+
sec2 = t;
914+
min2 = (int)(sec2 / 60.);
915+
sec2 -= min2 * 60;
916+
917+
dw_printf ("DCD[%d] %d:%06.3f - %d:%06.3f = %3.0f\n", chan, min1, sec1, min2, sec2, (t - dcd_start_time[chan]) * 1000.);
918+
}
919+
}
879920
return;
880921
}
881922

src/demod_9600.c

+17-6
Original file line numberDiff line numberDiff line change
@@ -44,14 +44,24 @@
4444
#include <assert.h>
4545
#include <ctype.h>
4646

47+
// Fine tuning for different demodulator types.
48+
49+
#define DCD_THRESH_ON 32 // Hysteresis: Can miss 0 out of 32 for detecting lock.
50+
// This is best for actual on-the-air signals.
51+
// Still too many brief false matches.
52+
#define DCD_THRESH_OFF 8 // Might want a little more fine tuning.
53+
#define DCD_GOOD_WIDTH 1024 // No more than 1024!!!
54+
#include "fsk_demod_state.h" // Values above override defaults.
55+
4756
#include "tune.h"
48-
#include "fsk_demod_state.h"
4957
#include "hdlc_rec.h"
5058
#include "demod_9600.h"
5159
#include "textcolor.h"
5260
#include "dsp.h"
5361

5462

63+
64+
5565
static float slice_point[MAX_SUBCHANS];
5666

5767

@@ -407,7 +417,7 @@ void demod_9600_process_sample (int chan, int sam, struct demodulator_state_s *D
407417
if (chan == 0) {
408418

409419
if (1) {
410-
//if (hdlc_rec_gathering (chan, subchan, slice)) {
420+
//if (D->slicer[slice].data_detect) {
411421
char fname[30];
412422
int slice = 0;
413423

@@ -516,6 +526,7 @@ inline static void nudge_pll (int chan, int subchan, int slice, float demod_out_
516526
/* Overflow. Was large positive, wrapped around, now large negative. */
517527

518528
hdlc_rec_bit (chan, subchan, slice, demod_out_f > 0, D->modem_type == MODEM_SCRAMBLE, D->slicer[slice].lfsr);
529+
pll_dcd_each_symbol2 (D, chan, subchan, slice);
519530
}
520531

521532
/*
@@ -526,11 +537,11 @@ inline static void nudge_pll (int chan, int subchan, int slice, float demod_out_
526537

527538
// Note: Test for this demodulator, not overall for channel.
528539

529-
float target = 0;
540+
pll_dcd_signal_transition2 (D, slice, D->slicer[slice].data_clock_pll);
530541

531-
target = D->pll_step_per_sample * demod_out_f / (demod_out_f - D->slicer[slice].prev_demod_out_f);
542+
float target = D->pll_step_per_sample * demod_out_f / (demod_out_f - D->slicer[slice].prev_demod_out_f);
532543

533-
if (hdlc_rec_gathering (chan, subchan, slice)) {
544+
if (D->slicer[slice].data_detect) {
534545
D->slicer[slice].data_clock_pll = (int)(D->slicer[slice].data_clock_pll * D->pll_locked_inertia + target * (1.0f - D->pll_locked_inertia) );
535546
}
536547
else {
@@ -542,7 +553,7 @@ inline static void nudge_pll (int chan, int subchan, int slice, float demod_out_
542553
#if DEBUG5
543554

544555
//if (chan == 0) {
545-
if (hdlc_rec_gathering (chan,subchan,slice)) {
556+
if (D->slicer[slice].data_detect) {
546557

547558
char fname[30];
548559

src/demod_afsk.c

+7-5
Original file line numberDiff line numberDiff line change
@@ -762,7 +762,7 @@ void demod_afsk_process_sample (int chan, int subchan, int sam, struct demodulat
762762
#if DEBUG4
763763

764764
if (chan == 0) {
765-
if (hdlc_rec_gathering (chan, subchan)) {
765+
if (D->slicer[slice].data_detect) {
766766
char fname[30];
767767

768768

@@ -839,16 +839,18 @@ inline static void nudge_pll (int chan, int subchan, int slice, int demod_data,
839839

840840
if (D->slicer[slice].data_clock_pll < 0 && D->slicer[slice].prev_d_c_pll > 0) {
841841

842-
/* Overflow. */
842+
/* Overflow - this is where we sample. */
843843
hdlc_rec_bit (chan, subchan, slice, demod_data, 0, -1);
844+
pll_dcd_each_symbol2 (D, chan, subchan, slice);
844845
}
845846

846-
// Even if we used alternative method to extract the data bit,
847-
// we still use the low pass output for the PLL.
847+
// Transitions nudge the DPLL phase toward the incoming signal.
848848

849849
if (demod_data != D->slicer[slice].prev_demod_data) {
850850

851-
if (hdlc_rec_gathering (chan, subchan, slice)) {
851+
pll_dcd_signal_transition2 (D, slice, D->slicer[slice].data_clock_pll);
852+
853+
if (D->slicer[slice].data_detect) {
852854
D->slicer[slice].data_clock_pll = (int)(D->slicer[slice].data_clock_pll * D->pll_locked_inertia);
853855
}
854856
else {

src/demod_psk.c

+14-2
Original file line numberDiff line numberDiff line change
@@ -86,16 +86,25 @@
8686
#include <assert.h>
8787
#include <ctype.h>
8888

89+
// Fine tuning for different demodulator types.
90+
91+
#define DCD_THRESH_ON 30 // Hysteresis: Can miss 2 out of 32 for detecting lock.
92+
#define DCD_THRESH_OFF 6 // Might want a little more fine tuning.
93+
#define DCD_GOOD_WIDTH 512
94+
#include "fsk_demod_state.h" // Values above override defaults.
8995

9096
#include "audio.h"
9197
#include "tune.h"
92-
#include "fsk_demod_state.h"
9398
#include "fsk_gen_filter.h"
9499
#include "hdlc_rec.h"
95100
#include "textcolor.h"
96101
#include "demod_psk.h"
97102
#include "dsp.h"
98103

104+
105+
106+
107+
99108
static const int phase_to_gray_v26[4] = {0, 1, 3, 2};
100109
static const int phase_to_gray_v27[8] = {1, 0, 2, 3, 7, 6, 4, 5};
101110

@@ -800,6 +809,7 @@ static void nudge_pll (int chan, int subchan, int slice, int demod_bits, struct
800809
hdlc_rec_bit (chan, subchan, slice, (gray >> 1) & 1, 0, bit_quality[1]);
801810
hdlc_rec_bit (chan, subchan, slice, gray & 1, 0, bit_quality[0]);
802811
}
812+
pll_dcd_each_symbol2 (D, chan, subchan, slice);
803813
}
804814

805815
/*
@@ -813,7 +823,9 @@ static void nudge_pll (int chan, int subchan, int slice, int demod_bits, struct
813823

814824
if (demod_bits != D->slicer[slice].prev_demod_data) {
815825

816-
if (hdlc_rec_gathering (chan, subchan, slice)) {
826+
pll_dcd_signal_transition2 (D, slice, D->slicer[slice].data_clock_pll);
827+
828+
if (D->slicer[slice].data_detect) {
817829
D->slicer[slice].data_clock_pll = (int)floorf((float)(D->slicer[slice].data_clock_pll) * D->pll_locked_inertia);
818830
}
819831
else {

src/fsk_demod_state.h

+110-1
Original file line numberDiff line numberDiff line change
@@ -226,6 +226,8 @@ struct demodulator_state_s
226226
signed int data_clock_pll; // PLL for data clock recovery.
227227
// It is incremented by pll_step_per_sample
228228
// for each audio sample.
229+
// Must be 32 bits!!!
230+
// So far, this is the case for every compiler used.
229231

230232
signed int prev_d_c_pll; // Previous value of above, before
231233
// incrementing, to detect overflows.
@@ -238,6 +240,18 @@ struct demodulator_state_s
238240

239241
int lfsr; // Descrambler shift register.
240242

243+
// This is for detecting phase lock to incoming signal.
244+
245+
int good_flag; // Set if transition is near where expected,
246+
// i.e. at a good time.
247+
int bad_flag; // Set if transition is not where expected,
248+
// i.e. at a bad time.
249+
unsigned char good_hist; // History of good transitions for past octet.
250+
unsigned char bad_hist; // History of bad transitions for past octet.
251+
unsigned int score; // History of whether good triumphs over bad
252+
// for past 32 symbols.
253+
int data_detect; // True when locked on to signal.
254+
241255
} slicer [MAX_SLICERS]; // Actual number in use is num_slicers.
242256
// Should be in range 1 .. MAX_SLICERS,
243257
/*
@@ -336,5 +350,100 @@ struct demodulator_state_s
336350

337351
};
338352

353+
354+
/*-------------------------------------------------------------------
355+
*
356+
* Name: pll_dcd_signal_transition2
357+
* dcd_each_symbol2
358+
*
359+
* Purpose: New DCD strategy for 1.6.
360+
*
361+
* Inputs: D Pointer to demodulator state.
362+
*
363+
* chan Radio channel: 0 to MAX_CHANS - 1
364+
*
365+
* subchan Which of multiple demodulators: 0 to MAX_SUBCHANS - 1
366+
*
367+
* slice Slicer number: 0 to MAX_SLICERS - 1.
368+
*
369+
* dpll_phase Signed 32 bit counter for DPLL phase.
370+
* Wraparound is where data is sampled.
371+
* Ideally transitions would occur close to 0.
372+
*
373+
* Output: D->slicer[slice].data_detect - true when PLL is locked to incoming signal.
374+
*
375+
* Description: From the beginning, DCD was based on finding several flag octets
376+
* in a row and dropping when eight bits with no transitions.
377+
* It was less than ideal but we limped along with it all these years.
378+
* This fell apart when FX.25 came along and a couple of the
379+
* correlation tags have eight "1" bits in a row.
380+
*
381+
* Our new strategy is to keep a running score of how well demodulator
382+
* output transitions match to where expected.
383+
*
384+
*--------------------------------------------------------------------*/
385+
386+
#include "hdlc_rec.h" // for dcd_change
387+
388+
// These are good for 1200 bps AFSK.
389+
// Might want to override for other modems.
390+
391+
#ifndef DCD_THRESH_ON
392+
#define DCD_THRESH_ON 30 // Hysteresis: Can miss 2 out of 32 for detecting lock.
393+
// 31 is best for TNC Test CD. 30 almost as good.
394+
// 30 better for 1200 regression test.
395+
#endif
396+
397+
#ifndef DCD_THRESH_OFF
398+
#define DCD_THRESH_OFF 6 // Might want a little more fine tuning.
399+
#endif
400+
401+
#ifndef DCD_GOOD_WIDTH
402+
#define DCD_GOOD_WIDTH 512 // No more than 1024!!!
403+
#endif
404+
405+
__attribute__((always_inline))
406+
inline static void pll_dcd_signal_transition2 (struct demodulator_state_s *D, int slice, int dpll_phase)
407+
{
408+
if (dpll_phase > - DCD_GOOD_WIDTH * 1024 * 1024 && dpll_phase < DCD_GOOD_WIDTH * 1024 * 1024) {
409+
D->slicer[slice].good_flag = 1;
410+
}
411+
else {
412+
D->slicer[slice].bad_flag = 1;
413+
}
414+
}
415+
416+
__attribute__((always_inline))
417+
inline static void pll_dcd_each_symbol2 (struct demodulator_state_s *D, int chan, int subchan, int slice)
418+
{
419+
D->slicer[slice].good_hist <<= 1;
420+
D->slicer[slice].good_hist |= D->slicer[slice].good_flag;
421+
D->slicer[slice].good_flag = 0;
422+
423+
D->slicer[slice].bad_hist <<= 1;
424+
D->slicer[slice].bad_hist |= D->slicer[slice].bad_flag;
425+
D->slicer[slice].bad_flag = 0;
426+
427+
D->slicer[slice].score <<= 1;
428+
// 2 is to detect 'flag' patterns with 2 transitions per octet.
429+
D->slicer[slice].score |= (signed)__builtin_popcount(D->slicer[slice].good_hist)
430+
- (signed)__builtin_popcount(D->slicer[slice].bad_hist) >= 2;
431+
432+
int s = __builtin_popcount(D->slicer[slice].score);
433+
if (s >= DCD_THRESH_ON) {
434+
if (D->slicer[slice].data_detect == 0) {
435+
D->slicer[slice].data_detect = 1;
436+
dcd_change (chan, subchan, slice, D->slicer[slice].data_detect);
437+
}
438+
}
439+
else if (s <= DCD_THRESH_OFF) {
440+
if (D->slicer[slice].data_detect != 0) {
441+
D->slicer[slice].data_detect = 0;
442+
dcd_change (chan, subchan, slice, D->slicer[slice].data_detect);
443+
}
444+
}
445+
}
446+
447+
339448
#define FSK_DEMOD_STATE_H 1
340-
#endif
449+
#endif

0 commit comments

Comments
 (0)