// // This file is part of Dire Wolf, an amateur radio packet TNC. // // Copyright (C) 2019 John Langner, WB2OSZ // // 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, see . // // ----------------------------------------------------------------------- // // // Some of this is based on: // // FX.25 Encoder // Author: Jim McGuire KB3MPL // Date: 23 October 2007 // // This program is a single-file implementation of the FX.25 encapsulation // structure for use with AX.25 data packets. Details of the FX.25 // specification are available at: // http://www.stensat.org/Docs/Docs.htm // // This program implements a single RS(255,239) FEC structure. Future // releases will incorporate more capabilities as accommodated in the FX.25 // spec. // // The Reed Solomon encoding routines are based on work performed by // Phil Karn. Phil was kind enough to release his code under the GPL, as // noted below. Consequently, this FX.25 implementation is also released // under the terms of the GPL. // // Phil Karn's original copyright notice: /* Test the Reed-Solomon codecs * for various block sizes and with random data and random error patterns * * Copyright 2002 Phil Karn, KA9Q * May be used under the terms of the GNU General Public License (GPL) * */ #include "direwolf.h" #include #include #include #include #include // uint64_t #include // PRIx64 #include #include "fx25.h" #include "textcolor.h" #define NTAB 3 static struct { int symsize; // Symbol size, bits (1-8). Always 8 for this application. int genpoly; // Field generator polynomial coefficients. int fcs; // First root of RS code generator polynomial, index form. int prim; // Primitive element to generate polynomial roots. int nroots; // RS code generator polynomial degree (number of roots). // Same as number of check bytes added. struct rs *rs; // Pointer to RS codec control block. Filled in at init time. } Tab[NTAB] = { {8, 0x11d, 1, 1, 16, NULL }, // RS(255,239) {8, 0x11d, 1, 1, 32, NULL }, // RS(255,223) {8, 0x11d, 1, 1, 64, NULL }, // RS(255,191) }; /* * Reference: http://www.stensat.org/docs/FX-25_01_06.pdf * FX.25 * Forward Error Correction Extension to * AX.25 Link Protocol For Amateur Packet Radio * Version: 0.01 DRAFT * Date: 01 September 2006 */ struct correlation_tag_s { uint64_t value; // 64 bit value, send LSB first. int n_block_radio; // Size of transmitted block, all in bytes. int k_data_radio; // Size of transmitted data part. int n_block_rs; // Size of RS algorithm block. int k_data_rs; // Size of RS algorithm data part. int itab; // Index into Tab array. }; static const struct correlation_tag_s tags[16] = { /* Tag_00 */ { 0x566ED2717946107ELL, 0, 0, 0, 0, -1 }, // Reserved /* Tag_01 */ { 0xB74DB7DF8A532F3ELL, 255, 239, 255, 239, 0 }, // RS(255, 239) 16-byte check value, 239 information bytes /* Tag_02 */ { 0x26FF60A600CC8FDELL, 144, 128, 255, 239, 0 }, // RS(144,128) - shortened RS(255, 239), 128 info bytes /* Tag_03 */ { 0xC7DC0508F3D9B09ELL, 80, 64, 255, 239, 0 }, // RS(80,64) - shortened RS(255, 239), 64 info bytes /* Tag_04 */ { 0x8F056EB4369660EELL, 48, 32, 255, 239, 0 }, // RS(48,32) - shortened RS(255, 239), 32 info bytes /* Tag_05 */ { 0x6E260B1AC5835FAELL, 255, 223, 255, 223, 1 }, // RS(255, 223) 32-byte check value, 223 information bytes /* Tag_06 */ { 0xFF94DC634F1CFF4ELL, 160, 128, 255, 223, 1 }, // RS(160,128) - shortened RS(255, 223), 128 info bytes /* Tag_07 */ { 0x1EB7B9CDBC09C00ELL, 96, 64, 255, 223, 1 }, // RS(96,64) - shortened RS(255, 223), 64 info bytes /* Tag_08 */ { 0xDBF869BD2DBB1776LL, 64, 32, 255, 223, 1 }, // RS(64,32) - shortened RS(255, 223), 32 info bytes /* Tag_09 */ { 0x3ADB0C13DEAE2836LL, 255, 191, 255, 191, 2 }, // RS(255, 191) 64-byte check value, 191 information bytes /* Tag_0A */ { 0xAB69DB6A543188D6LL, 192, 128, 255, 191, 2 }, // RS(192, 128) - shortened RS(255, 191), 128 info bytes /* Tag_0B */ { 0x4A4ABEC4A724B796LL, 128, 64, 255, 191, 2 }, // RS(128, 64) - shortened RS(255, 191), 64 info bytes /* Tag_0C */ { 0x0293D578626B67E6LL, 0, 0, 0, 0, -1 }, // Undefined /* Tag_0D */ { 0xE3B0B0D6917E58A6LL, 0, 0, 0, 0, -1 }, // Undefined /* Tag_0E */ { 0x720267AF1BE1F846LL, 0, 0, 0, 0, -1 }, // Undefined /* Tag_0F */ { 0x93210201E8F4C706LL, 0, 0, 0, 0, -1 } // Undefined }; #define CLOSE_ENOUGH 8 // How many bits can be wrong in tag yet consider it a match? // Needs to be large enough to match with significant errors // but not so large to get frequent false matches. // Probably don't want >= 16 because the hamming distance between // any two pairs is 32. // What is a good number? 8?? 12?? 15?? // 12 got many false matches with random noise. // Even 8 might be too high. We see 2 or 4 bit errors here // at the point where decoding the block is very improbable. // After 2 months of continuous operation as a digipeater/iGate, // no false triggers were observed. So 8 doesn't seem to be too // high for 1200 bps. No study has been done for 9600 bps. // Given a 64 bit correlation tag value, find acceptable match in table. // Return index into table or -1 for no match. // Both gcc and clang have a built in function to count the number of '1' bits // in an integer. This can result in a single machine instruction. You might need // to supply your own popcount function if using a different compiler. int fx25_tag_find_match (uint64_t t) { for (int c = CTAG_MIN; c <= CTAG_MAX; c++) { if (__builtin_popcountll(t ^ tags[c].value) <= CLOSE_ENOUGH) { //printf ("%016" PRIx64 " received\n", t); //printf ("%016" PRIx64 " tag %d\n", tags[c].value, c); //printf ("%016" PRIx64 " xor, popcount = %d\n", t ^ tags[c].value, __builtin_popcountll(t ^ tags[c].value)); return (c); } } return (-1); } void free_rs_char(struct rs *rs){ free(rs->alpha_to); free(rs->index_of); free(rs->genpoly); free(rs); } /*------------------------------------------------------------- * * Name: fx25_init * * Purpose: This must be called once before any of the other fx25 functions. * * Inputs: debug_level - Controls level of informational / debug messages. * * 0 Only errors. * 1 (default) Transmitting ctag. Currently no other way to know this. * 2 Receive correlation tag detected. FEC decode complete. * 3 Dump data going in and out. * * Use command line -dx to increase level or -qx for quiet. * * Description: Initialize 3 Reed-Solomon codecs, for 16, 32, and 64 check bytes. * *--------------------------------------------------------------*/ static int g_debug_level; void fx25_init ( int debug_level ) { g_debug_level = debug_level; for (int i = 0 ; i < NTAB ; i++) { Tab[i].rs = INIT_RS(Tab[i].symsize, Tab[i].genpoly, Tab[i].fcs, Tab[i].prim, Tab[i].nroots); if (Tab[i].rs == NULL) { text_color_set(DW_COLOR_ERROR); dw_printf("FX.25 internal error: init_rs_char failed!\n"); exit(EXIT_FAILURE); } } // Verify integrity of tables and assumptions. // This also does a quick check for the popcount function. for (int j = 0; j < 16 ; j++) { for (int k = 0; k < 16; k++) { if (j == k) { assert (__builtin_popcountll(tags[j].value ^ tags[k].value) == 0); } else { assert (__builtin_popcountll(tags[j].value ^ tags[k].value) == 32); } } } for (int j = CTAG_MIN; j <= CTAG_MAX; j++) { assert (tags[j].n_block_radio - tags[j].k_data_radio == Tab[tags[j].itab].nroots); assert (tags[j].n_block_rs - tags[j].k_data_rs == Tab[tags[j].itab].nroots); assert (tags[j].n_block_rs == FX25_BLOCK_SIZE); } assert (fx25_pick_mode (100+1, 239) == 1); assert (fx25_pick_mode (100+1, 240) == -1); assert (fx25_pick_mode (100+5, 223) == 5); assert (fx25_pick_mode (100+5, 224) == -1); assert (fx25_pick_mode (100+9, 191) == 9); assert (fx25_pick_mode (100+9, 192) == -1); assert (fx25_pick_mode (16, 32) == 4); assert (fx25_pick_mode (16, 64) == 3); assert (fx25_pick_mode (16, 128) == 2); assert (fx25_pick_mode (16, 239) == 1); assert (fx25_pick_mode (16, 240) == -1); assert (fx25_pick_mode (32, 32) == 8); assert (fx25_pick_mode (32, 64) == 7); assert (fx25_pick_mode (32, 128) == 6); assert (fx25_pick_mode (32, 223) == 5); assert (fx25_pick_mode (32, 234) == -1); assert (fx25_pick_mode (64, 64) == 11); assert (fx25_pick_mode (64, 128) == 10); assert (fx25_pick_mode (64, 191) == 9); assert (fx25_pick_mode (64, 192) == -1); assert (fx25_pick_mode (1, 32) == 4); assert (fx25_pick_mode (1, 33) == 3); assert (fx25_pick_mode (1, 64) == 3); assert (fx25_pick_mode (1, 65) == 6); assert (fx25_pick_mode (1, 128) == 6); assert (fx25_pick_mode (1, 191) == 9); assert (fx25_pick_mode (1, 223) == 5); assert (fx25_pick_mode (1, 239) == 1); assert (fx25_pick_mode (1, 240) == -1); } // fx25_init // Get properties of specified CTAG number. struct rs *fx25_get_rs (int ctag_num) { assert (ctag_num >= CTAG_MIN && ctag_num <= CTAG_MAX); assert (tags[ctag_num].itab >= 0 && tags[ctag_num].itab < NTAB); assert (Tab[tags[ctag_num].itab].rs != NULL); return (Tab[tags[ctag_num].itab].rs); } uint64_t fx25_get_ctag_value (int ctag_num) { assert (ctag_num >= CTAG_MIN && ctag_num <= CTAG_MAX); return (tags[ctag_num].value); } int fx25_get_k_data_radio (int ctag_num) { assert (ctag_num >= CTAG_MIN && ctag_num <= CTAG_MAX); return (tags[ctag_num].k_data_radio); } int fx25_get_k_data_rs (int ctag_num) { assert (ctag_num >= CTAG_MIN && ctag_num <= CTAG_MAX); return (tags[ctag_num].k_data_rs); } int fx25_get_nroots (int ctag_num) { assert (ctag_num >= CTAG_MIN && ctag_num <= CTAG_MAX); return (Tab[tags[ctag_num].itab].nroots); } int fx25_get_debug (void) { return (g_debug_level); } /*------------------------------------------------------------- * * Name: fx25_pick_mode * * Purpose: Pick suitable transmission format based on user preference * and size of data part required. * * Inputs: fx_mode - 0 = none. * 1 = pick a tag automatically. * 16, 32, 64 = use this many check bytes. * 100 + n = use tag n. * * 0 and 1 would be the most common. * Others are mostly for testing. * * dlen - Required size for transmitted "data" part, in bytes. * This includes the AX.25 frame with bit stuffing and a flag * pattern on each end. * * Returns: Correlation tag number in range of CTAG_MIN thru CTAG_MAX. * -1 is returned for failure. * The caller should fall back to using plain old AX.25. * *--------------------------------------------------------------*/ int fx25_pick_mode (int fx_mode, int dlen) { if (fx_mode <= 0) return (-1); // Specify a specific tag by adding 100 to the number. // Fails if data won't fit. if (fx_mode - 100 >= CTAG_MIN && fx_mode - 100 <= CTAG_MAX) { if (dlen <= fx25_get_k_data_radio(fx_mode - 100)) { return (fx_mode - 100); } else { return (-1); // Assuming caller prints failure message. } } // Specify number of check bytes. // Pick the shortest one that can handle the required data length. else if (fx_mode == 16 || fx_mode == 32 || fx_mode == 64) { for (int k = CTAG_MAX; k >= CTAG_MIN; k--) { if (fx_mode == fx25_get_nroots(k) && dlen <= fx25_get_k_data_radio(k)) { return (k); } } return (-1); } // For any other number, [[ or if the preference was not possible, ?? ]] // try to come up with something reasonable. For shorter frames, // use smaller overhead. For longer frames, where an error is // more probable, use more check bytes. When the data gets even // larger, check bytes must be reduced to fit in block size. // When all else fails, fall back to normal AX.25. // Some of this is from observing UZ7HO Soundmodem behavior. // // Tag Data Check Max Num // Number Bytes Bytes Repaired // ------ ----- ----- ----- // 0x04 32 16 8 // 0x03 64 16 8 // 0x06 128 32 16 // 0x09 191 64 32 // 0x05 223 32 16 // 0x01 239 16 8 // none larger // // The PRUG FX.25 TNC has additional modes that will handle larger frames // by using multiple RS blocks. This is a future possibility but needs // to be coordinated with other FX.25 developers so we maintain compatibility. // See https://web.tapr.org/meetings/DCC_2020/JE1WAZ/DCC-2020-PRUG-FINAL.pptx static const int prefer[6] = { 0x04, 0x03, 0x06, 0x09, 0x05, 0x01 }; for (int k = 0; k < 6; k++) { int m = prefer[k]; if (dlen <= fx25_get_k_data_radio(m)) { return (m); } } return (-1); // TODO: revisit error messages, produced by caller, when this returns -1. } /* Initialize a Reed-Solomon codec * symsize = symbol size, bits (1-8) - always 8 for this application. * gfpoly = Field generator polynomial coefficients * fcr = first root of RS code generator polynomial, index form * prim = primitive element to generate polynomial roots * nroots = RS code generator polynomial degree (number of roots) */ struct rs *INIT_RS(unsigned int symsize,unsigned int gfpoly,unsigned fcr,unsigned prim, unsigned int nroots){ struct rs *rs; int i, j, sr,root,iprim; if(symsize > 8*sizeof(DTYPE)) return NULL; /* Need version with ints rather than chars */ if(fcr >= (1<= (1<= (1<mm = symsize; rs->nn = (1<alpha_to = (DTYPE *)calloc((rs->nn+1),sizeof(DTYPE)); if(rs->alpha_to == NULL){ text_color_set(DW_COLOR_ERROR); dw_printf ("FATAL ERROR: Out of memory.\n"); exit (EXIT_FAILURE); } rs->index_of = (DTYPE *)calloc((rs->nn+1),sizeof(DTYPE)); if(rs->index_of == NULL){ text_color_set(DW_COLOR_ERROR); dw_printf ("FATAL ERROR: Out of memory.\n"); exit (EXIT_FAILURE); } /* Generate Galois field lookup tables */ rs->index_of[0] = A0; /* log(zero) = -inf */ rs->alpha_to[A0] = 0; /* alpha**-inf = 0 */ sr = 1; for(i=0;inn;i++){ rs->index_of[sr] = i; rs->alpha_to[i] = sr; sr <<= 1; if(sr & (1<nn; } if(sr != 1){ /* field generator polynomial is not primitive! */ free(rs->alpha_to); free(rs->index_of); free(rs); return NULL; } /* Form RS code generator polynomial from its roots */ rs->genpoly = (DTYPE *)calloc((nroots+1),sizeof(DTYPE)); if(rs->genpoly == NULL){ text_color_set(DW_COLOR_ERROR); dw_printf ("FATAL ERROR: Out of memory.\n"); exit (EXIT_FAILURE); } rs->fcr = fcr; rs->prim = prim; rs->nroots = nroots; /* Find prim-th root of 1, used in decoding */ for(iprim=1;(iprim % prim) != 0;iprim += rs->nn) ; rs->iprim = iprim / prim; rs->genpoly[0] = 1; for (i = 0,root=fcr*prim; i < nroots; i++,root += prim) { rs->genpoly[i+1] = 1; /* Multiply rs->genpoly[] by @**(root + x) */ for (j = i; j > 0; j--){ if (rs->genpoly[j] != 0) rs->genpoly[j] = rs->genpoly[j-1] ^ rs->alpha_to[modnn(rs,rs->index_of[rs->genpoly[j]] + root)]; else rs->genpoly[j] = rs->genpoly[j-1]; } /* rs->genpoly[0] can never be zero */ rs->genpoly[0] = rs->alpha_to[modnn(rs,rs->index_of[rs->genpoly[0]] + root)]; } /* convert rs->genpoly[] to index form for quicker encoding */ for (i = 0; i <= nroots; i++) { rs->genpoly[i] = rs->index_of[rs->genpoly[i]]; } // diagnostic prints #if 0 printf("Alpha To:\n\r"); for (i=0; i < sizeof(DTYPE)*(rs->nn+1); i++) printf("0x%2x,", rs->alpha_to[i]); printf("\n\r"); printf("Index Of:\n\r"); for (i=0; i < sizeof(DTYPE)*(rs->nn+1); i++) printf("0x%2x,", rs->index_of[i]); printf("\n\r"); printf("GenPoly:\n\r"); for (i = 0; i <= nroots; i++) printf("0x%2x,", rs->genpoly[i]); printf("\n\r"); #endif return rs; } // TEMPORARY!!! // FIXME: We already have multiple copies of this. // Consolidate them into one somewhere. void fx_hex_dump (unsigned char *p, int len) { int n, i, offset; offset = 0; while (len > 0) { n = len < 16 ? len : 16; dw_printf (" %03x: ", offset); for (i=0; i