You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							272 lines
						
					
					
						
							10 KiB
						
					
					
				
			
		
		
	
	
							272 lines
						
					
					
						
							10 KiB
						
					
					
				| #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;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| }
 |