261 lines
		
	
	
		
			9.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			261 lines
		
	
	
		
			9.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #include <stdint.h>
 | |
| #include "USBHID.h"
 | |
| #include "USBHID_Types.h"
 | |
| #include "USBDescriptor.h"
 | |
| #include "HIDKeyboard.h"
 | |
| 
 | |
| #define DEFAULT_CONFIGURATION (1)
 | |
| 
 | |
| HIDKeyboard::HIDKeyboard(uint16_t vendor_id, uint16_t product_id, uint16_t product_release) : USBDevice(vendor_id, product_id, product_release) { USBDevice::connect(); }
 | |
| 
 | |
| bool HIDKeyboard::sendReport(report_keyboard_t report) {
 | |
|     USBDevice::write(EP1IN, report.raw, sizeof(report), MAX_PACKET_SIZE_EP1);
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| uint8_t HIDKeyboard::leds() { return led_state; }
 | |
| 
 | |
| bool HIDKeyboard::USBCallback_setConfiguration(uint8_t configuration) {
 | |
|     if (configuration != DEFAULT_CONFIGURATION) {
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     // Configure endpoints > 0
 | |
|     addEndpoint(EPINT_IN, MAX_PACKET_SIZE_EPINT);
 | |
|     // addEndpoint(EPINT_OUT, MAX_PACKET_SIZE_EPINT);
 | |
| 
 | |
|     // We activate the endpoint to be able to recceive data
 | |
|     // readStart(EPINT_OUT, MAX_PACKET_SIZE_EPINT);
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| uint8_t *HIDKeyboard::stringImanufacturerDesc() {
 | |
|     static uint8_t stringImanufacturerDescriptor[] = {
 | |
|         0x18,              /*bLength*/
 | |
|         STRING_DESCRIPTOR, /*bDescriptorType 0x03*/
 | |
|         't',
 | |
|         0,
 | |
|         'm',
 | |
|         0,
 | |
|         'k',
 | |
|         0,
 | |
|         '-',
 | |
|         0,
 | |
|         'k',
 | |
|         0,
 | |
|         'b',
 | |
|         0,
 | |
|         'd',
 | |
|         0,
 | |
|         '.',
 | |
|         0,
 | |
|         'c',
 | |
|         0,
 | |
|         'o',
 | |
|         0,
 | |
|         'm',
 | |
|         0 /*bString iManufacturer*/
 | |
|     };
 | |
|     return stringImanufacturerDescriptor;
 | |
| }
 | |
| 
 | |
| uint8_t *HIDKeyboard::stringIproductDesc() {
 | |
|     static uint8_t stringIproductDescriptor[] = {
 | |
|         0x0a,              /*bLength*/
 | |
|         STRING_DESCRIPTOR, /*bDescriptorType 0x03*/
 | |
|         'm',
 | |
|         0,
 | |
|         'b',
 | |
|         0,
 | |
|         'e',
 | |
|         0,
 | |
|         'd',
 | |
|         0 /*bString iProduct*/
 | |
|     };
 | |
|     return stringIproductDescriptor;
 | |
| }
 | |
| 
 | |
| uint8_t *HIDKeyboard::stringIserialDesc() {
 | |
|     static uint8_t stringIserialDescriptor[] = {
 | |
|         0x04,              /*bLength*/
 | |
|         STRING_DESCRIPTOR, /*bDescriptorType 0x03*/
 | |
|         '0', 0             /*bString iSerial*/
 | |
|     };
 | |
|     return stringIserialDescriptor;
 | |
| }
 | |
| 
 | |
| uint8_t *HIDKeyboard::reportDesc() {
 | |
|     static uint8_t reportDescriptor[] = {
 | |
|         USAGE_PAGE(1),     0x01,  // Generic Desktop
 | |
|         USAGE(1),          0x06,  // Keyboard
 | |
|         COLLECTION(1),     0x01,  // Application
 | |
| 
 | |
|         USAGE_PAGE(1),     0x07,                                                                                                                                           // Key Codes
 | |
|         USAGE_MINIMUM(1),  0xE0, USAGE_MAXIMUM(1), 0xE7, LOGICAL_MINIMUM(1), 0x00, LOGICAL_MAXIMUM(1), 0x01, REPORT_SIZE(1), 0x01, REPORT_COUNT(1), 0x08, INPUT(1), 0x02,  // Data, Variable, Absolute
 | |
| 
 | |
|         REPORT_COUNT(1),   0x01, REPORT_SIZE(1),   0x08, INPUT(1),           0x01,  // Constant
 | |
| 
 | |
|         REPORT_COUNT(1),   0x05, REPORT_SIZE(1),   0x01, USAGE_PAGE(1),      0x08,  // LEDs
 | |
|         USAGE_MINIMUM(1),  0x01, USAGE_MAXIMUM(1), 0x05, OUTPUT(1),          0x02,  // Data, Variable, Absolute
 | |
| 
 | |
|         REPORT_COUNT(1),   0x01, REPORT_SIZE(1),   0x03, OUTPUT(1),          0x01,  // Constant
 | |
| 
 | |
|         REPORT_COUNT(1),   0x06, REPORT_SIZE(1),   0x08, LOGICAL_MINIMUM(1), 0x00, LOGICAL_MAXIMUM(1), 0xFF, USAGE_PAGE(1),  0x07,  // Key Codes
 | |
|         USAGE_MINIMUM(1),  0x00, USAGE_MAXIMUM(1), 0xFF, INPUT(1),           0x00,                                                  // Data, Array
 | |
|         END_COLLECTION(0),
 | |
|     };
 | |
|     reportLength = sizeof(reportDescriptor);
 | |
|     return reportDescriptor;
 | |
| }
 | |
| 
 | |
| uint16_t HIDKeyboard::reportDescLength() {
 | |
|     reportDesc();
 | |
|     return reportLength;
 | |
| }
 | |
| 
 | |
| #define TOTAL_DESCRIPTOR_LENGTH ((1 * CONFIGURATION_DESCRIPTOR_LENGTH) + (1 * INTERFACE_DESCRIPTOR_LENGTH) + (1 * HID_DESCRIPTOR_LENGTH) + (1 * ENDPOINT_DESCRIPTOR_LENGTH))
 | |
| uint8_t *HIDKeyboard::configurationDesc() {
 | |
|     static uint8_t configurationDescriptor[] = {
 | |
|         CONFIGURATION_DESCRIPTOR_LENGTH,  // bLength
 | |
|         CONFIGURATION_DESCRIPTOR,         // bDescriptorType
 | |
|         LSB(TOTAL_DESCRIPTOR_LENGTH),     // wTotalLength (LSB)
 | |
|         MSB(TOTAL_DESCRIPTOR_LENGTH),     // wTotalLength (MSB)
 | |
|         0x01,                             // bNumInterfaces
 | |
|         DEFAULT_CONFIGURATION,            // bConfigurationValue
 | |
|         0x00,                             // iConfiguration
 | |
|         C_RESERVED | C_REMOTE_WAKEUP,     // bmAttributes
 | |
|         C_POWER(100),                     // bMaxPowerHello World from Mbed
 | |
| 
 | |
|         INTERFACE_DESCRIPTOR_LENGTH,  // bLength
 | |
|         INTERFACE_DESCRIPTOR,         // bDescriptorType
 | |
|         0x00,                         // bInterfaceNumber
 | |
|         0x00,                         // bAlternateSetting
 | |
|         0x01,                         // bNumEndpoints
 | |
|         HID_CLASS,                    // bInterfaceClass
 | |
|         1,                            // bInterfaceSubClass (boot)
 | |
|         1,                            // bInterfaceProtocol (keyboard)
 | |
|         0x00,                         // iInterface
 | |
| 
 | |
|         HID_DESCRIPTOR_LENGTH,               // bLength
 | |
|         HID_DESCRIPTOR,                      // bDescriptorType
 | |
|         LSB(HID_VERSION_1_11),               // bcdHID (LSB)
 | |
|         MSB(HID_VERSION_1_11),               // bcdHID (MSB)
 | |
|         0x00,                                // bCountryCode
 | |
|         0x01,                                // bNumDescriptors
 | |
|         REPORT_DESCRIPTOR,                   // bDescriptorType
 | |
|         (uint8_t)(LSB(reportDescLength())),  // wDescriptorLength (LSB)
 | |
|         (uint8_t)(MSB(reportDescLength())),  // wDescriptorLength (MSB)
 | |
| 
 | |
|         ENDPOINT_DESCRIPTOR_LENGTH,  // bLength
 | |
|         ENDPOINT_DESCRIPTOR,         // bDescriptorType
 | |
|         PHY_TO_DESC(EP1IN),          // bEndpointAddress
 | |
|         E_INTERRUPT,                 // bmAttributes
 | |
|         LSB(MAX_PACKET_SIZE_EPINT),  // wMaxPacketSize (LSB)
 | |
|         MSB(MAX_PACKET_SIZE_EPINT),  // wMaxPacketSize (MSB)
 | |
|         1,                           // bInterval (milliseconds)
 | |
|     };
 | |
|     return configurationDescriptor;
 | |
| }
 | |
| 
 | |
| #if 0
 | |
| uint8_t * HIDKeyboard::deviceDesc() {
 | |
|     static uint8_t deviceDescriptor[] = {
 | |
|         DEVICE_DESCRIPTOR_LENGTH,       /* bLength */
 | |
|         DEVICE_DESCRIPTOR,              /* bDescriptorType */
 | |
|         LSB(USB_VERSION_2_0),           /* bcdUSB (LSB) */
 | |
|         MSB(USB_VERSION_2_0),           /* bcdUSB (MSB) */
 | |
|         0x00,                           /* bDeviceClass */
 | |
|         0x00,                           /* bDeviceSubClass */
 | |
|         0x00,                           /* bDeviceprotocol */
 | |
|         MAX_PACKET_SIZE_EP0,            /* bMaxPacketSize0 */
 | |
|         (uint8_t)(LSB(0xfeed)),                 /* idVendor (LSB) */
 | |
|         (uint8_t)(MSB(0xfeed)),                 /* idVendor (MSB) */
 | |
|         (uint8_t)(LSB(0x1bed)),                /* idProduct (LSB) */
 | |
|         (uint8_t)(MSB(0x1bed)),                /* idProduct (MSB) */
 | |
|         (uint8_t)(LSB(0x0002)),           /* bcdDevice (LSB) */
 | |
|         (uint8_t)(MSB(0x0002)),           /* bcdDevice (MSB) */
 | |
|         0,    /* iManufacturer */
 | |
|         0,         /* iProduct */
 | |
|         0,          /* iSerialNumber */
 | |
|         0x01                            /* bNumConfigurations */
 | |
|     };
 | |
|     return deviceDescriptor;
 | |
| }
 | |
| #endif
 | |
| 
 | |
| bool HIDKeyboard::USBCallback_request() {
 | |
|     bool              success  = false;
 | |
|     CONTROL_TRANSFER *transfer = getTransferPtr();
 | |
|     uint8_t *         hidDescriptor;
 | |
| 
 | |
|     // Process additional standard requests
 | |
| 
 | |
|     if ((transfer->setup.bmRequestType.Type == STANDARD_TYPE)) {
 | |
|         switch (transfer->setup.bRequest) {
 | |
|             case GET_DESCRIPTOR:
 | |
|                 switch (DESCRIPTOR_TYPE(transfer->setup.wValue)) {
 | |
|                     case REPORT_DESCRIPTOR:
 | |
|                         if ((reportDesc() != NULL) && (reportDescLength() != 0)) {
 | |
|                             transfer->remaining = reportDescLength();
 | |
|                             transfer->ptr       = reportDesc();
 | |
|                             transfer->direction = DEVICE_TO_HOST;
 | |
|                             success             = true;
 | |
|                         }
 | |
|                         break;
 | |
|                     case HID_DESCRIPTOR:
 | |
|                         // Find the HID descriptor, after the configuration descriptor
 | |
|                         hidDescriptor = findDescriptor(HID_DESCRIPTOR);
 | |
|                         if (hidDescriptor != NULL) {
 | |
|                             transfer->remaining = HID_DESCRIPTOR_LENGTH;
 | |
|                             transfer->ptr       = hidDescriptor;
 | |
|                             transfer->direction = DEVICE_TO_HOST;
 | |
|                             success             = true;
 | |
|                         }
 | |
|                         break;
 | |
| 
 | |
|                     default:
 | |
|                         break;
 | |
|                 }
 | |
|                 break;
 | |
|             default:
 | |
|                 break;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     // Process class-specific requests
 | |
|     if (transfer->setup.bmRequestType.Type == CLASS_TYPE) {
 | |
|         switch (transfer->setup.bRequest) {
 | |
|             case SET_REPORT:
 | |
|                 // LED indicator
 | |
|                 // TODO: check Interface and Report length?
 | |
|                 // if (transfer->setup.wIndex == INTERFACE_KEYBOAD) { }
 | |
|                 // if (transfer->setup.wLength == 1)
 | |
| 
 | |
|                 transfer->remaining = 1;
 | |
|                 // transfer->ptr = ?? what ptr should be set when OUT(not used?)
 | |
|                 transfer->direction = HOST_TO_DEVICE;
 | |
|                 transfer->notify    = true; /* notify with USBCallback_requestCompleted */
 | |
|                 success             = true;
 | |
|             default:
 | |
|                 break;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return success;
 | |
| }
 | |
| 
 | |
| void HIDKeyboard::USBCallback_requestCompleted(uint8_t *buf, uint32_t length) {
 | |
|     if (length > 0) {
 | |
|         CONTROL_TRANSFER *transfer = getTransferPtr();
 | |
|         if (transfer->setup.bmRequestType.Type == CLASS_TYPE) {
 | |
|             switch (transfer->setup.bRequest) {
 | |
|                 case SET_REPORT:
 | |
|                     led_state = buf[0];
 | |
|                     break;
 | |
|                 default:
 | |
|                     break;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| }
 |