| @ -0,0 +1,56 @@ | |||||
| /* | |||||
| Pedal to Key takes in 2 (drum pedal) inputs and turns | |||||
| them into two separate key presses | |||||
| Created in 2023 | |||||
| by Brett Bender for Will Dean of UW-Stout PONG | |||||
| This code is licensed under the MIT License | |||||
| */ | |||||
| #ifndef USER_USB_RAM | |||||
| #error "This program needs to be compiled with a USER USB setting" | |||||
| #endif | |||||
| #include "src/userUsbHidKeyboard/USBHIDKeyboard.h" | |||||
| #define BUTTON1_PIN 33 | |||||
| #define BUTTON1_CHAR 'd' | |||||
| #define BUTTON2_PIN 32 | |||||
| #define BUTTON2_CHAR 'k' | |||||
| bool button1PressPrev = false; | |||||
| bool button2PressPrev = false; | |||||
| void setup() { | |||||
| USBInit(); | |||||
| pinMode(BUTTON1_PIN, INPUT_PULLUP); | |||||
| pinMode(BUTTON2_PIN, INPUT_PULLUP); | |||||
| } | |||||
| void loop() { | |||||
| //button 1 is mapped to letter 'd' | |||||
| bool button1Press = !digitalRead(BUTTON1_PIN); | |||||
| if (button1PressPrev != button1Press) { | |||||
| button1PressPrev = button1Press; | |||||
| if (button1Press) { | |||||
| Keyboard_press(BUTTON1_CHAR); | |||||
| } else { | |||||
| Keyboard_release(BUTTON1_CHAR); | |||||
| } | |||||
| } | |||||
| //button 2 is mapped to letter 'd' | |||||
| bool button2Press = !digitalRead(BUTTON2_PIN); | |||||
| if (button2PressPrev != button2Press) { | |||||
| button2PressPrev = button2Press; | |||||
| if (button2Press) { | |||||
| Keyboard_press(BUTTON2_CHAR); | |||||
| } else { | |||||
| Keyboard_release(BUTTON2_CHAR); | |||||
| } | |||||
| } | |||||
| delay(50); //naive debouncing | |||||
| } | |||||
| @ -0,0 +1,284 @@ | |||||
| #include <stdint.h> | |||||
| #include <stdbool.h> | |||||
| #include "include/ch5xx.h" | |||||
| #include "include/ch5xx_usb.h" | |||||
| #include "USBconstant.h" | |||||
| #include "USBhandler.h" | |||||
| extern __xdata __at (EP0_ADDR) uint8_t Ep0Buffer[]; | |||||
| extern __xdata __at (EP1_ADDR) uint8_t Ep1Buffer[]; | |||||
| volatile __xdata uint8_t UpPoint1_Busy = 0; //Flag of whether upload pointer is busy | |||||
| __xdata uint8_t HIDKey[8] = {0x0,0x0,0x0,0x0,0x0,0x0,0x0,0x0}; | |||||
| #define SHIFT 0x80 | |||||
| __code uint8_t _asciimap[128] = | |||||
| { | |||||
| 0x00, // NUL | |||||
| 0x00, // SOH | |||||
| 0x00, // STX | |||||
| 0x00, // ETX | |||||
| 0x00, // EOT | |||||
| 0x00, // ENQ | |||||
| 0x00, // ACK | |||||
| 0x00, // BEL | |||||
| 0x2a, // BS Backspace | |||||
| 0x2b, // TAB Tab | |||||
| 0x28, // LF Enter | |||||
| 0x00, // VT | |||||
| 0x00, // FF | |||||
| 0x00, // CR | |||||
| 0x00, // SO | |||||
| 0x00, // SI | |||||
| 0x00, // DEL | |||||
| 0x00, // DC1 | |||||
| 0x00, // DC2 | |||||
| 0x00, // DC3 | |||||
| 0x00, // DC4 | |||||
| 0x00, // NAK | |||||
| 0x00, // SYN | |||||
| 0x00, // ETB | |||||
| 0x00, // CAN | |||||
| 0x00, // EM | |||||
| 0x00, // SUB | |||||
| 0x00, // ESC | |||||
| 0x00, // FS | |||||
| 0x00, // GS | |||||
| 0x00, // RS | |||||
| 0x00, // US | |||||
| 0x2c, // ' ' | |||||
| 0x1e|SHIFT, // ! | |||||
| 0x34|SHIFT, // " | |||||
| 0x20|SHIFT, // # | |||||
| 0x21|SHIFT, // $ | |||||
| 0x22|SHIFT, // % | |||||
| 0x24|SHIFT, // & | |||||
| 0x34, // ' | |||||
| 0x26|SHIFT, // ( | |||||
| 0x27|SHIFT, // ) | |||||
| 0x25|SHIFT, // * | |||||
| 0x2e|SHIFT, // + | |||||
| 0x36, // , | |||||
| 0x2d, // - | |||||
| 0x37, // . | |||||
| 0x38, // / | |||||
| 0x27, // 0 | |||||
| 0x1e, // 1 | |||||
| 0x1f, // 2 | |||||
| 0x20, // 3 | |||||
| 0x21, // 4 | |||||
| 0x22, // 5 | |||||
| 0x23, // 6 | |||||
| 0x24, // 7 | |||||
| 0x25, // 8 | |||||
| 0x26, // 9 | |||||
| 0x33|SHIFT, // : | |||||
| 0x33, // ; | |||||
| 0x36|SHIFT, // < | |||||
| 0x2e, // = | |||||
| 0x37|SHIFT, // > | |||||
| 0x38|SHIFT, // ? | |||||
| 0x1f|SHIFT, // @ | |||||
| 0x04|SHIFT, // A | |||||
| 0x05|SHIFT, // B | |||||
| 0x06|SHIFT, // C | |||||
| 0x07|SHIFT, // D | |||||
| 0x08|SHIFT, // E | |||||
| 0x09|SHIFT, // F | |||||
| 0x0a|SHIFT, // G | |||||
| 0x0b|SHIFT, // H | |||||
| 0x0c|SHIFT, // I | |||||
| 0x0d|SHIFT, // J | |||||
| 0x0e|SHIFT, // K | |||||
| 0x0f|SHIFT, // L | |||||
| 0x10|SHIFT, // M | |||||
| 0x11|SHIFT, // N | |||||
| 0x12|SHIFT, // O | |||||
| 0x13|SHIFT, // P | |||||
| 0x14|SHIFT, // Q | |||||
| 0x15|SHIFT, // R | |||||
| 0x16|SHIFT, // S | |||||
| 0x17|SHIFT, // T | |||||
| 0x18|SHIFT, // U | |||||
| 0x19|SHIFT, // V | |||||
| 0x1a|SHIFT, // W | |||||
| 0x1b|SHIFT, // X | |||||
| 0x1c|SHIFT, // Y | |||||
| 0x1d|SHIFT, // Z | |||||
| 0x2f, // [ | |||||
| 0x31, // bslash | |||||
| 0x30, // ] | |||||
| 0x23|SHIFT, // ^ | |||||
| 0x2d|SHIFT, // _ | |||||
| 0x35, // ` | |||||
| 0x04, // a | |||||
| 0x05, // b | |||||
| 0x06, // c | |||||
| 0x07, // d | |||||
| 0x08, // e | |||||
| 0x09, // f | |||||
| 0x0a, // g | |||||
| 0x0b, // h | |||||
| 0x0c, // i | |||||
| 0x0d, // j | |||||
| 0x0e, // k | |||||
| 0x0f, // l | |||||
| 0x10, // m | |||||
| 0x11, // n | |||||
| 0x12, // o | |||||
| 0x13, // p | |||||
| 0x14, // q | |||||
| 0x15, // r | |||||
| 0x16, // s | |||||
| 0x17, // t | |||||
| 0x18, // u | |||||
| 0x19, // v | |||||
| 0x1a, // w | |||||
| 0x1b, // x | |||||
| 0x1c, // y | |||||
| 0x1d, // z | |||||
| 0x2f|SHIFT, // { | |||||
| 0x31|SHIFT, // | | |||||
| 0x30|SHIFT, // } | |||||
| 0x35|SHIFT, // ~ | |||||
| 0 // DEL | |||||
| }; | |||||
| typedef void( *pTaskFn)( void ); | |||||
| void delayMicroseconds(uint16_t us); | |||||
| void USBInit(){ | |||||
| USBDeviceCfg(); //Device mode configuration | |||||
| USBDeviceEndPointCfg(); //Endpoint configuration | |||||
| USBDeviceIntCfg(); //Interrupt configuration | |||||
| UEP0_T_LEN = 0; | |||||
| UEP1_T_LEN = 0; //Pre-use send length must be cleared | |||||
| UEP2_T_LEN = 0; | |||||
| } | |||||
| void USB_EP1_IN(){ | |||||
| UEP1_T_LEN = 0; | |||||
| UEP1_CTRL = UEP1_CTRL & ~ MASK_UEP_T_RES | UEP_T_RES_NAK; // Default NAK | |||||
| UpPoint1_Busy = 0; //Clear busy flag | |||||
| } | |||||
| void USB_EP1_OUT(){ | |||||
| if ( U_TOG_OK ) // Discard unsynchronized packets | |||||
| { | |||||
| } | |||||
| } | |||||
| uint8_t USB_EP1_send(){ | |||||
| uint16_t waitWriteCount = 0; | |||||
| waitWriteCount = 0; | |||||
| while (UpPoint1_Busy){//wait for 250ms or give up | |||||
| waitWriteCount++; | |||||
| delayMicroseconds(5); | |||||
| if (waitWriteCount>=50000) return 0; | |||||
| } | |||||
| for (uint8_t i=0;i<sizeof(HIDKey);i++){ //load data for upload | |||||
| Ep1Buffer[64+i] = HIDKey[i]; | |||||
| } | |||||
| UEP1_T_LEN = sizeof(HIDKey); //data length | |||||
| UpPoint1_Busy = 1; | |||||
| UEP1_CTRL = UEP1_CTRL & ~ MASK_UEP_T_RES | UEP_T_RES_ACK; //upload data and respond ACK | |||||
| return 1; | |||||
| } | |||||
| uint8_t Keyboard_press(uint8_t k) { | |||||
| uint8_t i; | |||||
| if (k >= 136) { // it's a non-printing key (not a modifier) | |||||
| k = k - 136; | |||||
| } else if (k >= 128) { // it's a modifier key | |||||
| HIDKey[0] |= (1<<(k-128)); | |||||
| k = 0; | |||||
| } else { // it's a printing key | |||||
| k = _asciimap[k]; | |||||
| if (!k) { | |||||
| //setWriteError(); | |||||
| return 0; | |||||
| } | |||||
| if (k & 0x80) { // it's a capital letter or other character reached with shift | |||||
| HIDKey[0] |= 0x02; // the left shift modifier | |||||
| k &= 0x7F; | |||||
| } | |||||
| } | |||||
| // Add k to the key report only if it's not already present | |||||
| // and if there is an empty slot. | |||||
| if (HIDKey[2] != k && HIDKey[3] != k && | |||||
| HIDKey[4] != k && HIDKey[5] != k && | |||||
| HIDKey[6] != k && HIDKey[7] != k) { | |||||
| for (i=2; i<8; i++) { | |||||
| if (HIDKey[i] == 0x00) { | |||||
| HIDKey[i] = k; | |||||
| break; | |||||
| } | |||||
| } | |||||
| if (i == 8) { | |||||
| //setWriteError(); | |||||
| return 0; | |||||
| } | |||||
| } | |||||
| USB_EP1_send(); | |||||
| return 1; | |||||
| } | |||||
| uint8_t Keyboard_release(uint8_t k) { | |||||
| uint8_t i; | |||||
| if (k >= 136) { // it's a non-printing key (not a modifier) | |||||
| k = k - 136; | |||||
| } else if (k >= 128) { // it's a modifier key | |||||
| HIDKey[0] &= ~(1<<(k-128)); | |||||
| k = 0; | |||||
| } else { // it's a printing key | |||||
| k = _asciimap[k]; | |||||
| if (!k) { | |||||
| return 0; | |||||
| } | |||||
| if (k & 0x80) { // it's a capital letter or other character reached with shift | |||||
| HIDKey[0] &= ~(0x02); // the left shift modifier | |||||
| k &= 0x7F; | |||||
| } | |||||
| } | |||||
| // Test the key report to see if k is present. Clear it if it exists. | |||||
| // Check all positions in case the key is present more than once (which it shouldn't be) | |||||
| for (i=2; i<8; i++) { | |||||
| if (0 != k && HIDKey[i] == k) { | |||||
| HIDKey[i] = 0x00; | |||||
| } | |||||
| } | |||||
| USB_EP1_send(); | |||||
| return 1; | |||||
| } | |||||
| void Keyboard_releaseAll(void){ | |||||
| for (uint8_t i=0;i<sizeof(HIDKey);i++){ //load data for upload | |||||
| HIDKey[i] = 0; | |||||
| } | |||||
| USB_EP1_send(); | |||||
| } | |||||
| uint8_t Keyboard_write(uint8_t c){ | |||||
| uint8_t p = Keyboard_press(c); // Keydown | |||||
| Keyboard_release(c); // Keyup | |||||
| return p; // just return the result of press() since release() almost always returns 1 | |||||
| } | |||||
| uint8_t Keyboard_getLEDStatus(){ | |||||
| return Ep1Buffer[0]; //The only info we gets | |||||
| } | |||||
| @ -0,0 +1,77 @@ | |||||
| #ifndef __USB_HID_KBD_H__ | |||||
| #define __USB_HID_KBD_H__ | |||||
| #include <stdint.h> | |||||
| #include "include/ch5xx.h" | |||||
| #include "include/ch5xx_usb.h" | |||||
| #define KEY_LEFT_CTRL 0x80 | |||||
| #define KEY_LEFT_SHIFT 0x81 | |||||
| #define KEY_LEFT_ALT 0x82 | |||||
| #define KEY_LEFT_GUI 0x83 | |||||
| #define KEY_RIGHT_CTRL 0x84 | |||||
| #define KEY_RIGHT_SHIFT 0x85 | |||||
| #define KEY_RIGHT_ALT 0x86 | |||||
| #define KEY_RIGHT_GUI 0x87 | |||||
| #define KEY_UP_ARROW 0xDA | |||||
| #define KEY_DOWN_ARROW 0xD9 | |||||
| #define KEY_LEFT_ARROW 0xD8 | |||||
| #define KEY_RIGHT_ARROW 0xD7 | |||||
| #define KEY_BACKSPACE 0xB2 | |||||
| #define KEY_TAB 0xB3 | |||||
| #define KEY_RETURN 0xB0 | |||||
| #define KEY_ESC 0xB1 | |||||
| #define KEY_INSERT 0xD1 | |||||
| #define KEY_DELETE 0xD4 | |||||
| #define KEY_PAGE_UP 0xD3 | |||||
| #define KEY_PAGE_DOWN 0xD6 | |||||
| #define KEY_HOME 0xD2 | |||||
| #define KEY_END 0xD5 | |||||
| #define KEY_CAPS_LOCK 0xC1 | |||||
| #define KEY_F1 0xC2 | |||||
| #define KEY_F2 0xC3 | |||||
| #define KEY_F3 0xC4 | |||||
| #define KEY_F4 0xC5 | |||||
| #define KEY_F5 0xC6 | |||||
| #define KEY_F6 0xC7 | |||||
| #define KEY_F7 0xC8 | |||||
| #define KEY_F8 0xC9 | |||||
| #define KEY_F9 0xCA | |||||
| #define KEY_F10 0xCB | |||||
| #define KEY_F11 0xCC | |||||
| #define KEY_F12 0xCD | |||||
| #define KEY_F13 0xF0 | |||||
| #define KEY_F14 0xF1 | |||||
| #define KEY_F15 0xF2 | |||||
| #define KEY_F16 0xF3 | |||||
| #define KEY_F17 0xF4 | |||||
| #define KEY_F18 0xF5 | |||||
| #define KEY_F19 0xF6 | |||||
| #define KEY_F20 0xF7 | |||||
| #define KEY_F21 0xF8 | |||||
| #define KEY_F22 0xF9 | |||||
| #define KEY_F23 0xFA | |||||
| #define KEY_F24 0xFB | |||||
| #ifdef __cplusplus | |||||
| extern "C" { | |||||
| #endif | |||||
| void USBInit(void); | |||||
| uint8_t Keyboard_press(uint8_t k); | |||||
| uint8_t Keyboard_release(uint8_t k); | |||||
| void Keyboard_releaseAll(void); | |||||
| uint8_t Keyboard_write(uint8_t c); | |||||
| uint8_t Keyboard_getLEDStatus(); | |||||
| #ifdef __cplusplus | |||||
| } // extern "C" | |||||
| #endif | |||||
| #endif | |||||
| @ -0,0 +1,88 @@ | |||||
| #include "USBconstant.h" | |||||
| //Device descriptor | |||||
| __code uint8_t DevDesc[] = { | |||||
| 0x12,0x01, | |||||
| 0x10,0x01, //USB spec release number in BCD format, USB1.1 (0x10, 0x01). | |||||
| 0x00,0x00,0x00, //bDeviceClass, bDeviceSubClass, bDeviceProtocol | |||||
| DEFAULT_ENDP0_SIZE, //bNumConfigurations | |||||
| 0x09,0x12,0x5D,0xC5, // VID PID | |||||
| 0x00,0x01, //version | |||||
| 0x01,0x02,0x03, //bString | |||||
| 0x01 //bNumConfigurations | |||||
| }; | |||||
| __code uint16_t DevDescLen = sizeof(DevDesc); | |||||
| __code uint8_t CfgDesc[] ={ | |||||
| 0x09,0x02,sizeof(CfgDesc) & 0xff,sizeof(CfgDesc) >> 8, | |||||
| 0x01,0x01,0x00,0x80,0x64, //Configuration descriptor (1 interface) | |||||
| // Interface 1 (HID) descriptor | |||||
| 0x09,0x04,0x00,0x00,0x02,0x03,0x01,0x01,0x00, // HID Keyboard, 2 endpoints | |||||
| 0x09,0x21,0x10,0x01,0x21,0x01,0x22,sizeof(ReportDesc) & 0xff,sizeof(ReportDesc) >> 8, //HID Descriptor | |||||
| 0x07,0x05,0x01,0x03,0x08,0x00,0x0A, //endpoint descriptor | |||||
| 0x07,0x05,0x81,0x03,0x08,0x00,0x0A, //endpoint descriptor | |||||
| }; | |||||
| __code uint16_t ReportDescLen = sizeof(ReportDesc); | |||||
| __code uint8_t ReportDesc[] ={ | |||||
| 0x05, 0x01, // USAGE_PAGE (Generic Desktop) | |||||
| 0x09, 0x06, // USAGE (Keyboard) | |||||
| 0xa1, 0x01, // COLLECTION (Application) | |||||
| 0x05, 0x07, // USAGE_PAGE (Keyboard) | |||||
| 0x19, 0xe0, // USAGE_MINIMUM (Keyboard LeftControl) | |||||
| 0x29, 0xe7, // USAGE_MAXIMUM (Keyboard Right GUI) | |||||
| 0x15, 0x00, // LOGICAL_MINIMUM (0) | |||||
| 0x25, 0x01, // LOGICAL_MAXIMUM (1) | |||||
| 0x95, 0x08, // REPORT_COUNT (8) | |||||
| 0x75, 0x01, // REPORT_SIZE (1) | |||||
| 0x81, 0x02, // INPUT (Data,Var,Abs) | |||||
| 0x95, 0x01, // REPORT_COUNT (1) | |||||
| 0x75, 0x08, // REPORT_SIZE (8) | |||||
| 0x81, 0x03, // INPUT (Cnst,Var,Abs) | |||||
| 0x95, 0x06, // REPORT_COUNT (6) | |||||
| 0x75, 0x08, // REPORT_SIZE (8) | |||||
| 0x15, 0x00, // LOGICAL_MINIMUM (0) | |||||
| 0x25, 0xff, // LOGICAL_MAXIMUM (255) | |||||
| 0x05, 0x07, // USAGE_PAGE (Keyboard) | |||||
| 0x19, 0x00, // USAGE_MINIMUM (Reserved (no event indicated)) | |||||
| 0x29, 0xe7, // USAGE_MAXIMUM (Keyboard Right GUI) | |||||
| 0x81, 0x00, // INPUT (Data,Ary,Abs) | |||||
| 0x05, 0x08, // USAGE_PAGE (LEDs) | |||||
| 0x19, 0x01, // USAGE_MINIMUM (Num Lock) | |||||
| 0x29, 0x05, // USAGE_MAXIMUM (Kana) | |||||
| 0x15, 0x00, // LOGICAL_MINIMUM (0) | |||||
| 0x25, 0x01, // LOGICAL_MAXIMUM (1) | |||||
| 0x95, 0x05, // REPORT_COUNT (5) | |||||
| 0x75, 0x01, // REPORT_SIZE (1) | |||||
| 0x91, 0x02, // OUTPUT (Data,Var,Abs) | |||||
| 0x95, 0x01, // REPORT_COUNT (1) | |||||
| 0x75, 0x03, // REPORT_SIZE (3) | |||||
| 0x91, 0x03, // OUTPUT (Cnst,Var,Abs) | |||||
| 0xc0 // END_COLLECTION | |||||
| }; | |||||
| __code uint16_t CfgDescLen = sizeof(CfgDesc); | |||||
| //String Descriptors | |||||
| __code uint8_t LangDes[]={0x04,0x03,0x09,0x04}; //Language Descriptor | |||||
| __code uint16_t LangDesLen = sizeof(LangDes); | |||||
| __code uint8_t SerDes[]={ //Serial String Descriptor | |||||
| 0x14,0x03, | |||||
| 'C',0x00,'H',0x00,'5',0x00,'5',0x00,'x',0x00,' ',0x00,'k',0x00,'b',0x00,'d',0x00 | |||||
| }; | |||||
| __code uint16_t SerDesLen = sizeof(SerDes); | |||||
| __code uint8_t Prod_Des[]={ //Produce String Descriptor | |||||
| 0x16,0x03, | |||||
| 'C',0x00,'H',0x00,'5',0x00,'5',0x00,'x',0x00,'d',0x00, | |||||
| 'u',0x00,'i',0x00,'n',0x00,'o',0x00 | |||||
| }; | |||||
| __code uint16_t Prod_DesLen = sizeof(Prod_Des); | |||||
| __code uint8_t Manuf_Des[]={ | |||||
| 0x0E,0x03, | |||||
| 'D',0x00,'e',0x00,'q',0x00,'i',0x00,'n',0x00,'g',0x00, | |||||
| }; | |||||
| __code uint16_t Manuf_DesLen = sizeof(Manuf_Des); | |||||
| @ -0,0 +1,27 @@ | |||||
| #ifndef __USB_CONST_DATA_H__ | |||||
| #define __USB_CONST_DATA_H__ | |||||
| #include <stdint.h> | |||||
| #include "include/ch5xx.h" | |||||
| #include "include/ch5xx_usb.h" | |||||
| #define EP0_ADDR 0 | |||||
| #define EP1_ADDR 10 | |||||
| extern __code uint8_t DevDesc[]; | |||||
| extern __code uint8_t CfgDesc[]; | |||||
| extern __code uint8_t LangDes[]; | |||||
| extern __code uint8_t ReportDesc[]; | |||||
| extern __code uint8_t SerDes[]; | |||||
| extern __code uint8_t Prod_Des[]; | |||||
| extern __code uint8_t Manuf_Des[]; | |||||
| extern __code uint16_t DevDescLen; | |||||
| extern __code uint16_t CfgDescLen; | |||||
| extern __code uint16_t LangDesLen; | |||||
| extern __code uint16_t ReportDescLen; | |||||
| extern __code uint16_t SerDesLen; | |||||
| extern __code uint16_t Prod_DesLen; | |||||
| extern __code uint16_t Manuf_DesLen; | |||||
| #endif | |||||
| @ -0,0 +1,485 @@ | |||||
| #include "USBhandler.h" | |||||
| #include "USBconstant.h" | |||||
| //Keyboard functions: | |||||
| void USB_EP2_IN(); | |||||
| void USB_EP2_OUT(); | |||||
| __xdata __at (EP0_ADDR) uint8_t Ep0Buffer[8]; | |||||
| __xdata __at (EP1_ADDR) uint8_t Ep1Buffer[128]; //on page 47 of data sheet, the receive buffer need to be min(possible packet size+2,64), IN and OUT buffer, must be even address | |||||
| #if (EP1_ADDR+128) > USER_USB_RAM | |||||
| #error "This program needs more USB RAM. Increase this setting in the menu." | |||||
| #endif | |||||
| uint16_t SetupLen; | |||||
| uint8_t SetupReq,UsbConfig; | |||||
| __code uint8_t *pDescr; | |||||
| volatile uint8_t usbMsgFlags=0; // uint8_t usbMsgFlags copied from VUSB | |||||
| inline void NOP_Process(void) {} | |||||
| void USB_EP0_SETUP(){ | |||||
| uint8_t len = USB_RX_LEN; | |||||
| if(len == (sizeof(USB_SETUP_REQ))) | |||||
| { | |||||
| SetupLen = ((uint16_t)UsbSetupBuf->wLengthH<<8) | (UsbSetupBuf->wLengthL); | |||||
| len = 0; // Default is success and upload 0 length | |||||
| SetupReq = UsbSetupBuf->bRequest; | |||||
| usbMsgFlags = 0; | |||||
| if ( ( UsbSetupBuf->bRequestType & USB_REQ_TYP_MASK ) != USB_REQ_TYP_STANDARD )//Not standard request | |||||
| { | |||||
| //here is the commnunication starts, refer to usbFunctionSetup of USBtiny | |||||
| //or usb_setup in usbtiny | |||||
| switch( ( UsbSetupBuf->bRequestType & USB_REQ_TYP_MASK )) | |||||
| { | |||||
| case USB_REQ_TYP_VENDOR: | |||||
| { | |||||
| switch( SetupReq ) | |||||
| { | |||||
| default: | |||||
| len = 0xFF; //command not supported | |||||
| break; | |||||
| } | |||||
| break; | |||||
| } | |||||
| case USB_REQ_TYP_CLASS: | |||||
| { | |||||
| switch( SetupReq ) | |||||
| { | |||||
| default: | |||||
| len = 0xFF; //command not supported | |||||
| break; | |||||
| } | |||||
| break; | |||||
| } | |||||
| default: | |||||
| len = 0xFF; //command not supported | |||||
| break; | |||||
| } | |||||
| } | |||||
| else //Standard request | |||||
| { | |||||
| switch(SetupReq) //Request ccfType | |||||
| { | |||||
| case USB_GET_DESCRIPTOR: | |||||
| switch(UsbSetupBuf->wValueH) | |||||
| { | |||||
| case 1: //Device Descriptor | |||||
| pDescr = DevDesc; //Put Device Descriptor into outgoing buffer | |||||
| len = DevDescLen; | |||||
| break; | |||||
| case 2: //Configure Descriptor | |||||
| pDescr = CfgDesc; | |||||
| len = CfgDescLen; | |||||
| break; | |||||
| case 3: | |||||
| if(UsbSetupBuf->wValueL == 0) | |||||
| { | |||||
| pDescr = LangDes; | |||||
| len = LangDesLen; | |||||
| } | |||||
| else if(UsbSetupBuf->wValueL == 1) | |||||
| { | |||||
| pDescr = Manuf_Des; | |||||
| len = Manuf_DesLen; | |||||
| } | |||||
| else if(UsbSetupBuf->wValueL == 2) | |||||
| { | |||||
| pDescr = Prod_Des; | |||||
| len = Prod_DesLen; | |||||
| } | |||||
| else if(UsbSetupBuf->wValueL == 3) | |||||
| { | |||||
| pDescr = SerDes; | |||||
| len = SerDesLen; | |||||
| } | |||||
| else | |||||
| { | |||||
| len = 0xff; | |||||
| } | |||||
| break; | |||||
| case 0x22: | |||||
| if(UsbSetupBuf->wValueL == 0){ | |||||
| pDescr = ReportDesc; | |||||
| len = ReportDescLen; | |||||
| } | |||||
| else | |||||
| { | |||||
| len = 0xff; | |||||
| } | |||||
| break; | |||||
| default: | |||||
| len = 0xff; // Unsupported descriptors or error | |||||
| break; | |||||
| } | |||||
| if (len != 0xff){ | |||||
| if ( SetupLen > len ) | |||||
| { | |||||
| SetupLen = len; // Limit length | |||||
| } | |||||
| len = SetupLen >= DEFAULT_ENDP0_SIZE ? DEFAULT_ENDP0_SIZE : SetupLen; //transmit length for this packet | |||||
| for (uint8_t i=0;i<len;i++){ | |||||
| Ep0Buffer[i] = pDescr[i]; | |||||
| } | |||||
| SetupLen -= len; | |||||
| pDescr += len; | |||||
| } | |||||
| break; | |||||
| case USB_SET_ADDRESS: | |||||
| SetupLen = UsbSetupBuf->wValueL; // Save the assigned address | |||||
| break; | |||||
| case USB_GET_CONFIGURATION: | |||||
| Ep0Buffer[0] = UsbConfig; | |||||
| if ( SetupLen >= 1 ) | |||||
| { | |||||
| len = 1; | |||||
| } | |||||
| break; | |||||
| case USB_SET_CONFIGURATION: | |||||
| UsbConfig = UsbSetupBuf->wValueL; | |||||
| break; | |||||
| case USB_GET_INTERFACE: | |||||
| break; | |||||
| case USB_SET_INTERFACE: | |||||
| break; | |||||
| case USB_CLEAR_FEATURE: //Clear Feature | |||||
| if( ( UsbSetupBuf->bRequestType & 0x1F ) == USB_REQ_RECIP_DEVICE ) // Clear the device featuee. | |||||
| { | |||||
| if( ( ( ( uint16_t )UsbSetupBuf->wValueH << 8 ) | UsbSetupBuf->wValueL ) == 0x01 ) | |||||
| { | |||||
| if( CfgDesc[ 7 ] & 0x20 ) | |||||
| { | |||||
| // wake up | |||||
| } | |||||
| else | |||||
| { | |||||
| len = 0xFF; //Failed | |||||
| } | |||||
| } | |||||
| else | |||||
| { | |||||
| len = 0xFF; //Failed | |||||
| } | |||||
| } | |||||
| else if ( ( UsbSetupBuf->bRequestType & USB_REQ_RECIP_MASK ) == USB_REQ_RECIP_ENDP )// endpoint | |||||
| { | |||||
| switch( UsbSetupBuf->wIndexL ) | |||||
| { | |||||
| case 0x84: | |||||
| UEP4_CTRL = UEP4_CTRL & ~ ( bUEP_T_TOG | MASK_UEP_T_RES ) | UEP_T_RES_NAK; | |||||
| break; | |||||
| case 0x04: | |||||
| UEP4_CTRL = UEP4_CTRL & ~ ( bUEP_R_TOG | MASK_UEP_R_RES ) | UEP_R_RES_ACK; | |||||
| break; | |||||
| case 0x83: | |||||
| UEP3_CTRL = UEP3_CTRL & ~ ( bUEP_T_TOG | MASK_UEP_T_RES ) | UEP_T_RES_NAK; | |||||
| break; | |||||
| case 0x03: | |||||
| UEP3_CTRL = UEP3_CTRL & ~ ( bUEP_R_TOG | MASK_UEP_R_RES ) | UEP_R_RES_ACK; | |||||
| break; | |||||
| case 0x82: | |||||
| UEP2_CTRL = UEP2_CTRL & ~ ( bUEP_T_TOG | MASK_UEP_T_RES ) | UEP_T_RES_NAK; | |||||
| break; | |||||
| case 0x02: | |||||
| UEP2_CTRL = UEP2_CTRL & ~ ( bUEP_R_TOG | MASK_UEP_R_RES ) | UEP_R_RES_ACK; | |||||
| break; | |||||
| case 0x81: | |||||
| UEP1_CTRL = UEP1_CTRL & ~ ( bUEP_T_TOG | MASK_UEP_T_RES ) | UEP_T_RES_NAK; | |||||
| break; | |||||
| case 0x01: | |||||
| UEP1_CTRL = UEP1_CTRL & ~ ( bUEP_R_TOG | MASK_UEP_R_RES ) | UEP_R_RES_ACK; | |||||
| break; | |||||
| default: | |||||
| len = 0xFF; // Unsupported endpoint | |||||
| break; | |||||
| } | |||||
| } | |||||
| else | |||||
| { | |||||
| len = 0xFF; // Unsupported for non-endpoint | |||||
| } | |||||
| break; | |||||
| case USB_SET_FEATURE: // Set Feature | |||||
| if( ( UsbSetupBuf->bRequestType & 0x1F ) == USB_REQ_RECIP_DEVICE ) // Set the device featuee. | |||||
| { | |||||
| if( ( ( ( uint16_t )UsbSetupBuf->wValueH << 8 ) | UsbSetupBuf->wValueL ) == 0x01 ) | |||||
| { | |||||
| if( CfgDesc[ 7 ] & 0x20 ) | |||||
| { | |||||
| // suspend | |||||
| //while ( XBUS_AUX & bUART0_TX ); //Wait till uart0 sending complete | |||||
| //SAFE_MOD = 0x55; | |||||
| //SAFE_MOD = 0xAA; | |||||
| //WAKE_CTRL = bWAK_BY_USB | bWAK_RXD0_LO | bWAK_RXD1_LO; //wake up by USB or RXD0/1 signal | |||||
| //PCON |= PD; //sleep | |||||
| //SAFE_MOD = 0x55; | |||||
| //SAFE_MOD = 0xAA; | |||||
| //WAKE_CTRL = 0x00; | |||||
| } | |||||
| else | |||||
| { | |||||
| len = 0xFF; // Failed | |||||
| } | |||||
| } | |||||
| else | |||||
| { | |||||
| len = 0xFF; // Failed | |||||
| } | |||||
| } | |||||
| else if( ( UsbSetupBuf->bRequestType & 0x1F ) == USB_REQ_RECIP_ENDP ) //endpoint | |||||
| { | |||||
| if( ( ( ( uint16_t )UsbSetupBuf->wValueH << 8 ) | UsbSetupBuf->wValueL ) == 0x00 ) | |||||
| { | |||||
| switch( ( ( uint16_t )UsbSetupBuf->wIndexH << 8 ) | UsbSetupBuf->wIndexL ) | |||||
| { | |||||
| case 0x84: | |||||
| UEP4_CTRL = UEP4_CTRL & (~bUEP_T_TOG) | UEP_T_RES_STALL;// Set endpoint4 IN STALL | |||||
| break; | |||||
| case 0x04: | |||||
| UEP4_CTRL = UEP4_CTRL & (~bUEP_R_TOG) | UEP_R_RES_STALL;// Set endpoint4 OUT Stall | |||||
| break; | |||||
| case 0x83: | |||||
| UEP3_CTRL = UEP3_CTRL & (~bUEP_T_TOG) | UEP_T_RES_STALL;// Set endpoint3 IN STALL | |||||
| break; | |||||
| case 0x03: | |||||
| UEP3_CTRL = UEP3_CTRL & (~bUEP_R_TOG) | UEP_R_RES_STALL;// Set endpoint3 OUT Stall | |||||
| break; | |||||
| case 0x82: | |||||
| UEP2_CTRL = UEP2_CTRL & (~bUEP_T_TOG) | UEP_T_RES_STALL;// Set endpoint2 IN STALL | |||||
| break; | |||||
| case 0x02: | |||||
| UEP2_CTRL = UEP2_CTRL & (~bUEP_R_TOG) | UEP_R_RES_STALL;// Set endpoint2 OUT Stall | |||||
| break; | |||||
| case 0x81: | |||||
| UEP1_CTRL = UEP1_CTRL & (~bUEP_T_TOG) | UEP_T_RES_STALL;// Set endpoint1 IN STALL | |||||
| break; | |||||
| case 0x01: | |||||
| UEP1_CTRL = UEP1_CTRL & (~bUEP_R_TOG) | UEP_R_RES_STALL;// Set endpoint1 OUT Stall | |||||
| default: | |||||
| len = 0xFF; // Failed | |||||
| break; | |||||
| } | |||||
| } | |||||
| else | |||||
| { | |||||
| len = 0xFF; // Failed | |||||
| } | |||||
| } | |||||
| else | |||||
| { | |||||
| len = 0xFF; // Failed | |||||
| } | |||||
| break; | |||||
| case USB_GET_STATUS: | |||||
| Ep0Buffer[0] = 0x00; | |||||
| Ep0Buffer[1] = 0x00; | |||||
| if ( SetupLen >= 2 ) | |||||
| { | |||||
| len = 2; | |||||
| } | |||||
| else | |||||
| { | |||||
| len = SetupLen; | |||||
| } | |||||
| break; | |||||
| default: | |||||
| len = 0xff; // Failed | |||||
| break; | |||||
| } | |||||
| } | |||||
| } | |||||
| else | |||||
| { | |||||
| len = 0xff; //Wrong packet length | |||||
| } | |||||
| if(len == 0xff) | |||||
| { | |||||
| SetupReq = 0xFF; | |||||
| UEP0_CTRL = bUEP_R_TOG | bUEP_T_TOG | UEP_R_RES_STALL | UEP_T_RES_STALL;//STALL | |||||
| } | |||||
| else if(len <= DEFAULT_ENDP0_SIZE) // Tx data to host or send 0-length packet | |||||
| { | |||||
| UEP0_T_LEN = len; | |||||
| UEP0_CTRL = bUEP_R_TOG | bUEP_T_TOG | UEP_R_RES_ACK | UEP_T_RES_ACK;//Expect DATA1, Answer ACK | |||||
| } | |||||
| else | |||||
| { | |||||
| UEP0_T_LEN = 0; // Tx data to host or send 0-length packet | |||||
| UEP0_CTRL = bUEP_R_TOG | bUEP_T_TOG | UEP_R_RES_ACK | UEP_T_RES_ACK;//Expect DATA1, Answer ACK | |||||
| } | |||||
| } | |||||
| void USB_EP0_IN(){ | |||||
| switch(SetupReq) | |||||
| { | |||||
| case USB_GET_DESCRIPTOR: | |||||
| { | |||||
| uint8_t len = SetupLen >= DEFAULT_ENDP0_SIZE ? DEFAULT_ENDP0_SIZE : SetupLen; //send length | |||||
| for (uint8_t i=0;i<len;i++){ | |||||
| Ep0Buffer[i] = pDescr[i]; | |||||
| } | |||||
| //memcpy( Ep0Buffer, pDescr, len ); | |||||
| SetupLen -= len; | |||||
| pDescr += len; | |||||
| UEP0_T_LEN = len; | |||||
| UEP0_CTRL ^= bUEP_T_TOG; //Switch between DATA0 and DATA1 | |||||
| } | |||||
| break; | |||||
| case USB_SET_ADDRESS: | |||||
| USB_DEV_AD = USB_DEV_AD & bUDA_GP_BIT | SetupLen; | |||||
| UEP0_CTRL = UEP_R_RES_ACK | UEP_T_RES_NAK; | |||||
| break; | |||||
| default: | |||||
| UEP0_T_LEN = 0; // End of transaction | |||||
| UEP0_CTRL = UEP_R_RES_ACK | UEP_T_RES_NAK; | |||||
| break; | |||||
| } | |||||
| } | |||||
| void USB_EP0_OUT(){ | |||||
| { | |||||
| UEP0_T_LEN = 0; | |||||
| UEP0_CTRL |= UEP_R_RES_ACK | UEP_T_RES_NAK; //Respond Nak | |||||
| } | |||||
| } | |||||
| #pragma save | |||||
| #pragma nooverlay | |||||
| void USBInterrupt(void) { //inline not really working in multiple files in SDCC | |||||
| if(UIF_TRANSFER) { | |||||
| // Dispatch to service functions | |||||
| uint8_t callIndex=USB_INT_ST & MASK_UIS_ENDP; | |||||
| switch (USB_INT_ST & MASK_UIS_TOKEN) { | |||||
| case UIS_TOKEN_OUT: | |||||
| {//SDCC will take IRAM if array of function pointer is used. | |||||
| switch (callIndex) { | |||||
| case 0: EP0_OUT_Callback(); break; | |||||
| case 1: EP1_OUT_Callback(); break; | |||||
| case 2: EP2_OUT_Callback(); break; | |||||
| case 3: EP3_OUT_Callback(); break; | |||||
| case 4: EP4_OUT_Callback(); break; | |||||
| default: break; | |||||
| } | |||||
| } | |||||
| break; | |||||
| case UIS_TOKEN_SOF: | |||||
| {//SDCC will take IRAM if array of function pointer is used. | |||||
| switch (callIndex) { | |||||
| case 0: EP0_SOF_Callback(); break; | |||||
| case 1: EP1_SOF_Callback(); break; | |||||
| case 2: EP2_SOF_Callback(); break; | |||||
| case 3: EP3_SOF_Callback(); break; | |||||
| case 4: EP4_SOF_Callback(); break; | |||||
| default: break; | |||||
| } | |||||
| } | |||||
| break; | |||||
| case UIS_TOKEN_IN: | |||||
| {//SDCC will take IRAM if array of function pointer is used. | |||||
| switch (callIndex) { | |||||
| case 0: EP0_IN_Callback(); break; | |||||
| case 1: EP1_IN_Callback(); break; | |||||
| case 2: EP2_IN_Callback(); break; | |||||
| case 3: EP3_IN_Callback(); break; | |||||
| case 4: EP4_IN_Callback(); break; | |||||
| default: break; | |||||
| } | |||||
| } | |||||
| break; | |||||
| case UIS_TOKEN_SETUP: | |||||
| {//SDCC will take IRAM if array of function pointer is used. | |||||
| switch (callIndex) { | |||||
| case 0: EP0_SETUP_Callback(); break; | |||||
| case 1: EP1_SETUP_Callback(); break; | |||||
| case 2: EP2_SETUP_Callback(); break; | |||||
| case 3: EP3_SETUP_Callback(); break; | |||||
| case 4: EP4_SETUP_Callback(); break; | |||||
| default: break; | |||||
| } | |||||
| } | |||||
| break; | |||||
| } | |||||
| UIF_TRANSFER = 0; // Clear interrupt flag | |||||
| } | |||||
| // Device mode USB bus reset | |||||
| if(UIF_BUS_RST) { | |||||
| UEP0_CTRL = UEP_R_RES_ACK | UEP_T_RES_NAK; | |||||
| UEP1_CTRL = bUEP_AUTO_TOG | UEP_T_RES_NAK | UEP_R_RES_ACK; | |||||
| USB_DEV_AD = 0x00; | |||||
| UIF_SUSPEND = 0; | |||||
| UIF_TRANSFER = 0; | |||||
| UIF_BUS_RST = 0; // Clear interrupt flag | |||||
| } | |||||
| // USB bus suspend / wake up | |||||
| if (UIF_SUSPEND) { | |||||
| UIF_SUSPEND = 0; | |||||
| if ( USB_MIS_ST & bUMS_SUSPEND ) { // Suspend | |||||
| //while ( XBUS_AUX & bUART0_TX ); // Wait for Tx | |||||
| //SAFE_MOD = 0x55; | |||||
| //SAFE_MOD = 0xAA; | |||||
| //WAKE_CTRL = bWAK_BY_USB | bWAK_RXD0_LO; // Wake up by USB or RxD0 | |||||
| //PCON |= PD; // Chip sleep | |||||
| //SAFE_MOD = 0x55; | |||||
| //SAFE_MOD = 0xAA; | |||||
| //WAKE_CTRL = 0x00; | |||||
| } else { // Unexpected interrupt, not supposed to happen ! | |||||
| USB_INT_FG = 0xFF; // Clear interrupt flag | |||||
| } | |||||
| } | |||||
| } | |||||
| #pragma restore | |||||
| void USBDeviceCfg() | |||||
| { | |||||
| USB_CTRL = 0x00; //Clear USB control register | |||||
| USB_CTRL &= ~bUC_HOST_MODE; //This bit is the device selection mode | |||||
| USB_CTRL |= bUC_DEV_PU_EN | bUC_INT_BUSY | bUC_DMA_EN; //USB device and internal pull-up enable, automatically return to NAK before interrupt flag is cleared during interrupt | |||||
| USB_DEV_AD = 0x00; //Device address initialization | |||||
| // USB_CTRL |= bUC_LOW_SPEED; | |||||
| // UDEV_CTRL |= bUD_LOW_SPEED; //Run for 1.5M | |||||
| USB_CTRL &= ~bUC_LOW_SPEED; | |||||
| UDEV_CTRL &= ~bUD_LOW_SPEED; //Select full speed 12M mode, default mode | |||||
| #if defined(CH551) || defined(CH552) || defined(CH549) | |||||
| UDEV_CTRL = bUD_PD_DIS; // Disable DP/DM pull-down resistor | |||||
| #endif | |||||
| #if defined(CH559) | |||||
| UDEV_CTRL = bUD_DP_PD_DIS; // Disable DP/DM pull-down resistor | |||||
| #endif | |||||
| UDEV_CTRL |= bUD_PORT_EN; //Enable physical port | |||||
| } | |||||
| void USBDeviceIntCfg() | |||||
| { | |||||
| USB_INT_EN |= bUIE_SUSPEND; //Enable device hang interrupt | |||||
| USB_INT_EN |= bUIE_TRANSFER; //Enable USB transfer completion interrupt | |||||
| USB_INT_EN |= bUIE_BUS_RST; //Enable device mode USB bus reset interrupt | |||||
| USB_INT_FG |= 0x1F; //Clear interrupt flag | |||||
| IE_USB = 1; //Enable USB interrupt | |||||
| EA = 1; //Enable global interrupts | |||||
| } | |||||
| void USBDeviceEndPointCfg() | |||||
| { | |||||
| UEP1_DMA = (uint16_t) Ep1Buffer; //Endpoint 1 data transfer address | |||||
| UEP1_CTRL = bUEP_AUTO_TOG | UEP_T_RES_NAK | UEP_R_RES_ACK; //Endpoint 2 automatically flips the sync flag, IN transaction returns NAK, OUT returns ACK | |||||
| UEP0_DMA = (uint16_t) Ep0Buffer; //Endpoint 0 data transfer address | |||||
| UEP4_1_MOD = 0XC0; //endpoint1 TX RX enable | |||||
| UEP0_CTRL = UEP_R_RES_ACK | UEP_T_RES_NAK; //Manual flip, OUT transaction returns ACK, IN transaction returns NAK | |||||
| } | |||||
| @ -0,0 +1,61 @@ | |||||
| #ifndef __USB_HANDLER_H__ | |||||
| #define __USB_HANDLER_H__ | |||||
| #include <stdint.h> | |||||
| #include "include/ch5xx.h" | |||||
| #include "include/ch5xx_usb.h" | |||||
| #include "USBconstant.h" | |||||
| extern __xdata __at (EP0_ADDR) uint8_t Ep0Buffer[]; | |||||
| extern __xdata __at (EP1_ADDR) uint8_t Ep1Buffer[]; | |||||
| extern uint16_t SetupLen; | |||||
| extern uint8_t SetupReq,UsbConfig; | |||||
| extern const __code uint8_t *pDescr; | |||||
| #define UsbSetupBuf ((PUSB_SETUP_REQ)Ep0Buffer) | |||||
| // Out | |||||
| #define EP0_OUT_Callback USB_EP0_OUT | |||||
| #define EP1_OUT_Callback USB_EP1_OUT | |||||
| #define EP2_OUT_Callback NOP_Process | |||||
| #define EP3_OUT_Callback NOP_Process | |||||
| #define EP4_OUT_Callback NOP_Process | |||||
| // SOF | |||||
| #define EP0_SOF_Callback NOP_Process | |||||
| #define EP1_SOF_Callback NOP_Process | |||||
| #define EP2_SOF_Callback NOP_Process | |||||
| #define EP3_SOF_Callback NOP_Process | |||||
| #define EP4_SOF_Callback NOP_Process | |||||
| // IN | |||||
| #define EP0_IN_Callback USB_EP0_IN | |||||
| #define EP1_IN_Callback USB_EP1_IN | |||||
| #define EP2_IN_Callback NOP_Process | |||||
| #define EP3_IN_Callback NOP_Process | |||||
| #define EP4_IN_Callback NOP_Process | |||||
| // SETUP | |||||
| #define EP0_SETUP_Callback USB_EP0_SETUP | |||||
| #define EP1_SETUP_Callback NOP_Process | |||||
| #define EP2_SETUP_Callback NOP_Process | |||||
| #define EP3_SETUP_Callback NOP_Process | |||||
| #define EP4_SETUP_Callback NOP_Process | |||||
| #ifdef __cplusplus | |||||
| extern "C" { | |||||
| #endif | |||||
| void USBInterrupt(void); | |||||
| void USBDeviceCfg(); | |||||
| void USBDeviceIntCfg(); | |||||
| void USBDeviceEndPointCfg(); | |||||
| #ifdef __cplusplus | |||||
| } // extern "C" | |||||
| #endif | |||||
| #endif | |||||