/* * * mur.sat * * Somewhen in the year 2012, mur.at will have a nano satellite launched * into a low earth orbit (310 km above the surface of our planet). The * satellite itself is a TubeSat personal satellite kit, developed and * launched by interorbital systems. mur.sat is a joint venture of mur.at, * ESC im Labor and realraum. * * Please visit the project hompage at sat.mur.at for further information. * * * Copyright (C) 2012 Bernhard Tittelbach * * This file is part of mur.sat. * * mur.sat 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 * any later version. * * mur.sat 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 mur.sat. If not, see . * */ #include #include #include #include #include //#include #include "led.h" #include "util.h" #include "hhd70.h" #include "c1101lib.h" //todo: move to separte File: // If TTX_EN goes high, ~ <=500ms before TTX CW Sender forcefully disables our RX/TX Signal Path //IF TTX_EN goes low, we have ~ 1min (Beacon Interval) until the next CW Beacon is sent and our RX/TX Signal Path is forcefully disabled // Beacon Interval Time can be measured by counting with an interrupt triggered by RTC_SECONDS Input //#define TTX_EN //#define RTC_SECONDS #define CPU_PRESCALE(n) do { CLKPR = 0x80; CLKPR = (n); } while(0) char read_buffer[64]; // buffer for reading usb signals char write_buffer[64]; // buffer for writing usb signals //TODOs: //* make adc work ? // * remove code duplications // * speed up and simplify code // * make as much use of sleep modes as possible // * use adc noise canceler (i.e. automatic sampling during cpu sleep) // * read atmega temp // * safely save state in eeprom (2 memory regions, only use the one with the "written successfully bit" which is written last) // * what if c1101 resets spuriously and clears it's settings ? -> check peridically ? /* ###### ###### LUFA Library Copyright (C) Dean Camera, 2012. dean [at] fourwalledcubicle [dot] com www.lufa-lib.org */ //#include //#include #include #include "lufa-descriptor-usbserial.h" /* Global I/O Buffers: */ //static RingBuffer_t SPItoUSB_Buffer; //static uint8_t SPItoUSB_Buffer_Data[8]; /** LUFA CDC Class driver interface configuration and state information. This structure is * passed to all CDC Class driver functions, so that multiple instances of the same class * within a device can be differentiated from one another. */ USB_ClassInfo_CDC_Device_t VirtualSerial_CDC_Interface = { .Config = { .ControlInterfaceNumber = 0, .DataINEndpointNumber = CDC_TX_EPNUM, .DataINEndpointSize = CDC_TXRX_EPSIZE, .DataINEndpointDoubleBank = false, .DataOUTEndpointNumber = CDC_RX_EPNUM, .DataOUTEndpointSize = CDC_TXRX_EPSIZE, .DataOUTEndpointDoubleBank = false, .NotificationEndpointNumber = CDC_NOTIFICATION_EPNUM, .NotificationEndpointSize = CDC_NOTIFICATION_EPSIZE, .NotificationEndpointDoubleBank = false, }, }; void EVENT_USB_Device_ConfigurationChanged(void) { CDC_Device_ConfigureEndpoints(&VirtualSerial_CDC_Interface); } void EVENT_USB_Device_ControlRequest(void) { CDC_Device_ProcessControlRequest(&VirtualSerial_CDC_Interface); } /* ###### end LUFA CDC-ACM specific definitions ###### */ bool enable_tx_part=false; bool enable_rx_part=false; bool enable_beacon_part=false; void print_part_status(void) { CDC_Device_SendString(&VirtualSerial_CDC_Interface, "OOK Beacon: "); CDC_Device_SendString(&VirtualSerial_CDC_Interface, ((enable_beacon_part)? "Enabled": "Disabled") ); CDC_Device_SendString(&VirtualSerial_CDC_Interface, "\r\n"); CDC_Device_SendString(&VirtualSerial_CDC_Interface, "RX-Part: "); CDC_Device_SendString(&VirtualSerial_CDC_Interface, ((enable_rx_part)? "Enabled": "Disabled") ); CDC_Device_SendString(&VirtualSerial_CDC_Interface, "\r\n"); CDC_Device_SendString(&VirtualSerial_CDC_Interface, "TX-Part: "); CDC_Device_SendString(&VirtualSerial_CDC_Interface, ((enable_tx_part)? "Enabled": "Disabled") ); CDC_Device_SendString(&VirtualSerial_CDC_Interface, "\r\n"); } //void generate_morse_code_sequence(char *msg, *cw_buffer, uint16_t buffer_len) //{ //char cw_short[2] = {1,1}; //char cw_long[2] = {1,3}; //char cw_space[2] = {0, 1}; //char cw_letterspace[2] = {0,3}; //char cw_wordspace[2] = {0, 7}; //} int main(void) { MCUSR &= ~(1 << WDRF); wdt_disable(); // FIXME: disable WatchDog for now, but use it later... CPU_PRESCALE(0); //cpu_init led_init(); USB_Init(); hhd70_init(); DDRB &= ~(1< 0) { int16_t recv_byte = CDC_Device_ReceiveByte(&VirtualSerial_CDC_Interface); num_bytes_received--; if (recv_byte < 0) continue; if ((char) recv_byte == '!') { CDC_Device_SendString(&VirtualSerial_CDC_Interface, "resetting\r\n"); wdt_enable(WDTO_500MS); reset(); } else if ((char) recv_byte == 'B') { enable_beacon_part = !enable_beacon_part; if (enable_beacon_part) { enable_rx_part=false; enable_tx_part=false; c1101_init_ook_beacon(); hhd70_palna_txmode(); } else { hhd70_palna_rxmode(); } print_part_status(); } else if ((char) recv_byte == 'E') { enable_rx_part = !enable_rx_part; if (enable_rx_part) { enable_beacon_part=false; c1101_init(); } print_part_status(); } else if ((char) recv_byte == 'S') { enable_tx_part = !enable_tx_part; if (enable_tx_part) { enable_beacon_part=false; c1101_init(); } print_part_status(); } else if ((char) recv_byte == 'R') { c1101_spi_dump_registers_to_usb(); print_part_status(); } else if ((char) recv_byte == 'D' || (char) recv_byte == 'd') { sprintf(write_buffer,"FSK Freq Deviation Now: %u0 Hz\r\n", c1101_setFSKDeviationFromCarrier(fdev_m, fdev_e)); CDC_Device_SendString(&VirtualSerial_CDC_Interface, write_buffer); if ((char) recv_byte == 'd') { fdev_m--; if (fdev_m < 0) { fdev_e = (fdev_e - 1) % 8; if (fdev_e < 0) fdev_e = 7; fdev_m = 7; } fdev_m %= 8; } else { fdev_m++; if (fdev_m > 7) fdev_e = (fdev_e + 1) % 8; fdev_m %= 8; } } } CDC_Device_USBTask(&VirtualSerial_CDC_Interface); USB_USBTask(); if (enable_rx_part) { c1101_spi_strobe_command(SPIC1101_ADDR_SRX); // enter RX - Mode _delay_ms(1000); if (hhd70_rx_data_available()) { led_on(); CDC_Device_SendString(&VirtualSerial_CDC_Interface,"RX: GDO2 pin HIGH\r\n"); CDC_Device_SendString(&VirtualSerial_CDC_Interface,"c1101 rx bytes:"); debug_sprint_int16hex(write_buffer, c1101_spi_read_register(SPIC1101_ADDR_RXBYTES)); CDC_Device_SendString(&VirtualSerial_CDC_Interface,write_buffer); CDC_Device_SendString(&VirtualSerial_CDC_Interface,"\r\n"); c1101_recieveData(); led_off(); } CDC_Device_SendString(&VirtualSerial_CDC_Interface,"c1101 rssi: "); debug_sprint_int16hex(write_buffer, c1101_spi_read_register(SPIC1101_ADDR_RSSI)); CDC_Device_SendString(&VirtualSerial_CDC_Interface,write_buffer); CDC_Device_SendString(&VirtualSerial_CDC_Interface,"\r\n"); CDC_Device_SendString(&VirtualSerial_CDC_Interface,"c1101 tx bytes: "); debug_sprint_int16hex(write_buffer, c1101_spi_read_register(SPIC1101_ADDR_TXBYTES)); CDC_Device_SendString(&VirtualSerial_CDC_Interface,write_buffer); CDC_Device_SendString(&VirtualSerial_CDC_Interface,"\r\n"); CDC_Device_SendString(&VirtualSerial_CDC_Interface,"c1101 rx bytes: "); int16_t num_rx_bytes = c1101_spi_read_register(SPIC1101_ADDR_RXBYTES); debug_sprint_int16hex(write_buffer, num_rx_bytes); CDC_Device_SendString(&VirtualSerial_CDC_Interface,write_buffer); CDC_Device_SendString(&VirtualSerial_CDC_Interface,"\r\n"); //~ if (num_rx_bytes > 0) //~ { //~ led_on(); //~ CDC_Device_SendString(&VirtualSerial_CDC_Interface,"RX Buffer Non-Empty\r\n"); //~ c1101_recieveData(); //~ led_off(); //~ } // CDC_Device_SendString(&VirtualSerial_CDC_Interface,"c1101 status byte: "); // debug_sprint_int16hex(write_buffer, c1101_getStatus()); // CDC_Device_SendString(&VirtualSerial_CDC_Interface,write_buffer); // CDC_Device_SendString(&VirtualSerial_CDC_Interface,"\r\n"); } if (enable_tx_part) { write_buffer[0]='T'; write_buffer[1]='e'; write_buffer[2]='m'; write_buffer[3]='p'; write_buffer[4]='s'; write_buffer[5]=':'; adc_on(); _delay_ms(250); CDC_Device_SendString(&VirtualSerial_CDC_Interface,"temp c1101: "); debug_sprint_int16hex(write_buffer+6, c1101_measureTemp()); CDC_Device_SendString(&VirtualSerial_CDC_Interface,write_buffer+6); CDC_Device_SendString(&VirtualSerial_CDC_Interface,"\r\n"); _delay_ms(250); CDC_Device_SendString(&VirtualSerial_CDC_Interface,"temp atmega: "); debug_sprint_int16hex(write_buffer+10, adc_read(ADCMUX_ADC12)); CDC_Device_SendString(&VirtualSerial_CDC_Interface,write_buffer+10); CDC_Device_SendString(&VirtualSerial_CDC_Interface,"\r\n"); adc_off(); _delay_ms(250); led_on(); CDC_Device_SendString(&VirtualSerial_CDC_Interface,"TX Data: String\r\n"); c1101_transmitData_infPktMode("OE6EOF test mur.sat GFSK r:9k6 fdev:11kHz 1234567890123456789012345678901234567890 End of Test",93); led_off(); _delay_ms(100); led_on(); CDC_Device_SendString(&VirtualSerial_CDC_Interface,"TX Data: Temps\r\n"); c1101_transmitData_infPktMode((char*) write_buffer,14); led_off(); } if (enable_beacon_part) { // char mursat_beacon[8] = {0b11101110, 0b11100010, 0b00111010, 0b10101000, 0b10001110, 0b11101110, 0b00101011, 0b10100000}; //OE6EOF led_on(); CDC_Device_SendString(&VirtualSerial_CDC_Interface,"OOK Sending Beacon\r\n"); c1101_transmitData_infPktMode(mursat_beacon,sizeof(mursat_beacon)); led_off(); } } }