Meshtastic-Apple/Meshtastic/Audio/codec2-ios/fec_dec.c
Benjamin Faershtein 90aecb4d08 working changes
2026-02-23 15:20:08 -08:00

307 lines
10 KiB
C

/*---------------------------------------------------------------------------*\
FILE........: fec_dec.c
AUTHOR......: David Rowe
DATE CREATED: 4 march 2013
FEC decoder for data from modem containing compressed Codec 2 data
and FEC.
\*---------------------------------------------------------------------------*/
/*
Copyright (C) 2013 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/>.
*/
#include "codec2.h"
#include "codec2_fdmdv.h"
#include "golay23.h"
#include <assert.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <errno.h>
#define MODE_1600 0
#define MODE_1850 1
#define MODE_2000 2
int main(int argc, char *argv[])
{
void *codec2, *fdmdv;
FILE *fin;
FILE *fout;
int bits_per_input_frame, bytes_per_input_frame;
unsigned char *packed_input_bits;
int *unpacked_input_bits;
int bits_per_output_frame, bytes_per_output_frame;
unsigned char *packed_output_bits;
int *unpacked_output_bits;
int codec2_mode, mode, Nc, bit, byte;
int i,j;
int recd_codeword, codeword1, codeword2;
if (argc < 3) {
printf("%s InputFromModemWithFECFile OutputToCodec2File [2000|1600]\n", argv[0]);
exit(1);
}
if (strcmp(argv[1], "-") == 0) fin = stdin;
else if ( (fin = fopen(argv[1],"rb")) == NULL ) {
fprintf(stderr, "Error opening input file from Demod: %s: %s.\n",
argv[2], strerror(errno));
exit(1);
}
if (strcmp(argv[2], "-") == 0) fout = stdout;
else if ( (fout = fopen(argv[2],"wb")) == NULL ) {
fprintf(stderr, "Error opening output file to Codec : %s: %s.\n",
argv[3], strerror(errno));
exit(1);
}
/* input parameters and buffers. Note data is split into two 20ms
frames for transmission over modem. */
if ((argc != 4) || (strcmp(argv[3],"2000") == 0)) {
/* 2000 bit/s with FEC */
mode = MODE_2000;
codec2_mode = CODEC2_MODE_1400;
Nc = 20;
} else if ((strcmp(argv[3],"1850") == 0)) {
/* 1850 bit/s with FEC */
mode = MODE_1850;
codec2_mode = CODEC2_MODE_1300;
Nc = 20;
}
else if (strcmp(argv[3],"1600") == 0) {
/* 1600 bit/s with FEC (actually 1575 with one spare) */
mode = MODE_1600;
codec2_mode = CODEC2_MODE_1300;
Nc = 16;
}
else {
fprintf(stderr, "Error in mode: %s. Must be 2000 or 1600\n", argv[3]);
exit(1);
}
fdmdv = fdmdv_create(Nc);
bits_per_input_frame = 2*fdmdv_bits_per_frame(fdmdv);
bytes_per_input_frame = bits_per_input_frame / 8;
assert((bits_per_input_frame % 8) == 0); /* make sure integer number of bytes per frame */
packed_input_bits = (unsigned char*)malloc(bytes_per_input_frame*sizeof(char));
assert(packed_input_bits != NULL);
unpacked_input_bits = (int*)malloc(bits_per_input_frame*sizeof(int));
assert(unpacked_input_bits != NULL);
/*
Output parameters and buffers.
*/
codec2 = codec2_create(codec2_mode);
bits_per_output_frame = codec2_bits_per_frame(codec2);
bytes_per_output_frame = (bits_per_output_frame+7)/8;
packed_output_bits = (unsigned char*)malloc(bytes_per_output_frame*sizeof(char));
assert(packed_output_bits != NULL);
unpacked_output_bits = (int*)malloc(bits_per_output_frame*sizeof(int));
assert(unpacked_output_bits != NULL);
fprintf(stderr, "input bits: %d input_bytes: %d output_bits: %d output_bytes: %d\n",
bits_per_input_frame, bytes_per_input_frame, bits_per_output_frame, bytes_per_output_frame);
/* main loop */
golay23_init();
while(fread(packed_input_bits, sizeof(char), bytes_per_input_frame, fin) == (size_t)bytes_per_input_frame) {
/* unpack bits, MSB first */
bit = 7; byte = 0;
for(i=0; i<bits_per_input_frame; i++) {
unpacked_input_bits[i] = (packed_input_bits[byte] >> bit) & 0x1;
bit--;
if (bit < 0) {
bit = 7;
byte++;
}
}
#ifdef TEST
/* Some test bit errors (not comprehesnive) */
unpacked_input_bits[0] = (unpacked_input_bits[0] ^ 1) & 0x1;
unpacked_input_bits[23] = (unpacked_input_bits[23] ^ 1) & 0x1;
#endif
if (mode == MODE_2000) {
/* decode first codeword */
recd_codeword = 0;
for(i=0; i<12; i++) {
recd_codeword <<= 1;
recd_codeword |= unpacked_input_bits[i];
}
for(i=bits_per_output_frame; i<bits_per_output_frame+11; i++) {
recd_codeword <<= 1;
recd_codeword |= unpacked_input_bits[i];
}
codeword1 = golay23_decode(recd_codeword);
//codeword1 = recd_codeword;
//fprintf(stderr, "received codeword1: 0x%x decoded codeword1: 0x%x\n", recd_codeword, codeword1);
for(i=0; i<12; i++) {
unpacked_output_bits[i] = (codeword1 >> (22-i)) & 0x1;
}
/* decode second codeword */
recd_codeword = 0;
for(i=12; i<24; i++) {
recd_codeword <<= 1;
recd_codeword |= unpacked_input_bits[i];
}
for(i=bits_per_output_frame+11; i<bits_per_output_frame+11+11; i++) {
recd_codeword <<= 1;
recd_codeword |= unpacked_input_bits[i];
}
codeword2 = golay23_decode(recd_codeword);
//codeword2 = recd_codeword;
//fprintf(stderr, "received codeword2: 0x%x decoded codeword2: 0x%x\n", recd_codeword, codeword2);
for(i=0; i<12; i++) {
unpacked_output_bits[12+i] = (codeword2 >> (22-i)) & 0x1;
}
/* unprotected bits */
for(i=24; i<bits_per_output_frame; i++)
unpacked_output_bits[i] = unpacked_input_bits[i];
}
if (mode == MODE_1600) {
recd_codeword = 0;
for(i=0; i<8; i++) {
recd_codeword <<= 1;
recd_codeword |= unpacked_input_bits[i];
}
for(i=11; i<15; i++) {
recd_codeword <<= 1;
recd_codeword |= unpacked_input_bits[i];
}
for(i=bits_per_output_frame; i<bits_per_output_frame+11; i++) {
recd_codeword <<= 1;
recd_codeword |= unpacked_input_bits[i];
}
codeword1 = golay23_decode(recd_codeword);
//codeword1 = recd_codeword;
//fprintf(stderr, "received codeword1: 0x%x decoded codeword1: 0x%x\n", recd_codeword, codeword1);
for(i=0; i<bits_per_output_frame; i++)
unpacked_output_bits[i] = unpacked_input_bits[i];
for(i=0; i<8; i++) {
unpacked_output_bits[i] = (codeword1 >> (22-i)) & 0x1;
}
for(i=8,j=11; i<12; i++,j++) {
unpacked_output_bits[j] = (codeword1 >> (22-i)) & 0x1;
}
}
if (mode == MODE_1850) {
recd_codeword = 0;
for(i=0; i<8; i++) {
recd_codeword <<= 1;
recd_codeword |= unpacked_input_bits[i];
}
for(i=11; i<15; i++) {
recd_codeword <<= 1;
recd_codeword |= unpacked_input_bits[i];
}
for(i=bits_per_output_frame; i<bits_per_output_frame+11; i++) {
recd_codeword <<= 1;
recd_codeword |= unpacked_input_bits[i];
}
codeword1 = golay23_decode(recd_codeword);
//codeword1 = recd_codeword;
//fprintf(stderr, "received codeword1: 0x%x decoded codeword1: 0x%x\n", recd_codeword, codeword1);
recd_codeword = 0;
for(i=16; i<28; i++) {
recd_codeword <<= 1;
recd_codeword |= unpacked_input_bits[i];
}
for(i=bits_per_output_frame+11; i<bits_per_output_frame+11+11; i++) {
recd_codeword <<= 1;
recd_codeword |= unpacked_input_bits[i];
}
codeword2 = golay23_decode(recd_codeword);
fprintf(stderr, "received codeword2: 0x%x decoded codeword2: 0x%x\n", recd_codeword, codeword2);
for(i=0; i<bits_per_output_frame; i++)
unpacked_output_bits[i] = unpacked_input_bits[i];
for(i=0; i<8; i++) {
unpacked_output_bits[i] = (codeword1 >> (22-i)) & 0x1;
}
for(i=8,j=11; i<12; i++,j++) {
unpacked_output_bits[j] = (codeword1 >> (22-i)) & 0x1;
}
for(i=0,j=16; i<12; i++,j++) {
unpacked_output_bits[j] = (codeword2 >> (22-i)) & 0x1;
}
}
/* pack bits, MSB first */
bit = 7; byte = 0;
memset(packed_output_bits, 0, bytes_per_output_frame);
for(i=0; i<bits_per_output_frame; i++) {
packed_output_bits[byte] |= (unpacked_output_bits[i] << bit);
bit--;
if (bit < 0) {
bit = 7;
byte++;
}
}
fwrite(packed_output_bits, sizeof(char), bytes_per_output_frame, fout);
/* if this is in a pipeline, we probably don't want the usual
buffering to occur */
if (fout == stdout) fflush(stdout);
if (fin == stdin) fflush(stdin);
}
codec2_destroy(codec2);
fdmdv_destroy(fdmdv);
free(packed_input_bits);
free(unpacked_input_bits);
free(packed_output_bits);
free(unpacked_output_bits);
fclose(fin);
fclose(fout);
return 0;
}