Skip to content

Commit 909b703

Browse files
committed
FX.25 enable value 1 now selects appropriate tag for frame length.
1 parent 4b0395a commit 909b703

File tree

3 files changed

+103
-46
lines changed

3 files changed

+103
-46
lines changed

src/fx25_init.c

+73-24
Original file line numberDiff line numberDiff line change
@@ -130,6 +130,9 @@ static const struct correlation_tag_s tags[16] = {
130130
// 12 got many false matches with random noise.
131131
// Even 8 might be too high. We see 2 or 4 bit errors here
132132
// at the point where decoding the block is very improbable.
133+
// After 2 months of continuous operation as a digipeater/iGate,
134+
// no false triggers were observed. So 8 doesn't seem to be too
135+
// high for 1200 bps. No study has been done for 9600 bps.
133136

134137
// Given a 64 bit correlation tag value, find acceptable match in table.
135138
// Return index into table or -1 for no match.
@@ -215,14 +218,14 @@ void fx25_init ( int debug_level )
215218
assert (tags[j].n_block_rs == FX25_BLOCK_SIZE);
216219
}
217220

218-
assert (fx25_pick_mode (1, 239) == 1);
219-
assert (fx25_pick_mode (1, 240) == -1);
221+
assert (fx25_pick_mode (100+1, 239) == 1);
222+
assert (fx25_pick_mode (100+1, 240) == -1);
220223

221-
assert (fx25_pick_mode (5, 223) == 5);
222-
assert (fx25_pick_mode (5, 224) == -1);
224+
assert (fx25_pick_mode (100+5, 223) == 5);
225+
assert (fx25_pick_mode (100+5, 224) == -1);
223226

224-
assert (fx25_pick_mode (9, 191) == 9);
225-
assert (fx25_pick_mode (9, 192) == -1);
227+
assert (fx25_pick_mode (100+9, 191) == 9);
228+
assert (fx25_pick_mode (100+9, 192) == -1);
226229

227230
assert (fx25_pick_mode (16, 32) == 4);
228231
assert (fx25_pick_mode (16, 64) == 3);
@@ -241,6 +244,16 @@ void fx25_init ( int debug_level )
241244
assert (fx25_pick_mode (64, 191) == 9);
242245
assert (fx25_pick_mode (64, 192) == -1);
243246

247+
assert (fx25_pick_mode (1, 32) == 4);
248+
assert (fx25_pick_mode (1, 33) == 3);
249+
assert (fx25_pick_mode (1, 64) == 3);
250+
assert (fx25_pick_mode (1, 65) == 6);
251+
assert (fx25_pick_mode (1, 128) == 6);
252+
assert (fx25_pick_mode (1, 191) == 9);
253+
assert (fx25_pick_mode (1, 223) == 5);
254+
assert (fx25_pick_mode (1, 239) == 1);
255+
assert (fx25_pick_mode (1, 240) == -1);
256+
244257
} // fx25_init
245258

246259

@@ -290,37 +303,43 @@ int fx25_get_debug (void)
290303
* Purpose: Pick suitable transmission format based on user preference
291304
* and size of data part required.
292305
*
293-
* Inputs: fx_mode - Normally, this would be 16, 32, or 64 for the desired number
294-
* of check bytes. The shortest format, adequate for the
295-
* required data length will be picked automatically.
296-
* 0x01 thru 0x0b may also be specified for a specific format
297-
* but this is expected to be mostly for testing, not normal
298-
* operation.
306+
* Inputs: fx_mode - 0 = none.
307+
* 1 = pick a tag automatically.
308+
* 16, 32, 64 = use this many check bytes.
309+
* 100 + n = use tag n.
310+
*
311+
* 0 and 1 would be the most common.
312+
* Others are mostly for testing.
299313
*
300314
* dlen - Required size for transmitted "data" part, in bytes.
301315
* This includes the AX.25 frame with bit stuffing and a flag
302316
* pattern on each end.
303317
*
304318
* Returns: Correlation tag number in range of CTAG_MIN thru CTAG_MAX.
305319
* -1 is returned for failure.
306-
*
307-
* Future: Might be more accomodating.
308-
* For example, if 64 check bytes were specified for 200 data bytes,
309-
* we might automatically drop it down to 32 check bytes, print a
310-
* warning and continue. Keep it simple at first. Fine tune later.
320+
* The caller should fall back to using plain old AX.25.
311321
*
312322
*--------------------------------------------------------------*/
313323

314324
int fx25_pick_mode (int fx_mode, int dlen)
315325
{
316-
if (fx_mode >= CTAG_MIN && fx_mode <= CTAG_MAX) {
317-
if (dlen <= fx25_get_k_data_radio(fx_mode)) {
318-
return (fx_mode);
326+
if (fx_mode <= 0) return (-1);
327+
328+
// Specify a specific tag by adding 100 to the number.
329+
// Fails if data won't fit.
330+
331+
if (fx_mode - 100 >= CTAG_MIN && fx_mode - 100 <= CTAG_MAX) {
332+
if (dlen <= fx25_get_k_data_radio(fx_mode - 100)) {
333+
return (fx_mode - 100);
319334
}
320335
else {
321336
return (-1); // Assuming caller prints failure message.
322337
}
323338
}
339+
340+
// Specify number of check bytes.
341+
// Pick the shortest one that can handle the required data length.
342+
324343
else if (fx_mode == 16 || fx_mode == 32 || fx_mode == 64) {
325344
for (int k = CTAG_MAX; k >= CTAG_MIN; k--) {
326345
if (fx_mode == fx25_get_nroots(k) && dlen <= fx25_get_k_data_radio(k)) {
@@ -329,11 +348,41 @@ int fx25_pick_mode (int fx_mode, int dlen)
329348
}
330349
return (-1);
331350
}
332-
else {
333-
text_color_set(DW_COLOR_ERROR);
334-
dw_printf("FX.25: Transmission format %d must be 0x00 thru 0x0b, 16, 32, or 64.\n", fx_mode);
335-
exit(EXIT_FAILURE);
351+
352+
// For any other number, [[ or if the preference was not possible, ?? ]]
353+
// try to come up with something reasonable. For shorter frames,
354+
// use smaller overhead. For longer frames, where an error is
355+
// more probable, use more check bytes. When the data gets even
356+
// larger, check bytes must be reduced to fit in block size.
357+
// When all else fails, fall back to normal AX.25.
358+
// Some of this is from observing UZ7HO Soundmodem behavior.
359+
//
360+
// Tag Data Check Max Num
361+
// Number Bytes Bytes Repaired
362+
// ------ ----- ----- -----
363+
// 0x04 32 16 8
364+
// 0x03 64 16 8
365+
// 0x06 128 32 16
366+
// 0x09 191 64 32
367+
// 0x05 223 32 16
368+
// 0x01 239 16 8
369+
// none larger
370+
//
371+
// The PRUG FX.25 TNC has additional modes that will handle larger frames
372+
// by using multiple RS blocks. This is a future possibility but needs
373+
// to be coordinated with other FX.25 developers so we maintain compatibility.
374+
375+
static const int prefer[6] = { 0x04, 0x03, 0x06, 0x09, 0x05, 0x01 };
376+
for (int k = 0; k < 6; k++) {
377+
int m = prefer[k];
378+
if (dlen <= fx25_get_k_data_radio(m)) {
379+
return (m);
380+
}
336381
}
382+
return (-1);
383+
384+
// TODO: revisit error messages, produced by caller, when this returns -1.
385+
337386
}
338387

339388

src/fx25_rec.c

+19-16
Original file line numberDiff line numberDiff line change
@@ -63,7 +63,7 @@ static struct fx_context_s *fx_context[MAX_CHANS][MAX_SUBCHANS][MAX_SLICERS];
6363

6464
static void process_rs_block (int chan, int subchan, int slice, struct fx_context_s *F);
6565

66-
static int my_unstuff (unsigned char * restrict pin, int ilen, unsigned char * restrict frame_buf);
66+
static int my_unstuff (int chan, int subchan, int slice, unsigned char * restrict pin, int ilen, unsigned char * restrict frame_buf);
6767

6868
//#define FXTEST 1 // Define for standalone test application.
6969
// It expects to find files fx01.dat, fx02.dat, ..., fx0b.dat/
@@ -182,7 +182,8 @@ void fx25_rec_bit (int chan, int subchan, int slice, int dbit)
182182

183183
if (fx25_get_debug() >= 2) {
184184
text_color_set(DW_COLOR_INFO);
185-
dw_printf ("FX.25: Matched correlation tag 0x%02x with %d bit errors. Expecting %d data & %d check bytes.\n",
185+
dw_printf ("FX.25[%d.%d]: Matched correlation tag 0x%02x with %d bit errors. Expecting %d data & %d check bytes.\n",
186+
chan, slice, // ideally subchan too only if applicable
186187
c,
187188
__builtin_popcountll(F->accum ^ fx25_get_ctag_value(c)),
188189
F->k_data_radio, F->nroots);
@@ -308,7 +309,7 @@ static void process_rs_block (int chan, int subchan, int slice, struct fx_contex
308309
{
309310
if (fx25_get_debug() >= 3) {
310311
text_color_set(DW_COLOR_DEBUG);
311-
dw_printf ("FX.25: Received RS codeblock.\n");
312+
dw_printf ("FX.25[%d.%d]: Received RS codeblock.\n", chan, slice);
312313
fx_hex_dump (F->block, FX25_BLOCK_SIZE);
313314
}
314315
assert (F->block[FX25_BLOCK_SIZE] == FENCE);
@@ -323,10 +324,10 @@ static void process_rs_block (int chan, int subchan, int slice, struct fx_contex
323324
if (fx25_get_debug() >= 2) {
324325
text_color_set(DW_COLOR_INFO);
325326
if (derrors == 0) {
326-
dw_printf ("FX.25: FEC complete with no errors.\n");
327+
dw_printf ("FX.25[%d.%d]: FEC complete with no errors.\n", chan, slice);
327328
}
328329
else {
329-
dw_printf ("FX.25: FEC complete, fixed %2d errors in byte positions:",derrors);
330+
dw_printf ("FX.25[%d.%d]: FEC complete, fixed %2d errors in byte positions:", chan, slice, derrors);
330331
for (int k = 0; k < derrors; k++) {
331332
dw_printf (" %d", derrlocs[k]);
332333
}
@@ -335,7 +336,7 @@ static void process_rs_block (int chan, int subchan, int slice, struct fx_contex
335336
}
336337

337338
unsigned char frame_buf[FX25_MAX_DATA+1]; // Out must be shorter than input.
338-
int frame_len = my_unstuff (F->block, F->dlen, frame_buf);
339+
int frame_len = my_unstuff (chan, subchan, slice, F->block, F->dlen, frame_buf);
339340

340341
if (frame_len >= 14 + 1 + 2) { // Minimum length: Two addresses & control & FCS.
341342

@@ -345,7 +346,7 @@ static void process_rs_block (int chan, int subchan, int slice, struct fx_contex
345346

346347
if (fx25_get_debug() >= 3) {
347348
text_color_set(DW_COLOR_DEBUG);
348-
dw_printf ("FX.25: Extracted AX.25 frame:\n");
349+
dw_printf ("FX.25[%d.%d]: Extracted AX.25 frame:\n", chan, slice);
349350
fx_hex_dump (frame_buf, frame_len);
350351
}
351352

@@ -360,22 +361,22 @@ static void process_rs_block (int chan, int subchan, int slice, struct fx_contex
360361
} else {
361362
// Most likely cause is defective sender software.
362363
text_color_set(DW_COLOR_ERROR);
363-
dw_printf ("FX.25: Bad FCS for AX.25 frame.\n");
364+
dw_printf ("FX.25[%d.%d]: Bad FCS for AX.25 frame.\n", chan, slice);
364365
fx_hex_dump (F->block, F->dlen);
365366
fx_hex_dump (frame_buf, frame_len);
366367
}
367368
}
368369
else {
369370
// Most likely cause is defective sender software.
370371
text_color_set(DW_COLOR_ERROR);
371-
dw_printf ("FX.25: AX.25 frame is shorter than minimum length.\n");
372+
dw_printf ("FX.25[%d.%d]: AX.25 frame is shorter than minimum length.\n", chan, slice);
372373
fx_hex_dump (F->block, F->dlen);
373374
fx_hex_dump (frame_buf, frame_len);
374375
}
375376
}
376377
else if (fx25_get_debug() >= 2) {
377378
text_color_set(DW_COLOR_ERROR);
378-
dw_printf ("FX.25: FEC failed. Too many errors.\n");
379+
dw_printf ("FX.25[%d.%d]: FEC failed. Too many errors.\n", chan, slice);
379380
}
380381

381382
} // process_rs_block
@@ -387,7 +388,9 @@ static void process_rs_block (int chan, int subchan, int slice, struct fx_contex
387388
*
388389
* Purpose: Remove HDLC it stuffing and surrounding flag delimiters.
389390
*
390-
* Inputs: pin - "data" part of RS codeblock.
391+
* Inputs: chan, subchan, slice - For error messages.
392+
*
393+
* pin - "data" part of RS codeblock.
391394
* First byte must be HDLC "flag".
392395
* May be followed by additional flags.
393396
* There must be terminating flag but it might not be byte aligned.
@@ -409,7 +412,7 @@ static void process_rs_block (int chan, int subchan, int slice, struct fx_contex
409412
*
410413
***********************************************************************************/
411414

412-
static int my_unstuff (unsigned char * restrict pin, int ilen, unsigned char * restrict frame_buf)
415+
static int my_unstuff (int chan, int subchan, int slice, unsigned char * restrict pin, int ilen, unsigned char * restrict frame_buf)
413416
{
414417
unsigned char pat_det = 0; // Pattern detector.
415418
unsigned char oacc = 0; // Accumulator for a byte out.
@@ -418,7 +421,7 @@ static int my_unstuff (unsigned char * restrict pin, int ilen, unsigned char * r
418421

419422
if (*pin != 0x7e) {
420423
text_color_set(DW_COLOR_ERROR);
421-
dw_printf ("FX.25 error: Data section did not start with 0x7e.\n");
424+
dw_printf ("FX.25[%d.%d] error: Data section did not start with 0x7e.\n", chan, slice);
422425
fx_hex_dump (pin, ilen);
423426
return (0);
424427
}
@@ -436,7 +439,7 @@ static int my_unstuff (unsigned char * restrict pin, int ilen, unsigned char * r
436439

437440
if (pat_det == 0xfe) {
438441
text_color_set(DW_COLOR_ERROR);
439-
dw_printf ("FX.25: Invalid AX.25 frame - Seven '1' bits in a row.\n");
442+
dw_printf ("FX.25[%d.%d]: Invalid AX.25 frame - Seven '1' bits in a row.\n", chan, slice);
440443
fx_hex_dump (pin, ilen);
441444
return 0;
442445
}
@@ -451,7 +454,7 @@ static int my_unstuff (unsigned char * restrict pin, int ilen, unsigned char * r
451454
}
452455
else {
453456
text_color_set(DW_COLOR_ERROR);
454-
dw_printf ("FX.25: Invalid AX.25 frame - Not a whole number of bytes.\n");
457+
dw_printf ("FX.25[%d.%d]: Invalid AX.25 frame - Not a whole number of bytes.\n", chan, slice);
455458
fx_hex_dump (pin, ilen);
456459
return (0);
457460
}
@@ -470,7 +473,7 @@ static int my_unstuff (unsigned char * restrict pin, int ilen, unsigned char * r
470473
} /* end of loop on all bits in block */
471474

472475
text_color_set(DW_COLOR_ERROR);
473-
dw_printf ("FX.25: Invalid AX.25 frame - Terminating flag not found.\n");
476+
dw_printf ("FX.25[%d.%d]: Invalid AX.25 frame - Terminating flag not found.\n", chan, slice);
474477
fx_hex_dump (pin, ilen);
475478

476479
return (0); // Should never fall off the end.

src/fx25_send.c

+11-6
Original file line numberDiff line numberDiff line change
@@ -61,7 +61,7 @@ int main ()
6161
dw_printf("Run fxrec as second part of test.\n");
6262

6363
fx25_init (3);
64-
for (int i = CTAG_MIN; i <= CTAG_MAX; i++) {
64+
for (int i = 100 + CTAG_MIN; i <= 100 + CTAG_MAX; i++) {
6565
fx25_send_frame (0, preload, (int)sizeof(preload)-3, i);
6666
}
6767
exit(EXIT_SUCCESS);
@@ -115,7 +115,7 @@ int fx25_send_frame (int chan, unsigned char *fbuf, int flen, int fx_mode)
115115
if (fx25_get_debug() >= 3) {
116116
text_color_set(DW_COLOR_DEBUG);
117117
dw_printf ("------\n");
118-
dw_printf ("FX.25 send frame: chan = %d, FX.25 mode = %d\n", chan, fx_mode);
118+
dw_printf ("FX.25[%d] send frame: FX.25 mode = %d\n", chan, fx_mode);
119119
fx_hex_dump (fbuf, flen);
120120
}
121121

@@ -138,7 +138,7 @@ int fx25_send_frame (int chan, unsigned char *fbuf, int flen, int fx_mode)
138138
assert (data[FX25_MAX_DATA] == fence);
139139
if (dlen < 0) {
140140
text_color_set(DW_COLOR_ERROR);
141-
dw_printf ("FX.25: Frame length of %d + overhead is too large to encode.\n", flen);
141+
dw_printf ("FX.25[%d]: Frame length of %d + overhead is too large to encode.\n", chan, flen);
142142
return (-1);
143143
}
144144

@@ -150,7 +150,7 @@ int fx25_send_frame (int chan, unsigned char *fbuf, int flen, int fx_mode)
150150

151151
if (ctag_num < CTAG_MIN || ctag_num > CTAG_MAX) {
152152
text_color_set(DW_COLOR_ERROR);
153-
dw_printf ("FX.25: Could not find suitable format for requested %d and data length %d.\n", fx_mode, dlen);
153+
dw_printf ("FX.25[%d]: Could not find suitable format for requested %d and data length %d.\n", chan, fx_mode, dlen);
154154
return (-1);
155155
}
156156

@@ -179,9 +179,9 @@ int fx25_send_frame (int chan, unsigned char *fbuf, int flen, int fx_mode)
179179

180180
if (fx25_get_debug() >= 3) {
181181
text_color_set(DW_COLOR_DEBUG);
182-
dw_printf ("FX.25: transmit %d data bytes, ctag number 0x%02x\n", k_data_radio, ctag_num);
182+
dw_printf ("FX.25[%d]: transmit %d data bytes, ctag number 0x%02x\n", chan, k_data_radio, ctag_num);
183183
fx_hex_dump (data, k_data_radio);
184-
dw_printf ("FX.25: transmit %d check bytes:\n", NROOTS);
184+
dw_printf ("FX.25[%d]: transmit %d check bytes:\n", chan, NROOTS);
185185
fx_hex_dump (check, NROOTS);
186186
dw_printf ("------\n");
187187
}
@@ -212,6 +212,11 @@ int fx25_send_frame (int chan, unsigned char *fbuf, int flen, int fx_mode)
212212
#else
213213
// Normal usage. Send bits to modulator.
214214

215+
// Temp hack for testing. Corrupt first 8 bytes.
216+
// for (int j = 0; j < 16; j++) {
217+
// data[j] = ~ data[j];
218+
// }
219+
215220
for (int k = 0; k < 8; k++) {
216221
unsigned char b = (ctag_value >> (k * 8)) & 0xff;
217222
send_bytes (chan, &b, 1);

0 commit comments

Comments
 (0)