270 lines
11 KiB
Plaintext
270 lines
11 KiB
Plaintext
|
// qra64.h
|
||
|
// Encoding/decoding functions for the QRA64 mode
|
||
|
//
|
||
|
// (c) 2016 - Nico Palermo, IV3NWV
|
||
|
// ------------------------------------------------------------------------------
|
||
|
// This file is part of the qracodes project, a Forward Error Control
|
||
|
// encoding/decoding package based on Q-ary RA (Repeat and Accumulate) LDPC codes.
|
||
|
//
|
||
|
// qracodes 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 3 of the License, or
|
||
|
// (at your option) any later version.
|
||
|
// qracodes 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 qracodes source distribution.
|
||
|
// If not, see <http://www.gnu.org/licenses/>.
|
||
|
|
||
|
#ifndef _qra64_h_
|
||
|
#define _qra64_h_
|
||
|
|
||
|
// qra64_init(...) initialization flags
|
||
|
#define QRA_NOAP 0 // don't use a-priori knowledge
|
||
|
#define QRA_AUTOAP 1 // use auto a-priori knowledge
|
||
|
#define QRA_USERAP 2 // a-priori knowledge messages provided by the user
|
||
|
|
||
|
// QRA code parameters
|
||
|
#define QRA64_K 12 // information symbols
|
||
|
#define QRA64_N 63 // codeword length
|
||
|
#define QRA64_C 51 // (number of parity checks C=(N-K))
|
||
|
#define QRA64_M 64 // code alphabet size
|
||
|
#define QRA64_m 6 // bits per symbol
|
||
|
|
||
|
// packed predefined callsigns and fields as defined in JT65
|
||
|
#define CALL_CQ 0xFA08319
|
||
|
#define CALL_QRZ 0xFA0831A
|
||
|
#define CALL_CQ000 0xFA0831B
|
||
|
#define CALL_CQ999 0xFA08702
|
||
|
#define CALL_CQDX 0x5624C39
|
||
|
#define CALL_DE 0xFF641D1
|
||
|
#define GRID_BLANK 0x7E91
|
||
|
|
||
|
// Types of a-priori knowledge messages
|
||
|
#define APTYPE_CQQRZ 0 // [cq/qrz ? ?/blank]
|
||
|
#define APTYPE_MYCALL 1 // [mycall ? ?/blank]
|
||
|
#define APTYPE_HISCALL 2 // [? hiscall ?/blank]
|
||
|
#define APTYPE_BOTHCALLS 3 // [mycall hiscall ?]
|
||
|
#define APTYPE_FULL 4 // [mycall hiscall grid]
|
||
|
#define APTYPE_CQHISCALL 5 // [cq/qrz hiscall ?/blank]
|
||
|
#define APTYPE_SIZE (APTYPE_CQHISCALL+1)
|
||
|
|
||
|
typedef struct {
|
||
|
float decEsNoMetric;
|
||
|
int apflags;
|
||
|
int apmsg_set[APTYPE_SIZE]; // indicate which ap type knowledge has
|
||
|
// been set by the user
|
||
|
// ap messages buffers
|
||
|
int apmsg_cqqrz[12]; // [cq/qrz ? ?/blank]
|
||
|
int apmsg_call1[12]; // [mycall ? ?/blank]
|
||
|
int apmsg_call2[12]; // [? hiscall ?/blank]
|
||
|
int apmsg_call1_call2[12]; // [mycall hiscall ?]
|
||
|
int apmsg_call1_call2_grid[12]; // [mycall hiscall grid]
|
||
|
int apmsg_cq_call2[12]; // [cq hiscall ?/blank]
|
||
|
int apmsg_cq_call2_grid[12]; // [cq hiscall grid]
|
||
|
|
||
|
// ap messages masks
|
||
|
int apmask_cqqrz[12];
|
||
|
int apmask_cqqrz_ooo[12];
|
||
|
int apmask_call1[12];
|
||
|
int apmask_call1_ooo[12];
|
||
|
int apmask_call2[12];
|
||
|
int apmask_call2_ooo[12];
|
||
|
int apmask_call1_call2[12];
|
||
|
int apmask_call1_call2_grid[12];
|
||
|
int apmask_cq_call2[12];
|
||
|
int apmask_cq_call2_ooo[12];
|
||
|
} qra64codec;
|
||
|
|
||
|
#ifdef __cplusplus
|
||
|
extern "C" {
|
||
|
#endif
|
||
|
|
||
|
qra64codec *qra64_init(int flags);
|
||
|
// QRA64 mode initialization function
|
||
|
// arguments:
|
||
|
// flags: set the decoder mode
|
||
|
// QRA_NOAP use no a-priori information
|
||
|
// QRA_AUTOAP use any relevant previous decodes
|
||
|
// QRA_USERAP use a-priori information provided via qra64_apset(...)
|
||
|
// returns:
|
||
|
// Pointer to initialized qra64codec data structure
|
||
|
// this pointer should be passed to the encoding/decoding functions
|
||
|
//
|
||
|
// 0 if unsuccessful (can't allocate memory)
|
||
|
// ----------------------------------------------------------------------------
|
||
|
|
||
|
void qra64_encode(qra64codec *pcodec, int *y, const int *x);
|
||
|
// QRA64 encoder
|
||
|
// arguments:
|
||
|
// pcodec = pointer to a qra64codec data structure as returned by qra64_init
|
||
|
// x = pointer to the message to be encoded, int x[12]
|
||
|
// x must point to an array of integers (i.e. defined as int x[12])
|
||
|
// y = pointer to encoded message, int y[63]=
|
||
|
// ----------------------------------------------------------------------------
|
||
|
|
||
|
int qra64_decode(qra64codec *pcodec, float *ebno, int *x, const float *r);
|
||
|
// QRA64 mode decoder
|
||
|
// arguments:
|
||
|
// pcodec = pointer to a qra64codec data structure as returned by qra64_init
|
||
|
// ebno = pointer to a float where the avg Eb/No (in dB) will be stored
|
||
|
// in case of successfull decoding
|
||
|
// (pass a null pointer if not interested)
|
||
|
// x = pointer to decoded message, int x[12]
|
||
|
// r = pointer to received symbol energies (squared amplitudes)
|
||
|
// r must point to an array of length QRA64_M*QRA64_N (=64*63=4032)
|
||
|
// The first QRA_M entries should be the energies of the first
|
||
|
// symbol in the codeword; the last QRA_M entries should be the
|
||
|
// energies of the last symbol in the codeword
|
||
|
//
|
||
|
// return code:
|
||
|
//
|
||
|
// The return code is <0 when decoding is unsuccessful
|
||
|
// -16 indicates that the definition of QRA64_NMSG does not match what required by the code
|
||
|
// If the decoding process is successfull the return code is accordingly to the following table
|
||
|
// rc=0 [? ? ?] AP0 (decoding with no a-priori)
|
||
|
// rc=1 [CQ ? ?] AP27
|
||
|
// rc=2 [CQ ? ] AP44
|
||
|
// rc=3 [CALL ? ?] AP29
|
||
|
// rc=4 [CALL ? ] AP45
|
||
|
// rc=5 [CALL CALL ?] AP57
|
||
|
// rc=6 [? CALL ?] AP29
|
||
|
// rc=7 [? CALL ] AP45
|
||
|
// rc=8 [CALL CALL GRID] AP72 (actually a AP68 mask to reduce false decodes)
|
||
|
// rc=9 [CQ CALL ?] AP55
|
||
|
// rc=10 [CQ CALL ] AP70 (actaully a AP68 mask to reduce false decodes)
|
||
|
|
||
|
// return codes in the range 1-10 indicate the amount and the type of a-priori
|
||
|
// information was required to decode the received message.
|
||
|
|
||
|
|
||
|
// Decode a QRA64 msg using a fast-fading metric
|
||
|
int qra64_decode_fastfading(
|
||
|
qra64codec *pcodec, // ptr to the codec structure
|
||
|
float *ebno, // ptr to where the estimated Eb/No value will be saved
|
||
|
int *x, // ptr to decoded message
|
||
|
const float *rxen, // ptr to received symbol energies array
|
||
|
const int submode, // submode idx (0=QRA64A ... 4=QRA64E)
|
||
|
const float B90, // spread bandwidth (90% fractional energy)
|
||
|
const int fadingModel); // 0=Gaussian 1=Lorentzian fade model
|
||
|
//
|
||
|
// rxen: The array of the received bin energies
|
||
|
// Bins must be spaced by integer multiples of the symbol rate (1/Ts Hz)
|
||
|
// The array must be an array of total length U = L x N where:
|
||
|
// L: is the number of frequency bins per message symbol (see after)
|
||
|
// N: is the number of symbols in a QRA64 msg (63)
|
||
|
//
|
||
|
// The number of bins/symbol L depends on the selected submode accordingly to
|
||
|
// the following rule:
|
||
|
// L = (64+64*2^submode+64) = 64*(2+2^submode)
|
||
|
// Tone 0 is always supposed to be at offset 64 in the array.
|
||
|
// The m-th tone nominal frequency is located at offset 64 + m*2^submode (m=0..63)
|
||
|
//
|
||
|
// Submode A: (2^submode = 1)
|
||
|
// L = 64*3 = 196 bins/symbol
|
||
|
// Total length of the energies array: U = 192*63 = 12096 floats
|
||
|
//
|
||
|
// Submode B: (2^submode = 2)
|
||
|
// L = 64*4 = 256 bins/symbol
|
||
|
// Total length of the energies array: U = 256*63 = 16128 floats
|
||
|
//
|
||
|
// Submode C: (2^submode = 4)
|
||
|
// L = 64*6 = 384 bins/symbol
|
||
|
// Total length of the energies array: U = 384*63 = 24192 floats
|
||
|
//
|
||
|
// Submode D: (2^submode = 8)
|
||
|
// L = 64*10 = 640 bins/symbol
|
||
|
// Total length of the energies array: U = 640*63 = 40320 floats
|
||
|
//
|
||
|
// Submode E: (2^submode = 16)
|
||
|
// L = 64*18 = 1152 bins/symbol
|
||
|
// Total length of the energies array: U = 1152*63 = 72576 floats
|
||
|
//
|
||
|
// Note: The rxen array is modified and reused for internal calculations.
|
||
|
//
|
||
|
//
|
||
|
// B90: spread fading bandwidth in Hz (90% fractional average energy)
|
||
|
//
|
||
|
// B90 should be in the range 1 Hz ... 238 Hz
|
||
|
// The value passed to the call is rounded to the closest value among the
|
||
|
// 64 available values:
|
||
|
// B = 1.09^k Hz, with k=0,1,...,63
|
||
|
//
|
||
|
// I.e. B90=27 Hz will be approximated in this way:
|
||
|
// k = rnd(log(27)/log(1.09)) = 38
|
||
|
// B90 = 1.09^k = 1.09^38 = 26.4 Hz
|
||
|
//
|
||
|
// For any input value the maximum rounding error is not larger than +/- 5%
|
||
|
//
|
||
|
// return codes: same return codes of qra64_decode (+some additional error codes)
|
||
|
|
||
|
|
||
|
// Simulate the fast-fading channel (to be used with qra64_decode_fastfading)
|
||
|
int qra64_fastfading_channel(
|
||
|
float **rxen,
|
||
|
const int *xmsg,
|
||
|
const int submode,
|
||
|
const float EbN0dB,
|
||
|
const float B90,
|
||
|
const int fadingModel);
|
||
|
// Simulate transmission over a fading channel with given B90, fading model and submode
|
||
|
// and non coherent detection.
|
||
|
// Sets rxen to point to an array of bin energies formatted as required
|
||
|
// by the (fast-fading) decoding routine.
|
||
|
// returns 0 on success or negative values on error conditions
|
||
|
|
||
|
|
||
|
int qra64_apset(qra64codec *pcodec, const int mycall, const int hiscall, const int grid, const int aptype);
|
||
|
// Set decoder a-priori knowledge accordingly to the type of the message to
|
||
|
// look up for
|
||
|
// arguments:
|
||
|
// pcodec = pointer to a qra64codec data structure as returned by qra64_init
|
||
|
// mycall = mycall to look for
|
||
|
// hiscall = hiscall to look for
|
||
|
// grid = grid to look for
|
||
|
// aptype = define the type of AP to be set:
|
||
|
// APTYPE_CQQRZ set [cq/qrz ? ?/blank]
|
||
|
// APTYPE_MYCALL set [mycall ? ?/blank]
|
||
|
// APTYPE_HISCALL set [? hiscall ?/blank]
|
||
|
// APTYPE_BOTHCALLS set [mycall hiscall ?]
|
||
|
// APTYPE_FULL set [mycall hiscall grid]
|
||
|
// APTYPE_CQHISCALL set [cq/qrz hiscall ?/blank]
|
||
|
|
||
|
// returns:
|
||
|
// 0 on success
|
||
|
// -1 when qra64_init was called with the QRA_NOAP flag
|
||
|
// -2 invalid apytpe (valid range [APTYPE_CQQRZ..APTYPE_CQHISCALL]
|
||
|
// (APTYPE_CQQRZ [cq/qrz ? ?] is set by default )
|
||
|
|
||
|
void qra64_apdisable(qra64codec *pcodec, const int aptype);
|
||
|
// disable specific AP type
|
||
|
// arguments:
|
||
|
// pcodec = pointer to a qra64codec data structure as returned by qra64_init
|
||
|
// aptype = define the type of AP to be disabled
|
||
|
// APTYPE_CQQRZ disable [cq/qrz ? ?/blank]
|
||
|
// APTYPE_MYCALL disable [mycall ? ?/blank]
|
||
|
// APTYPE_HISCALL disable [ ? hiscall ?/blank]
|
||
|
// APTYPE_BOTHCALLS disable [mycall hiscall ? ]
|
||
|
// APTYPE_FULL disable [mycall hiscall grid]
|
||
|
// APTYPE_CQHISCALL set [cq/qrz hiscall ?/blank]
|
||
|
|
||
|
void qra64_close(qra64codec *pcodec);
|
||
|
// Free memory allocated by qra64_init
|
||
|
// arguments:
|
||
|
// pcodec = pointer to a qra64codec data structure as returned by qra64_init
|
||
|
|
||
|
// ----------------------------------------------------------------------------
|
||
|
|
||
|
// encode/decode std msgs in 12 symbols as done in jt65
|
||
|
void encodemsg_jt65(int *y, const int call1, const int call2, const int grid);
|
||
|
void decodemsg_jt65(int *call1, int *call2, int *grid, const int *x);
|
||
|
|
||
|
#ifdef __cplusplus
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
#endif // _qra64_h_
|