/* * spreadspace avr utils * * * Copyright (C) 2013 Christian Pointner * Othmar Gsenger * * This file is part of spreadspace avr utils. * * spreadspace avr utils 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. * * spreadspace avr utils 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 spreadspace avr utils. If not, see . */ #include #include #include #include #include "util.h" #include "led.h" /* this program listens on a usb serial/acm interface. When the send command ('s') is received the send buffer gets filled with the next 3 bytes. These bites are sent out serialized to a specified port using PWM. Before the data is sent, a sync bit is sent. The data is repeated multiple times. This version of the program is optimized for 433MHz radio controlled power plugs. To control them you need to add a ASK modulator to the output pin (default F0). */ #include #include "lufa-descriptor-usbserial.h" 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*/ /* Start Our Code */ // RF433 Sender is Connected to PF0 #define SEND_PORT PORTF #define SEND_DDR DDRF #define SEND_SERIAL 0 // Logical Zero (Sender turned off) is when output is low #define SEND_ZERO 1 // rf433 command length is 3 bytes #define SEND_CMD_LENGTH 3 // pwm is SEND_PWM_BASE timers high and SEND_PWN_LONG_MULT * SEND_PWM_BASE timers low for a low bit (low bit = low is longer) #define SEND_PWM_BASE 1 #define SEND_PWN_LONG_MULT 3 #define SEND_PWN_SYNC_MULT 31 //bits are read LSB from first byte first #define SEND_CURRENT_BIT (send_cmd_buffer[send_cmd_buffer_bit_pos/8] & (1<< (send_cmd_buffer_bit_pos % 8 ))) static char send_cmd_buffer[SEND_CMD_LENGTH]; static int8_t send_cmd_buffer_bit_pos=0; static uint8_t send_pwm_remain=0; static uint8_t send_pwm_output=0; static uint8_t send_repeat=0; static uint8_t send_sync=0; inline void sender_on(void) { if(SEND_ZERO) SEND_PORT |= (1 << SEND_SERIAL); else SEND_PORT &= ~(1 << SEND_SERIAL); send_pwm_output=1; } inline void sender_off(void) { if(SEND_ZERO) SEND_PORT &= ~(1 << SEND_SERIAL); else SEND_PORT |= (1 << SEND_SERIAL); send_pwm_output=0; } inline void sender_toggle(void) { SEND_PORT ^= (1 << SEND_SERIAL); } void sender_timer_enable(void) { TCCR0A = 1<>OCIE0A; } ISR(TIMER0_COMPA_vect) { PORTF^=2; //debug; if(send_cmd_buffer_bit_pos < SEND_CMD_LENGTH * 8) { if(send_pwm_remain) { //NOTHING } else if (send_sync) { sender_off(); send_pwm_remain=SEND_PWM_BASE*SEND_PWN_SYNC_MULT; send_sync=0; send_cmd_buffer_bit_pos=-1; } else if (send_pwm_output) { sender_off(); if(SEND_CURRENT_BIT) { send_pwm_remain=SEND_PWM_BASE; } else { send_pwm_remain=SEND_PWM_BASE*SEND_PWN_LONG_MULT; } } else { send_cmd_buffer_bit_pos++; sender_on(); if(SEND_CURRENT_BIT) { send_pwm_remain=SEND_PWM_BASE*SEND_PWN_LONG_MULT; } else { send_pwm_remain=SEND_PWM_BASE; } } send_pwm_remain--; } else { send_cmd_buffer_bit_pos=0; send_pwm_remain=0; if(send_repeat) { send_repeat--; send_sync=1; send_pwm_remain=SEND_PWM_BASE; sender_on(); } else { sender_timer_disable(); sender_off(); } } } void init_pins(void) { sender_off(); SEND_DDR |= 1 << SEND_SERIAL; DDRF |= 2; //DEBUG } void init_timer(void) { } static uint8_t command_pos=0; void handle_cmd(uint8_t cmd) { if (!command_pos) { switch(cmd) { case '0': led_off(); break; case '1': led_on(); break; case 't': led_toggle(); break; case 'r': reset2bootloader(); break; case 's': CDC_Device_SendString(&VirtualSerial_CDC_Interface, "Expecting multibyte command now\n\r"); command_pos=1; break; default: CDC_Device_SendString(&VirtualSerial_CDC_Interface, "error\n\r"); return; } CDC_Device_SendString(&VirtualSerial_CDC_Interface, "ok\n\r"); } else { send_cmd_buffer[command_pos-1]=cmd; if (command_pos==SEND_CMD_LENGTH) { CDC_Device_SendString(&VirtualSerial_CDC_Interface, "Enabling Timer\n\r"); command_pos=0; send_sync=1; send_pwm_remain=SEND_PWM_BASE; send_repeat=15; sender_on(); sender_timer_enable(); } else command_pos++; } } int main(void) { MCUSR &= ~(1 << WDRF); wdt_disable(); cpu_init(); led_init(); USB_Init(); init_pins(); init_timer(); sei(); for(;;) { int16_t BytesReceived = CDC_Device_BytesReceived(&VirtualSerial_CDC_Interface); while(BytesReceived > 0) { int16_t ReceivedByte = CDC_Device_ReceiveByte(&VirtualSerial_CDC_Interface); if(!(ReceivedByte < 0)) { handle_cmd(ReceivedByte); } BytesReceived--; } CDC_Device_USBTask(&VirtualSerial_CDC_Interface); USB_USBTask(); } }