OpenModem/Modem/main.c

205 lines
6.9 KiB
C
Raw Normal View History

2014-04-03 16:21:37 -04:00
2014-04-06 14:26:24 -04:00
//////////////////////////////////////////////////////
// First things first, all the includes we need //
//////////////////////////////////////////////////////
2014-05-20 05:20:19 -04:00
#include <cpu/irq.h> // Interrupt functionality from BertOS
2014-04-06 11:17:13 -04:00
2014-05-20 05:20:19 -04:00
#include <drv/ser.h> // Serial driver from BertOS
#include <drv/timer.h> // Timer driver from BertOS
2014-04-06 11:17:13 -04:00
2014-05-20 05:20:19 -04:00
#include <stdio.h> // Standard input/output
#include <string.h> // String operations
#define F_CPU 16000000UL
#include <util/delay.h>
2014-04-03 16:21:37 -04:00
2014-05-29 05:03:25 -04:00
#include <net/ax25.h>
2014-05-20 05:20:19 -04:00
#include "afsk.h" // Header for AFSK modem
2014-04-03 16:21:37 -04:00
#include "protocol/SimpleSerial.h" // Simple serial control protocol
#include "protocol/KISS.h" // KISS TNC protocol
2014-04-28 11:47:32 -04:00
#if SERIAL_DEBUG
2014-05-20 05:20:19 -04:00
#include "cfg/debug.h" // Debug configuration from BertOS
2014-04-28 11:47:32 -04:00
#endif
2014-04-06 14:26:24 -04:00
//////////////////////////////////////////////////////
2014-05-20 05:20:19 -04:00
// A few definitions //
2014-04-06 14:26:24 -04:00
//////////////////////////////////////////////////////
2014-05-20 05:20:19 -04:00
static Afsk afsk; // Declare a AFSK modem struct
2014-05-29 05:47:49 -04:00
static AX25Ctx ax25; // Declare a protocol struct
2014-05-20 05:20:19 -04:00
static Serial ser; // Declare a serial interface struct
2014-04-03 16:21:37 -04:00
2014-05-20 05:20:19 -04:00
#define ADC_CH 0 // Define which channel (pin) we want
// for the ADC (this is A0 on arduino)
2014-04-03 16:21:37 -04:00
2014-11-09 18:02:19 -05:00
#if SERIAL_PROTOCOL == PROTOCOL_SIMPLE_SERIAL
static uint8_t serialBuffer[CONFIG_AX25_FRAME_BUF_LEN+1]; // Buffer for holding incoming serial data
static int sbyte; // For holding byte read from serial port
static size_t serialLen = 0; // Counter for counting length of data from serial
static bool sertx = false; // Flag signifying whether it's time to send data
// received on the serial port.
#endif
2014-11-09 18:02:19 -05:00
#if SERIAL_PROTOCOL == PROTOCOL_KISS
static uint8_t sbyte; // For holding byte read from serial port
#endif
2014-06-23 13:33:42 -04:00
#define SER_BUFFER_FULL (serialLen < CONFIG_AX25_FRAME_BUF_LEN-1)
2014-04-03 16:21:37 -04:00
2014-04-06 14:26:24 -04:00
//////////////////////////////////////////////////////
// And here comes the actual program :) //
//////////////////////////////////////////////////////
2014-04-06 11:17:13 -04:00
// This is a callback we register with the protocol,
// so we can process each packet as they are decoded.
// Right now it just prints the packet to the serial port.
2014-11-09 18:02:19 -05:00
#if SERIAL_PROTOCOL == PROTOCOL_SIMPLE_SERIAL
2014-05-29 05:47:49 -04:00
static void message_callback(struct AX25Msg *msg)
{
2014-11-09 18:02:19 -05:00
ss_messageCallback(msg, &ser);
2014-04-04 05:17:47 -04:00
}
2014-11-09 18:02:19 -05:00
#endif
#if SERIAL_PROTOCOL == PROTOCOL_KISS
static void message_callback(struct AX25Ctx *ctx)
{
kiss_messageCallback(ctx);
}
#endif
2014-04-03 18:39:51 -04:00
2014-04-06 11:17:13 -04:00
// Simple initialization function.
2014-04-03 16:21:37 -04:00
static void init(void)
{
2014-05-20 05:20:19 -04:00
// Enable interrupts
IRQ_ENABLE;
2014-04-04 11:05:53 -04:00
2014-04-06 11:17:13 -04:00
// Initialize hardware timers
2014-05-20 05:20:19 -04:00
timer_init();
2014-04-03 16:21:37 -04:00
2014-05-20 05:20:19 -04:00
// Initialize serial comms on UART0,
// which is the hardware serial on arduino
ser_init(&ser, SER_UART0);
ser_setbaudrate(&ser, 9600);
2014-04-06 11:17:13 -04:00
2014-05-20 05:20:19 -04:00
// For some reason BertOS sets the serial
// to 7 bit characters by default. We set
// it to 8 instead.
UCSR0C = _BV(UCSZ01) | _BV(UCSZ00);
2014-04-23 16:49:04 -04:00
2014-05-20 05:20:19 -04:00
// Create a modem context
afsk_init(&afsk, ADC_CH);
// ... and a protocol context with the modem
2014-05-29 05:47:49 -04:00
ax25_init(&ax25, &afsk.fd, message_callback);
2014-04-04 05:17:47 -04:00
2014-11-09 18:02:19 -05:00
#if SERIAL_PROTOCOL == PROTOCOL_SIMPLE_SERIAL
// Init SimpleSerial
ss_init(&ax25);
#endif
#if SERIAL_PROTOCOL == PROTOCOL_KISS
// Init KISS
kiss_init(&ax25, &ser);
#endif
2014-05-20 05:20:19 -04:00
// That's all!
2014-04-03 16:21:37 -04:00
}
int main(void)
{
2014-05-20 05:20:19 -04:00
// Start by running the main initialization
init();
2014-11-09 18:02:19 -05:00
#if SERIAL_PROTOCOL == PROTOCOL_KISS
while (1) {
// First we instruct the protocol to check for
// incoming data
ax25_poll(&ax25);
if (ser_available(&ser)) {
sbyte = ser_getchar_nowait(&ser);
kiss_serialCallback(sbyte);
}
}
#endif
#if SERIAL_PROTOCOL == PROTOCOL_SIMPLE_SERIAL
// Record the current tick count for time-keeping
ticks_t start = timer_clock();
// Go into ye good ol' infinite loop
while (1) {
// First we instruct the protocol to check for
// incoming data
ax25_poll(&ax25);
// Poll for incoming serial data
if (!sertx && ser_available(&ser)) {
// We then read a byte from the serial port.
// Notice that we use "_nowait" since we can't
// have this blocking execution until a byte
// comes in.
sbyte = ser_getchar_nowait(&ser);
// If SERIAL_DEBUG is specified we'll handle
// serial data as direct human input and only
// transmit when we get a LF character
#if SERIAL_DEBUG
// If we have not yet surpassed the maximum frame length
// and the byte is not a "transmit" (newline) character,
// we should store it for transmission.
if ((serialLen < CONFIG_AX25_FRAME_BUF_LEN) && (sbyte != 10)) {
// Put the read byte into the buffer;
serialBuffer[serialLen] = sbyte;
// Increment the read length counter
serialLen++;
} else {
// If one of the above conditions were actually the
// case, it means we have to transmit, se we set
// transmission flag to true.
sertx = true;
}
#else
// Otherwise we assume the modem is running
// in automated mode, and we push out data
// as it becomes available. We either transmit
// immediately when the max frame length has
// been reached, or when we get no input for
// a certain amount of time.
if (serialLen < CONFIG_AX25_FRAME_BUF_LEN-1) {
// Put the read byte into the buffer;
serialBuffer[serialLen] = sbyte;
// Increment the read length counter
serialLen++;
} else {
// If max frame length has been reached
// we need to transmit.
serialBuffer[serialLen] = sbyte;
serialLen++;
sertx = true;
}
start = timer_clock();
#endif
} else {
if (!SERIAL_DEBUG && serialLen > 0 && timer_clock() - start > ms_to_ticks(TX_MAXWAIT)) {
sertx = true;
}
2014-11-09 18:02:19 -05:00
}
2014-11-09 18:02:19 -05:00
if (sertx) {
ss_serialCallback(serialBuffer, serialLen, &ser, &ax25);
sertx = false;
serialLen = 0;
}
}
2014-11-09 18:02:19 -05:00
#endif
2014-05-20 05:20:19 -04:00
return 0;
2014-04-03 16:41:49 -04:00
}