Rev 40 | Rev 62 | Go to most recent revision | Blame | Compare with Previous | Last modification | View Log | RSS feed
#include <avr/io.h>#include <avr/pgmspace.h>#include <avr/interrupt.h>#define F_CPU 12000000#include <util/delay.h>#include <avr/wdt.h>#include <usbdrv.h>#include <stdlib.h>#include <string.h>#include "util.h"#include "wire.h"#include "config.h"#include "hiddesc.h"#define ROTS_ATTACHED 1#define STAT 0#define SENT 1#define SWITCHES 1#define ON 1#define OFF 0#define DETECTED 0#define TIMER 1#define DEBOUNCED 2/** Keyboard modifier codes*/#define MOD_CONTROL_LEFT (1<<0)#define MOD_SHIFT_LEFT (1<<1)#define MOD_ALT_LEFT (1<<2)#define MOD_GUI_LEFT (1<<3)#define MOD_CONTROL_RIGHT (1<<4)#define MOD_SHIFT_RIGHT (1<<5)#define MOD_ALT_RIGHT (1<<6)#define MOD_GUI_RIGHT (1<<7)void doInt(uint8_t pcint);uint8_t getKey(void);volatile uint8_t pcIntCurr[3] = {0,0,0};volatile uint8_t pcIntLast[3] = {0,0,0};volatile uint8_t pcIntMask[3] = {0,0,0};// switches = [detected][timer][debounced]volatile uint8_t switches[1][3] = { {0,0,0} };// rotdata = [rot#][(stat|sent)]volatile uint8_t rotdata[2][2] = { {0,0}, {0,0} };uint8_t keyMap[] = { 0x1E, 0x1F, 0x20,0x21, 0x22, 0x23,0x24, 0x25, 0x26,0x25, 0x27, 0x20 };uint8_t keySelect = 1;struct {uint8_t report_id;uint8_t modifier;uint8_t keycode;} reportKeyboard;struct{uint8_t report_id;union {uint8_t data1[2];struct {uint8_t rx:8;uint8_t ry:8;};};union {uint16_t data2;struct {uint16_t buttons:12;uint16_t rot2a:1;uint16_t rot2b:1;uint16_t rot1a:1;uint16_t rot1b:1;};};} reportJoystick;void usbSendHidReport(uchar * data, uchar len) {while(1){usbPoll();if (usbInterruptIsReady()){usbSetInterrupt(data, len);break;}}}usbMsgLen_t usbFunctionSetup(uchar data[8]) {usbRequest_t *rq = (void *)data;if((rq->bmRequestType & USBRQ_TYPE_MASK) == USBRQ_TYPE_CLASS) {switch (rq->bRequest) {case USBRQ_HID_GET_REPORT:if (rq->wValue.bytes[0] == 1)return sizeof(reportKeyboard);else if (rq->wValue.bytes[0] == 2)return sizeof(reportJoystick);elsereturn 0;case USBRQ_HID_GET_IDLE:return 1;default:return 0;}}return 0;}void hadUsbReset(void) {}int main(void) {ACSR |= (1<<ACD); // Disable analog comparator/*Setup ADCADMUX: 8 bit mode, Avcc refADCSRA: Enable, 128 prescale*/ADMUX = (1<<ADLAR) | (0<<REFS0) | (1<<REFS1);ADCSRA = (1<<ADEN) | (1<<ADPS2) | (1<<ADPS1) | (1<<ADPS0) ;/*DDR : 1 = Output, 0 = InputPORT: 1 = Pullup for Input, otherwise set outputPIN : Read input pin*//*PB0 - Output - Keypad 2PB1 - Output - Keypad 7PB2 - Output - Keypad 6PB3 - Output - Keypad 4PB4 - Input, Pullup - Function selectPB5 - Input, Pullup - Function select*/DDRB = 0B00001111;PORTB = 0B00111111;/*PD0 - Input, Pullup, PCINT16 - Rotary 1aPD1 - Input, Pullup, PCINT17 - Rotary 1bPD4 - Output - Keypad select status ledPD5 - Input, Pullup - Keypad 3PD6 - Input, Pullup - Keypad 1PD7 - Input, Pullup - Keypad 5*/DDRD = 0B00010000;PORTD = 0B11100011;PCMSK2 |= (( 1 << PCINT16 ) | ( 1 << PCINT17 )); //enable encoder pins interrupt sourcesPCICR |= ( 1 << PCIE2 ); //enable pin change interupts// Timers not used for the moment// Setup timer0 - Enable overflow, 8 times prescalerTIMSK0 = (1<<TOIE0); // Eable timer overflow for Timer0TCNT0 = 0x00; // Set Timer0 to 0TCCR0B = (1<< CS01) ; // /8 prescaleri2c_master();i2c_beginTransmission(0x27);i2c_writeByte(0x01);i2c_endTransmission(1);usbDeviceDisconnect(); /* enforce re-enumeration, do this while interrupts are disabled! */_delay_ms(500);usbDeviceConnect();wdt_enable(WDTO_1S);usbInit();sei();reportKeyboard.report_id = 1;reportJoystick.report_id = 2;for(;;) {wdt_reset();usbPoll();/** This is some really bad deboucing code*/// Detect the button press, wait 100 timer cycles (1.7ms / cycle)if (rbi(PINB, PB5) == 0 && switches[0][DETECTED]==0) {switches[0][DETECTED] = 1;switches[0][TIMER] = 100;}// After timer is zero, check switch again.// If switch still pressed, its debounced// Otherwise, reset the debounceif (switches[0][DETECTED] == 1 && switches[0][TIMER] == 0 && switches[0][DEBOUNCED]==0) {if (rbi(PINB, PB5) == 0)switches[0][DEBOUNCED] = 1;else {switches[0][DETECTED] = 0;switches[0][TIMER] = 0;}}// If the switch has come up, do another debounceif (rbi(PINB, PB5) && switches[0][DETECTED]==2) {switches[0][TIMER] = 100;switches[0][DETECTED] = 3;}// After the up switch timer is zero, do the debounce check// Otherwise, assume switch is still downif (switches[0][DETECTED] == 3 && switches[0][TIMER] == 0) {if (rbi(PINB, PB5))switches[0][DETECTED] = 0;}// Process the switchif (switches[0][DEBOUNCED] == 1) {xbi(keySelect, 0);if (keySelect == 0)sbi(PORTD, PD4);elsecbi(PORTD, PD4);switches[0][DETECTED] = 2;switches[0][DEBOUNCED] = 0;}if(usbInterruptIsReady()){reportJoystick.data1[0] = (-128 + analogRead(0));reportJoystick.data1[1] = (-128 + analogRead(1));reportJoystick.data2 = 0x0000; // Clear all the buttonsreportKeyboard.modifier = 0x00;reportKeyboard.keycode = 0x00;uint8_t key = getKey();if (rbi(keySelect, 0)) {// Keypad is joystickif (key > 0)reportJoystick.data2 |= (1 << (--key));} else {// Keypad is keyboardif (key > 0) {if (key==10 || key==12)reportKeyboard.modifier |= (1<<1); //Left shiftreportKeyboard.keycode = keyMap[--key];}}// Now work out what rotary to send, if any// Also record if we sent a positive response,// so we can send a '0' next time (if selected on PD4)// rotdata = [rot#][(stat|sent)]uint8_t rot = 0;for (rot=0; rot<=(ROTS_ATTACHED - 1); rot++) {if (rotdata[rot][STAT] == 0x01 && rotdata[rot][SENT] == 0) {rotdata[rot][SENT] = 1;switch (rot) {case(0): reportJoystick.rot1a = 1; break;case(1): reportJoystick.rot2a = 1; break;}} else if (rotdata[rot][STAT] == 0x02 && rotdata[rot][SENT] == 0) {rotdata[rot][SENT] = 1;switch (rot) {case(0): reportJoystick.rot1b = 1; break;case(1): reportJoystick.rot2b = 1; break;}} else {rotdata[rot][SENT] = 0;}rotdata[rot][STAT] = 0;if (rbi(PINB, PB4))rotdata[rot][SENT] = 0;}/* called after every poll of the interrupt endpoint *///usbSetInterrupt(&reportKeyboard, sizeof(reportKeyboard));//usbSetInterrupt(&reportJoystick, sizeof(reportJoystick));usbSendHidReport(&reportKeyboard, sizeof(reportKeyboard));usbSendHidReport(&reportJoystick, sizeof(reportJoystick));}}}uint8_t getKey() {uint8_t col, row = 0;uint8_t key = 0;uint8_t n = 1;for (row=0; row<=3; row++) {cbi(PORTB, row);_delay_us(10); // Wait for the port to changefor (col=5; col<=7; col++) {if (rbi(PIND, col) == 0)key = n;n++;}sbi(PORTB, row);}return key;}/*** Process the Pin Change Interrupt.* pcint provides what bank caused the interrupt**/void doInt(uint8_t pcint) {// Select what rotary we are dealing with// based on the pc interrupt that fired.uint8_t rot = 0;if (pcint == 1)rot = 1;// If rot stat is not 0, we havn't sent// our last results yet. Skip this click.if (rotdata[rot][STAT] != 0) {pcIntMask[pcint] = 0;return;}// Check which pin caused the interrupt. If they both// equal 0, the pin that interrupted is the directionif (rbi(pcIntCurr[pcint], PCINT17) == 0&& rbi(pcIntCurr[pcint], PCINT17) == 0&& rbi(pcIntMask[pcint], PCINT16) ) {rotdata[rot][STAT] = 1;} else if (rbi(pcIntCurr[pcint], PCINT16) == 0&& rbi(pcIntCurr[pcint], PCINT17) == 0&& rbi(pcIntMask[pcint], PCINT17) ) {rotdata[rot][STAT] = 2;}// Clear the mask so we know we've delth with itpcIntMask[pcint] = 0;}ISR(TIMER0_OVF_vect) {if (switches[0][DETECTED] && switches[0][TIMER])switches[0][TIMER]--;}ISR(PCINT1_vect){// Save the state and work out which pin caused// the interrupt to occurpcIntCurr[1] = PIND;pcIntMask[1] = pcIntCurr[1] ^ pcIntLast[1];pcIntLast[1] = pcIntCurr[1];doInt(1);}ISR(PCINT2_vect){// Save the state and work out which pin caused// the interrupt to occurpcIntCurr[2] = PIND;pcIntMask[2] = pcIntCurr[2] ^ pcIntLast[2];pcIntLast[2] = pcIntCurr[2];doInt(2);}