2014-12-03 01:10:06 +01:00
|
|
|
#include <string.h>
|
|
|
|
#include "AFSK.h"
|
|
|
|
#include "util/time.h"
|
2019-01-12 15:12:51 +01:00
|
|
|
#include "hardware/LED.h"
|
2019-01-08 20:56:58 +01:00
|
|
|
#include "protocol/KISS.h"
|
2019-01-29 16:41:27 +01:00
|
|
|
#include "hardware/SD.h"
|
2019-02-08 13:18:49 +01:00
|
|
|
#include "util/Config.h"
|
2018-12-27 20:24:21 +01:00
|
|
|
|
2019-01-12 16:30:26 +01:00
|
|
|
extern volatile ticks_t _clock;
|
2019-11-04 11:00:31 +01:00
|
|
|
extern volatile uint32_t _rtc_seconds;
|
|
|
|
extern volatile uint16_t _rtc_seconds_accu;
|
2014-12-03 01:10:06 +01:00
|
|
|
|
|
|
|
bool hw_afsk_dac_isr = false;
|
2014-12-18 23:45:36 +01:00
|
|
|
bool hw_5v_ref = false;
|
2014-12-03 01:10:06 +01:00
|
|
|
Afsk *AFSK_modem;
|
|
|
|
|
2019-02-08 15:48:13 +01:00
|
|
|
int8_t afsk_peak = 0;
|
|
|
|
uint16_t peak_ticks = 0;
|
|
|
|
|
2014-12-03 01:10:06 +01:00
|
|
|
// Forward declerations
|
2018-04-24 14:38:48 +02:00
|
|
|
int afsk_getchar(FILE *strem);
|
|
|
|
int afsk_putchar(char c, FILE *stream);
|
2014-12-03 01:10:06 +01:00
|
|
|
|
2018-12-27 20:24:21 +01:00
|
|
|
// ADC and clock setup
|
2014-12-03 01:10:06 +01:00
|
|
|
void AFSK_hw_init(void) {
|
2019-01-12 16:30:26 +01:00
|
|
|
_clock = 0;
|
|
|
|
|
2019-01-05 13:47:46 +01:00
|
|
|
// Run ADC initialisation
|
|
|
|
AFSK_adc_init();
|
|
|
|
|
|
|
|
// Run DAC initialisation
|
|
|
|
AFSK_dac_init();
|
|
|
|
}
|
|
|
|
|
|
|
|
void AFSK_dac_init(void) {
|
|
|
|
// DAC uses all 8 pins of one port,
|
|
|
|
// so set all to output
|
|
|
|
DAC_DDR |= 0xFF;
|
|
|
|
|
|
|
|
// Set Timer3 to normal operation
|
|
|
|
TCCR3A = 0;
|
|
|
|
TCCR3B = _BV(CS10) |
|
|
|
|
_BV(WGM33)|
|
|
|
|
_BV(WGM32);
|
|
|
|
|
|
|
|
ICR3 = DAC_TICKS_BETWEEN_SAMPLES;
|
|
|
|
|
|
|
|
TIMSK3 = _BV(ICIE3);
|
|
|
|
|
2019-01-27 20:25:11 +01:00
|
|
|
PTT_DDR |= _BV(PTT_PIN);
|
|
|
|
PTT_DDR |= _BV(PTT_NEG_PIN);
|
|
|
|
|
2019-01-05 13:47:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void AFSK_adc_init(void) {
|
2019-11-04 11:00:31 +01:00
|
|
|
_clock = 0;
|
|
|
|
_rtc_seconds = 0;
|
|
|
|
_rtc_seconds_accu = 0;
|
|
|
|
|
2018-12-27 20:24:21 +01:00
|
|
|
// Set Timer1 to normal operation
|
|
|
|
TCCR1A = 0;
|
2014-12-18 23:45:36 +01:00
|
|
|
|
2018-12-27 20:24:21 +01:00
|
|
|
TCCR1B = _BV(WGM13) | // Enable Timer1 Waveform Generation Mode 12:
|
|
|
|
_BV(WGM12) | // Mode = CTC, TOP = ICR1
|
|
|
|
_BV(CS10); // Set clock source to 0b001 = System clock without prescaling
|
2014-12-03 01:10:06 +01:00
|
|
|
|
2018-12-27 20:24:21 +01:00
|
|
|
// Set ICR1 register to the amount of ticks needed between
|
|
|
|
// each sample capture/synthesis
|
2019-01-05 13:47:46 +01:00
|
|
|
ICR1 = ADC_TICKS_BETWEEN_SAMPLES;
|
2018-12-27 20:24:21 +01:00
|
|
|
|
|
|
|
// Set ADMUX register to use external AREF, channel ADC0
|
|
|
|
// and left adjust result
|
|
|
|
ADMUX = _BV(ADLAR) | 0;
|
2014-12-18 23:45:36 +01:00
|
|
|
|
2018-12-27 20:24:21 +01:00
|
|
|
// Set ADC port directions and outputs
|
|
|
|
// TODO: Check this
|
|
|
|
ADC_DDR &= ~_BV(0); // 0b11111110 - All pins are outputs, except ADC0
|
|
|
|
ADC_PORT &= 0x00; // 0b00000000 - All pins are at GND level
|
|
|
|
|
|
|
|
// Set Digital Input Disable Register mask to 0b00000001,
|
|
|
|
// which disables the input buffer on ADC0 pin to avoid
|
|
|
|
// current through the pin.
|
2015-11-17 08:09:39 +01:00
|
|
|
DIDR0 |= _BV(0);
|
2018-12-27 20:24:21 +01:00
|
|
|
|
|
|
|
|
|
|
|
ADCSRB = _BV(ADTS2) |
|
2014-12-03 01:10:06 +01:00
|
|
|
_BV(ADTS1) |
|
2018-12-27 20:24:21 +01:00
|
|
|
_BV(ADTS0); // Set ADC Trigger Source to 0b111 = Timer1 Capture Event
|
|
|
|
|
|
|
|
ADCSRA = _BV(ADEN) | // ADC Enable
|
|
|
|
_BV(ADSC) | // ADC Start Conversion
|
|
|
|
_BV(ADATE)| // ADC Interrupt Flag
|
|
|
|
_BV(ADIE) | // ADC Interrupt Enable
|
|
|
|
_BV(ADPS0)|
|
|
|
|
_BV(ADPS2); // Set ADC prescaler bits to 0b101 = 32
|
2019-01-01 20:34:02 +01:00
|
|
|
// At 20MHz, this gives an ADC clock of 625 KHz
|
2014-12-03 01:10:06 +01:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
void AFSK_init(Afsk *afsk) {
|
|
|
|
// Allocate modem struct memory
|
|
|
|
memset(afsk, 0, sizeof(*afsk));
|
|
|
|
AFSK_modem = afsk;
|
|
|
|
// Set phase increment
|
|
|
|
afsk->phaseInc = MARK_INC;
|
2018-04-24 17:34:59 +02:00
|
|
|
afsk->silentSamples = 0;
|
|
|
|
|
2014-12-03 01:10:06 +01:00
|
|
|
// Initialise FIFO buffers
|
2020-06-01 15:25:29 +02:00
|
|
|
fifo_init(&afsk->delayFifo, (uint8_t *)afsk->delayBuf, sizeof(afsk->delayBuf)-1);
|
|
|
|
fifo_init(&afsk->rxFifo, afsk->rxBuf, CONFIG_AFSK_RX_BUFLEN);
|
|
|
|
fifo_init(&afsk->txFifo, afsk->txBuf, CONFIG_AFSK_TX_BUFLEN);
|
2014-12-03 01:10:06 +01:00
|
|
|
|
2014-12-04 15:22:25 +01:00
|
|
|
// Fill delay FIFO with zeroes
|
2020-06-01 15:25:29 +02:00
|
|
|
while (fifo_isfull(&afsk->delayFifo)) {
|
|
|
|
fifo_push_locked(&afsk->delayFifo, 0);
|
2014-12-04 15:22:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
AFSK_hw_init();
|
|
|
|
|
2014-12-03 01:10:06 +01:00
|
|
|
// Set up streams
|
|
|
|
FILE afsk_fd = FDEV_SETUP_STREAM(afsk_putchar, afsk_getchar, _FDEV_SETUP_RW);
|
|
|
|
afsk->fd = afsk_fd;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void AFSK_txStart(Afsk *afsk) {
|
|
|
|
if (!afsk->sending) {
|
|
|
|
afsk->phaseInc = MARK_INC;
|
|
|
|
afsk->phaseAcc = 0;
|
|
|
|
afsk->bitstuffCount = 0;
|
|
|
|
afsk->sending = true;
|
2018-04-24 14:34:59 +02:00
|
|
|
afsk->sending_data = true;
|
2019-11-03 20:50:55 +01:00
|
|
|
LED_RX_OFF();
|
2014-12-03 01:10:06 +01:00
|
|
|
LED_TX_ON();
|
2019-02-08 13:18:49 +01:00
|
|
|
afsk->preambleLength = DIV_ROUND(config_preamble * BITRATE, 8000);
|
2014-12-03 01:10:06 +01:00
|
|
|
AFSK_DAC_IRQ_START();
|
|
|
|
}
|
|
|
|
ATOMIC_BLOCK(ATOMIC_RESTORESTATE) {
|
2019-02-08 13:18:49 +01:00
|
|
|
afsk->tailLength = DIV_ROUND(config_tail * BITRATE, 8000);
|
2014-12-03 01:10:06 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-24 14:38:48 +02:00
|
|
|
int afsk_putchar(char c, FILE *stream) {
|
2014-12-03 01:10:06 +01:00
|
|
|
AFSK_txStart(AFSK_modem);
|
|
|
|
while(fifo_isfull_locked(&AFSK_modem->txFifo)) { /* Wait */ }
|
|
|
|
fifo_push_locked(&AFSK_modem->txFifo, c);
|
2018-04-24 14:38:48 +02:00
|
|
|
return 1;
|
2014-12-03 01:10:06 +01:00
|
|
|
}
|
|
|
|
|
2018-04-24 14:38:48 +02:00
|
|
|
int afsk_getchar(FILE *stream) {
|
2014-12-03 01:10:06 +01:00
|
|
|
if (fifo_isempty_locked(&AFSK_modem->rxFifo)) {
|
|
|
|
return EOF;
|
|
|
|
} else {
|
|
|
|
return fifo_pop_locked(&AFSK_modem->rxFifo);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t AFSK_dac_isr(Afsk *afsk) {
|
|
|
|
if (afsk->sampleIndex == 0) {
|
|
|
|
if (afsk->txBit == 0) {
|
|
|
|
if (fifo_isempty(&afsk->txFifo) && afsk->tailLength == 0) {
|
|
|
|
AFSK_DAC_IRQ_STOP();
|
|
|
|
afsk->sending = false;
|
2018-04-24 14:34:59 +02:00
|
|
|
afsk->sending_data = false;
|
2014-12-03 01:10:06 +01:00
|
|
|
LED_TX_OFF();
|
|
|
|
return 0;
|
|
|
|
} else {
|
|
|
|
if (!afsk->bitStuff) afsk->bitstuffCount = 0;
|
|
|
|
afsk->bitStuff = true;
|
|
|
|
if (afsk->preambleLength == 0) {
|
|
|
|
if (fifo_isempty(&afsk->txFifo)) {
|
2018-04-24 14:34:59 +02:00
|
|
|
afsk->sending_data = false;
|
2014-12-03 01:10:06 +01:00
|
|
|
afsk->tailLength--;
|
|
|
|
afsk->currentOutputByte = HDLC_FLAG;
|
|
|
|
} else {
|
|
|
|
afsk->currentOutputByte = fifo_pop(&afsk->txFifo);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
afsk->preambleLength--;
|
|
|
|
afsk->currentOutputByte = HDLC_FLAG;
|
|
|
|
}
|
|
|
|
if (afsk->currentOutputByte == AX25_ESC) {
|
|
|
|
if (fifo_isempty(&afsk->txFifo)) {
|
|
|
|
AFSK_DAC_IRQ_STOP();
|
|
|
|
afsk->sending = false;
|
|
|
|
LED_TX_OFF();
|
|
|
|
return 0;
|
|
|
|
} else {
|
|
|
|
afsk->currentOutputByte = fifo_pop(&afsk->txFifo);
|
|
|
|
}
|
|
|
|
} else if (afsk->currentOutputByte == HDLC_FLAG || afsk->currentOutputByte == HDLC_RESET) {
|
|
|
|
afsk->bitStuff = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
afsk->txBit = 0x01;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (afsk->bitStuff && afsk->bitstuffCount >= BIT_STUFF_LEN) {
|
|
|
|
afsk->bitstuffCount = 0;
|
|
|
|
afsk->phaseInc = SWITCH_TONE(afsk->phaseInc);
|
|
|
|
} else {
|
|
|
|
if (afsk->currentOutputByte & afsk->txBit) {
|
|
|
|
afsk->bitstuffCount++;
|
|
|
|
} else {
|
|
|
|
afsk->bitstuffCount = 0;
|
|
|
|
afsk->phaseInc = SWITCH_TONE(afsk->phaseInc);
|
|
|
|
}
|
|
|
|
afsk->txBit <<= 1;
|
|
|
|
}
|
|
|
|
|
2019-01-05 13:47:46 +01:00
|
|
|
afsk->sampleIndex = DAC_SAMPLESPERBIT;
|
2014-12-03 01:10:06 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
afsk->phaseAcc += afsk->phaseInc;
|
|
|
|
afsk->phaseAcc %= SIN_LEN;
|
|
|
|
afsk->sampleIndex--;
|
|
|
|
|
|
|
|
return sinSample(afsk->phaseAcc);
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool hdlcParse(Hdlc *hdlc, bool bit, FIFOBuffer *fifo) {
|
|
|
|
// Initialise a return value. We start with the
|
|
|
|
// assumption that all is going to end well :)
|
|
|
|
bool ret = true;
|
|
|
|
|
|
|
|
// Bitshift our byte of demodulated bits to
|
|
|
|
// the left by one bit, to make room for the
|
|
|
|
// next incoming bit
|
|
|
|
hdlc->demodulatedBits <<= 1;
|
|
|
|
// And then put the newest bit from the
|
|
|
|
// demodulator into the byte.
|
|
|
|
hdlc->demodulatedBits |= bit ? 1 : 0;
|
|
|
|
|
|
|
|
// Now we'll look at the last 8 received bits, and
|
|
|
|
// check if we have received a HDLC flag (01111110)
|
|
|
|
if (hdlc->demodulatedBits == HDLC_FLAG) {
|
|
|
|
// If we have, check that our output buffer is
|
|
|
|
// not full.
|
|
|
|
if (!fifo_isfull(fifo)) {
|
|
|
|
// If it isn't, we'll push the HDLC_FLAG into
|
|
|
|
// the buffer and indicate that we are now
|
|
|
|
// receiving data. For bling we also turn
|
|
|
|
// on the RX LED.
|
|
|
|
fifo_push(fifo, HDLC_FLAG);
|
|
|
|
hdlc->receiving = true;
|
2018-04-24 15:34:50 +02:00
|
|
|
|
|
|
|
if (hdlc->dcd_count < DCD_MIN_COUNT) {
|
|
|
|
hdlc->dcd = false;
|
|
|
|
hdlc->dcd_count++;
|
|
|
|
} else {
|
|
|
|
hdlc->dcd = true;
|
|
|
|
}
|
|
|
|
|
2014-12-19 00:50:14 +01:00
|
|
|
#if OPEN_SQUELCH == false
|
|
|
|
LED_RX_ON();
|
|
|
|
#endif
|
2014-12-03 01:10:06 +01:00
|
|
|
} else {
|
|
|
|
// If the buffer is full, we have a problem
|
2014-12-04 15:22:25 +01:00
|
|
|
// and abort by setting the return value to
|
2014-12-03 01:10:06 +01:00
|
|
|
// false and stopping the here.
|
2014-12-04 15:22:25 +01:00
|
|
|
|
2014-12-03 01:10:06 +01:00
|
|
|
ret = false;
|
|
|
|
hdlc->receiving = false;
|
2018-04-24 15:34:50 +02:00
|
|
|
hdlc->dcd = false;
|
|
|
|
hdlc->dcd_count = 0;
|
2014-12-03 01:10:06 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Everytime we receive a HDLC_FLAG, we reset the
|
|
|
|
// storage for our current incoming byte and bit
|
|
|
|
// position in that byte. This effectively
|
|
|
|
// synchronises our parsing to the start and end
|
|
|
|
// of the received bytes.
|
|
|
|
hdlc->currentByte = 0;
|
|
|
|
hdlc->bitIndex = 0;
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check if we have received a RESET flag (01111111)
|
|
|
|
// In this comparison we also detect when no transmission
|
|
|
|
// (or silence) is taking place, and the demodulator
|
2018-11-28 21:18:54 +01:00
|
|
|
// returns an endless stream of zeroes. Due to the NRZ-S
|
2018-12-29 15:57:49 +01:00
|
|
|
// coding, the actual bits sent to this function will
|
2014-12-03 01:10:06 +01:00
|
|
|
// be an endless stream of ones, which this AND operation
|
|
|
|
// will also detect.
|
|
|
|
if ((hdlc->demodulatedBits & HDLC_RESET) == HDLC_RESET) {
|
|
|
|
// If we have, something probably went wrong at the
|
|
|
|
// transmitting end, and we abort the reception.
|
|
|
|
hdlc->receiving = false;
|
2018-04-24 15:34:50 +02:00
|
|
|
hdlc->dcd = false;
|
|
|
|
hdlc->dcd_count = 0;
|
2014-12-03 01:10:06 +01:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2018-04-24 15:34:50 +02:00
|
|
|
// Check the DCD status and set RX LED appropriately
|
|
|
|
if (hdlc->dcd) {
|
|
|
|
LED_RX_ON();
|
|
|
|
} else {
|
|
|
|
LED_RX_OFF();
|
|
|
|
}
|
|
|
|
|
2014-12-03 01:10:06 +01:00
|
|
|
// If we have not yet seen a HDLC_FLAG indicating that
|
|
|
|
// a transmission is actually taking place, don't bother
|
|
|
|
// with anything.
|
2018-04-24 17:34:59 +02:00
|
|
|
if (!hdlc->receiving) {
|
|
|
|
hdlc->dcd = false;
|
|
|
|
hdlc->dcd_count = 0;
|
|
|
|
|
2014-12-03 01:10:06 +01:00
|
|
|
return ret;
|
2018-04-24 17:34:59 +02:00
|
|
|
}
|
2014-12-03 01:10:06 +01:00
|
|
|
|
|
|
|
// First check if what we are seeing is a stuffed bit.
|
|
|
|
// Since the different HDLC control characters like
|
|
|
|
// HDLC_FLAG, HDLC_RESET and such could also occur in
|
|
|
|
// a normal data stream, we employ a method known as
|
|
|
|
// "bit stuffing". All control characters have more than
|
|
|
|
// 5 ones in a row, so if the transmitting party detects
|
|
|
|
// this sequence in the _data_ to be transmitted, it inserts
|
|
|
|
// a zero to avoid the receiving party interpreting it as
|
|
|
|
// a control character. Therefore, if we detect such a
|
|
|
|
// "stuffed bit", we simply ignore it and wait for the
|
|
|
|
// next bit to come in.
|
|
|
|
//
|
|
|
|
// We do the detection by applying an AND bit-mask to the
|
|
|
|
// stream of demodulated bits. This mask is 00111111 (0x3f)
|
|
|
|
// if the result of the operation is 00111110 (0x3e), we
|
|
|
|
// have detected a stuffed bit.
|
|
|
|
if ((hdlc->demodulatedBits & 0x3f) == 0x3e)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
// If we have an actual 1 bit, push this to the current byte
|
|
|
|
// If it's a zero, we don't need to do anything, since the
|
|
|
|
// bit is initialized to zero when we bitshifted earlier.
|
|
|
|
if (hdlc->demodulatedBits & 0x01)
|
|
|
|
hdlc->currentByte |= 0x80;
|
|
|
|
|
|
|
|
// Increment the bitIndex and check if we have a complete byte
|
|
|
|
if (++hdlc->bitIndex >= 8) {
|
|
|
|
// If we have a HDLC control character, put a AX.25 escape
|
|
|
|
// in the received data. We know we need to do this,
|
|
|
|
// because at this point we must have already seen a HDLC
|
|
|
|
// flag, meaning that this control character is the result
|
|
|
|
// of a bitstuffed byte that is equal to said control
|
|
|
|
// character, but is actually part of the data stream.
|
|
|
|
// By inserting the escape character, we tell the protocol
|
|
|
|
// layer that this is not an actual control character, but
|
|
|
|
// data.
|
|
|
|
if ((hdlc->currentByte == HDLC_FLAG ||
|
|
|
|
hdlc->currentByte == HDLC_RESET ||
|
|
|
|
hdlc->currentByte == AX25_ESC)) {
|
|
|
|
// We also need to check that our received data buffer
|
|
|
|
// is not full before putting more data in
|
|
|
|
if (!fifo_isfull(fifo)) {
|
|
|
|
fifo_push(fifo, AX25_ESC);
|
|
|
|
} else {
|
|
|
|
// If it is, abort and return false
|
|
|
|
hdlc->receiving = false;
|
2018-04-24 17:34:59 +02:00
|
|
|
hdlc->dcd = false;
|
|
|
|
hdlc->dcd_count = 0;
|
2014-12-03 01:10:06 +01:00
|
|
|
LED_RX_OFF();
|
|
|
|
ret = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Push the actual byte to the received data FIFO,
|
|
|
|
// if it isn't full.
|
|
|
|
if (!fifo_isfull(fifo)) {
|
|
|
|
fifo_push(fifo, hdlc->currentByte);
|
|
|
|
} else {
|
|
|
|
// If it is, well, you know by now!
|
|
|
|
hdlc->receiving = false;
|
2018-04-24 17:34:59 +02:00
|
|
|
hdlc->dcd = false;
|
|
|
|
hdlc->dcd_count = 0;
|
2014-12-03 01:10:06 +01:00
|
|
|
LED_RX_OFF();
|
|
|
|
ret = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Wipe received byte and reset bit index to 0
|
|
|
|
hdlc->currentByte = 0;
|
|
|
|
hdlc->bitIndex = 0;
|
|
|
|
|
|
|
|
} else {
|
|
|
|
// We don't have a full byte yet, bitshift the byte
|
|
|
|
// to make room for the next bit
|
|
|
|
hdlc->currentByte >>= 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2019-02-08 15:48:13 +01:00
|
|
|
#define AFSK_PEAK_DECAY 96
|
2014-12-03 01:10:06 +01:00
|
|
|
void AFSK_adc_isr(Afsk *afsk, int8_t currentSample) {
|
2019-02-08 15:48:13 +01:00
|
|
|
if (config_output_diagnostics) {
|
|
|
|
peak_ticks++;
|
|
|
|
if (currentSample > afsk_peak) {
|
|
|
|
afsk_peak = currentSample;
|
|
|
|
} else {
|
|
|
|
if (peak_ticks >= AFSK_PEAK_DECAY) {
|
|
|
|
peak_ticks = 0;
|
|
|
|
if (afsk_peak > 0) {
|
|
|
|
afsk_peak--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-12-03 01:10:06 +01:00
|
|
|
// To determine the received frequency, and thereby
|
|
|
|
// the bit of the sample, we multiply the sample by
|
|
|
|
// a sample delayed by (samples per bit / 2).
|
|
|
|
// We then lowpass-filter the samples with a
|
|
|
|
// Chebyshev filter. The lowpass filtering serves
|
|
|
|
// to "smooth out" the variations in the samples.
|
|
|
|
|
|
|
|
afsk->iirX[0] = afsk->iirX[1];
|
2015-04-24 10:47:50 +02:00
|
|
|
|
2019-01-07 13:41:49 +01:00
|
|
|
#if CONFIG_ADC_SAMPLERATE == 4800
|
|
|
|
#if FILTER_CUTOFF == 600
|
|
|
|
#define IIR_GAIN 2 // Really 2.228465666
|
|
|
|
#define IIR_POLE 10 // Really Y[0] * 0.1025215106
|
|
|
|
afsk->iirX[1] = ((int8_t)fifo_pop(&afsk->delayFifo) * currentSample) / IIR_GAIN;
|
|
|
|
afsk->iirY[0] = afsk->iirY[1];
|
|
|
|
afsk->iirY[1] = afsk->iirX[0] + afsk->iirX[1] + (afsk->iirY[0] / IIR_POLE);
|
2019-04-11 16:29:32 +02:00
|
|
|
#elif FILTER_CUTOFF == 155
|
|
|
|
#define IIR_GAIN 6 // Really 5.99865959
|
|
|
|
#define IIR_POLE 2 // Really Y[0] * 0.6665921828
|
|
|
|
afsk->iirX[1] = ((int8_t)fifo_pop(&afsk->delayFifo) * currentSample) / IIR_GAIN;
|
|
|
|
afsk->iirY[0] = afsk->iirY[1];
|
|
|
|
afsk->iirY[1] = afsk->iirX[0] + afsk->iirX[1] + (afsk->iirY[0] / IIR_POLE);
|
|
|
|
#elif FILTER_CUTOFF == 100
|
|
|
|
#define IIR_GAIN 9 // Really 8.763507115
|
|
|
|
#define IIR_POLE 0.77 // Really Y[0] * 0.7717808665
|
|
|
|
afsk->iirX[1] = ((int8_t)fifo_pop(&afsk->delayFifo) * currentSample) / IIR_GAIN;
|
|
|
|
afsk->iirY[0] = afsk->iirY[1];
|
|
|
|
afsk->iirY[1] = afsk->iirX[0] + afsk->iirX[1] + (afsk->iirY[0] * IIR_POLE);
|
2019-01-07 13:41:49 +01:00
|
|
|
#else
|
|
|
|
#error Unsupported filter cutoff!
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#elif CONFIG_ADC_SAMPLERATE == 9600
|
2018-12-31 13:24:28 +01:00
|
|
|
#if FILTER_CUTOFF == 500
|
|
|
|
#define IIR_GAIN 4 // Really 4.082041675
|
|
|
|
#define IIR_POLE 2 // Really Y[0] * 0.5100490981
|
|
|
|
afsk->iirX[1] = ((int8_t)fifo_pop(&afsk->delayFifo) * currentSample) / IIR_GAIN;
|
|
|
|
afsk->iirY[0] = afsk->iirY[1];
|
2019-01-07 13:41:49 +01:00
|
|
|
afsk->iirY[1] = afsk->iirX[0] + afsk->iirX[1] + (afsk->iirY[0] / IIR_POLE);
|
2019-04-11 16:29:32 +02:00
|
|
|
#elif FILTER_CUTOFF == 200
|
|
|
|
#define IIR_GAIN 9 // Really 8.763507115
|
|
|
|
#define IIR_POLE 2 // Really Y[0] * 0.7717808665
|
|
|
|
afsk->iirX[1] = ((int8_t)fifo_pop(&afsk->delayFifo) * currentSample) / IIR_GAIN;
|
|
|
|
afsk->iirY[0] = afsk->iirY[1];
|
|
|
|
afsk->iirY[1] = afsk->iirX[0] + afsk->iirX[1] + ((afsk->iirY[0] / 4) * 3);
|
2018-12-30 00:32:19 +01:00
|
|
|
#else
|
|
|
|
#error Unsupported filter cutoff!
|
|
|
|
#endif
|
2018-12-31 13:24:28 +01:00
|
|
|
|
2019-01-05 13:47:46 +01:00
|
|
|
#elif CONFIG_ADC_SAMPLERATE == 19200
|
2018-12-31 13:24:28 +01:00
|
|
|
#if FILTER_CUTOFF == 150
|
|
|
|
#define IIR_GAIN 2 // Really 2.172813446e
|
|
|
|
#define IIR_POLE 2 // Really Y[0] * 0.9079534415
|
|
|
|
afsk->iirX[1] = ((int8_t)fifo_pop(&afsk->delayFifo) * currentSample) / IIR_GAIN;
|
|
|
|
afsk->iirY[0] = afsk->iirY[1];
|
|
|
|
afsk->iirY[1] = afsk->iirX[0] + afsk->iirX[1] + (afsk->iirY[0] / IIR_POLE);
|
|
|
|
|
|
|
|
#elif FILTER_CUTOFF == 500
|
|
|
|
#define IIR_GAIN 7 // Really 5.006847792
|
|
|
|
#define IIR_POLE 2 // Really Y[0] * 0.6005470741
|
|
|
|
afsk->iirX[1] = ((int8_t)fifo_pop(&afsk->delayFifo) * currentSample) / IIR_GAIN;
|
|
|
|
afsk->iirY[0] = afsk->iirY[1];
|
|
|
|
afsk->iirY[1] = afsk->iirX[0] + afsk->iirX[1] + (afsk->iirY[0] / IIR_POLE);
|
|
|
|
|
|
|
|
#elif FILTER_CUTOFF == 600
|
|
|
|
#define IIR_GAIN 6 // Really 6.166411713
|
|
|
|
#define IIR_POLE 2 // Really Y[0] * 0.6756622663
|
|
|
|
afsk->iirX[1] = ((int8_t)fifo_pop(&afsk->delayFifo) * currentSample) / IIR_GAIN;
|
|
|
|
afsk->iirY[0] = afsk->iirY[1];
|
|
|
|
afsk->iirY[1] = afsk->iirX[0] + afsk->iirX[1] + (afsk->iirY[0] / IIR_POLE);
|
|
|
|
|
|
|
|
#elif FILTER_CUTOFF == 772
|
|
|
|
#define IIR_GAIN 5 // Really 5.006847792
|
|
|
|
#define IIR_POLE 2 // Really Y[0] * 0.6005470741
|
|
|
|
afsk->iirX[1] = ((int8_t)fifo_pop(&afsk->delayFifo) * currentSample) / IIR_GAIN;
|
|
|
|
afsk->iirY[0] = afsk->iirY[1];
|
|
|
|
afsk->iirY[1] = afsk->iirX[0] + afsk->iirX[1] + (afsk->iirY[0] / IIR_POLE);
|
|
|
|
|
|
|
|
#elif FILTER_CUTOFF == 1000
|
|
|
|
#define IIR_GAIN 4 // Really 4.082041675
|
|
|
|
#define IIR_POLE 2 // Really Y[0] * 0.5100490981
|
|
|
|
afsk->iirX[1] = ((int8_t)fifo_pop(&afsk->delayFifo) * currentSample) / IIR_GAIN;
|
|
|
|
afsk->iirY[0] = afsk->iirY[1];
|
|
|
|
afsk->iirY[1] = afsk->iirX[0] + afsk->iirX[1] + (afsk->iirY[0] / IIR_POLE);
|
|
|
|
|
|
|
|
#elif FILTER_CUTOFF == 1400
|
|
|
|
#define IIR_GAIN 3 // Really 3.182326364
|
|
|
|
#define IIR_POLE 3 // Really Y[0] * 0.3715289474
|
|
|
|
afsk->iirX[1] = ((int8_t)fifo_pop(&afsk->delayFifo) * currentSample) / IIR_GAIN;
|
|
|
|
afsk->iirY[0] = afsk->iirY[1];
|
|
|
|
afsk->iirY[1] = afsk->iirX[0] + afsk->iirX[1] + (afsk->iirY[0] / IIR_POLE);
|
|
|
|
|
2018-12-30 00:32:19 +01:00
|
|
|
#else
|
|
|
|
#error Unsupported filter cutoff!
|
|
|
|
#endif
|
2015-04-24 10:47:50 +02:00
|
|
|
#else
|
2018-12-31 13:24:28 +01:00
|
|
|
#error No filters defined for specified samplerate!
|
2015-04-24 10:47:50 +02:00
|
|
|
#endif
|
2014-12-03 01:10:06 +01:00
|
|
|
|
|
|
|
afsk->sampledBits <<= 1;
|
2018-12-30 00:32:19 +01:00
|
|
|
|
2015-08-05 21:02:16 +02:00
|
|
|
afsk->sampledBits |= (afsk->iirY[1] > 0) ? 0 : 1;
|
2014-12-03 01:10:06 +01:00
|
|
|
|
|
|
|
fifo_push(&afsk->delayFifo, currentSample);
|
|
|
|
|
|
|
|
if (SIGNAL_TRANSITIONED(afsk->sampledBits)) {
|
|
|
|
if (afsk->currentPhase < PHASE_THRESHOLD) {
|
|
|
|
afsk->currentPhase += PHASE_INC;
|
|
|
|
} else {
|
|
|
|
afsk->currentPhase -= PHASE_INC;
|
|
|
|
}
|
2018-04-24 17:34:59 +02:00
|
|
|
afsk->silentSamples = 0;
|
|
|
|
} else {
|
|
|
|
afsk->silentSamples++;
|
2014-12-03 01:10:06 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
afsk->currentPhase += PHASE_BITS;
|
|
|
|
|
|
|
|
if (afsk->currentPhase >= PHASE_MAX) {
|
|
|
|
afsk->currentPhase %= PHASE_MAX;
|
|
|
|
|
|
|
|
afsk->actualBits <<= 1;
|
2019-04-11 16:29:32 +02:00
|
|
|
|
2014-12-03 01:10:06 +01:00
|
|
|
uint8_t bits = afsk->sampledBits & 0x07;
|
|
|
|
if (bits == 0x07 || // 111
|
|
|
|
bits == 0x06 || // 110
|
|
|
|
bits == 0x05 || // 101
|
|
|
|
bits == 0x03 // 011
|
|
|
|
) {
|
|
|
|
afsk->actualBits |= 1;
|
|
|
|
}
|
|
|
|
|
2019-04-11 16:29:32 +02:00
|
|
|
|
2014-12-03 01:10:06 +01:00
|
|
|
if (!hdlcParse(&afsk->hdlc, !TRANSITION_FOUND(afsk->actualBits), &afsk->rxFifo)) {
|
|
|
|
afsk->status |= 1;
|
|
|
|
if (fifo_isfull(&afsk->rxFifo)) {
|
|
|
|
fifo_flush(&afsk->rxFifo);
|
|
|
|
afsk->status = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-24 17:34:59 +02:00
|
|
|
if (afsk->silentSamples > DCD_TIMEOUT_SAMPLES) {
|
|
|
|
afsk->silentSamples = 0;
|
|
|
|
afsk->hdlc.dcd = false;
|
|
|
|
LED_RX_OFF();
|
|
|
|
}
|
|
|
|
|
2014-12-03 01:10:06 +01:00
|
|
|
}
|
|
|
|
|
2019-01-29 21:58:35 +01:00
|
|
|
uint8_t timed_functions_timer = 0;
|
2019-01-27 20:25:11 +01:00
|
|
|
inline void timed_functions(void) {
|
2019-01-29 21:58:35 +01:00
|
|
|
timed_functions_timer++;
|
|
|
|
if (timed_functions_timer >= CLOCK_TICKS_PER_10_MS) {
|
|
|
|
timed_functions_timer = 0;
|
2019-01-29 16:41:27 +01:00
|
|
|
sd_scheduler();
|
2019-01-27 20:25:11 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-05 13:47:46 +01:00
|
|
|
ISR(TIMER3_CAPT_vect) {
|
2014-12-03 01:10:06 +01:00
|
|
|
if (hw_afsk_dac_isr) {
|
2018-12-27 20:24:21 +01:00
|
|
|
DAC_PORT = AFSK_dac_isr(AFSK_modem);
|
|
|
|
LED_TX_ON();
|
2019-01-27 20:25:11 +01:00
|
|
|
PTT_PORT |= _BV(PTT_PIN);
|
|
|
|
PTT_PORT &= ~_BV(PTT_NEG_PIN);
|
2014-12-03 01:10:06 +01:00
|
|
|
} else {
|
2018-12-29 15:57:49 +01:00
|
|
|
LED_TX_OFF();
|
2019-01-05 13:47:46 +01:00
|
|
|
DAC_PORT = 127;
|
2019-01-27 20:25:11 +01:00
|
|
|
PTT_PORT &= ~_BV(PTT_PIN);
|
|
|
|
PTT_PORT |= _BV(PTT_NEG_PIN);
|
2019-01-05 13:47:46 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ISR(ADC_vect) {
|
|
|
|
TIFR1 = _BV(ICF1);
|
2019-01-08 20:56:58 +01:00
|
|
|
|
2019-01-29 17:52:56 +01:00
|
|
|
if (!hw_afsk_dac_isr) {
|
2019-01-05 13:47:46 +01:00
|
|
|
AFSK_adc_isr(AFSK_modem, (ADCH - 128));
|
2019-01-29 17:52:56 +01:00
|
|
|
timed_functions();
|
2014-12-03 01:10:06 +01:00
|
|
|
}
|
2018-12-27 20:24:21 +01:00
|
|
|
|
2019-01-29 17:52:56 +01:00
|
|
|
update_led_status();
|
2019-01-12 16:30:26 +01:00
|
|
|
|
2014-12-04 15:22:25 +01:00
|
|
|
++_clock;
|
2019-11-04 11:00:31 +01:00
|
|
|
if (++_rtc_seconds_accu >= CLOCK_TICKS_PER_SEC) {
|
|
|
|
_rtc_seconds++;
|
|
|
|
_rtc_seconds_accu = 0;
|
|
|
|
}
|
2015-11-17 08:09:39 +01:00
|
|
|
}
|