291 lines
		
	
	
		
			9.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			291 lines
		
	
	
		
			9.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * The MIT License (MIT)
 | |
|  *
 | |
|  * Copyright (c) 2019 Ha Thach (tinyusb.org)
 | |
|  *
 | |
|  * Permission is hereby granted, free of charge, to any person obtaining a copy
 | |
|  * of this software and associated documentation files (the "Software"), to deal
 | |
|  * in the Software without restriction, including without limitation the rights
 | |
|  * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 | |
|  * copies of the Software, and to permit persons to whom the Software is
 | |
|  * furnished to do so, subject to the following conditions:
 | |
|  *
 | |
|  * The above copyright notice and this permission notice shall be included in
 | |
|  * all copies or substantial portions of the Software.
 | |
|  *
 | |
|  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 | |
|  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 | |
|  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 | |
|  * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 | |
|  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 | |
|  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
 | |
|  * THE SOFTWARE.
 | |
|  *
 | |
|  */
 | |
| 
 | |
| // This example runs both host and device concurrently. The USB host receive
 | |
| // reports from HID device and print it out over USB Device CDC interface.
 | |
| 
 | |
| #include <stdlib.h>
 | |
| #include <stdio.h>
 | |
| #include <string.h>
 | |
| 
 | |
| #include "bsp/board_api.h"
 | |
| #include "tusb.h"
 | |
| 
 | |
| //--------------------------------------------------------------------+
 | |
| // MACRO CONSTANT TYPEDEF PROTYPES
 | |
| //--------------------------------------------------------------------+
 | |
| 
 | |
| // uncomment if you are using colemak layout
 | |
| // #define KEYBOARD_COLEMAK
 | |
| 
 | |
| #ifdef KEYBOARD_COLEMAK
 | |
| const uint8_t colemak[128] = {
 | |
|   0  ,  0,  0,  0,  0,  0,  0, 22,
 | |
|   9  , 23,  7,  0, 24, 17,  8, 12,
 | |
|   0  , 14, 28, 51,  0, 19, 21, 10,
 | |
|   15 ,  0,  0,  0, 13,  0,  0,  0,
 | |
|   0  ,  0,  0,  0,  0,  0,  0,  0,
 | |
|   0  ,  0,  0,  0,  0,  0,  0,  0,
 | |
|   0  ,  0,  0, 18,  0,  0,  0,  0,
 | |
|   0  ,  0,  0,  0,  0,  0,  0,  0,
 | |
|   0  ,  0,  0,  0,  0,  0,  0,  0,
 | |
|   0  ,  0,  0,  0,  0,  0,  0,  0
 | |
| };
 | |
| #endif
 | |
| 
 | |
| static uint8_t const keycode2ascii[128][2] = {HID_KEYCODE_TO_ASCII};
 | |
| 
 | |
| /* Blink pattern
 | |
|  * - 250 ms  : device not mounted
 | |
|  * - 1000 ms : device mounted
 | |
|  * - 2500 ms : device is suspended
 | |
|  */
 | |
| enum {
 | |
|   BLINK_NOT_MOUNTED = 250,
 | |
|   BLINK_MOUNTED = 1000,
 | |
|   BLINK_SUSPENDED = 2500,
 | |
| };
 | |
| 
 | |
| static uint32_t blink_interval_ms = BLINK_NOT_MOUNTED;
 | |
| 
 | |
| void led_blinking_task(void);
 | |
| 
 | |
| /*------------- MAIN -------------*/
 | |
| int main(void) {
 | |
|   board_init();
 | |
| 
 | |
|   printf("TinyUSB Host HID <-> Device CDC Example\r\n");
 | |
| 
 | |
|   // init device and host stack on configured roothub port
 | |
|   tusb_rhport_init_t dev_init = {
 | |
|     .role = TUSB_ROLE_DEVICE,
 | |
|     .speed = TUSB_SPEED_AUTO
 | |
|   };
 | |
|   tusb_init(BOARD_TUD_RHPORT, &dev_init);
 | |
| 
 | |
|   tusb_rhport_init_t host_init = {
 | |
|     .role = TUSB_ROLE_HOST,
 | |
|     .speed = TUSB_SPEED_AUTO
 | |
|   };
 | |
|   tusb_init(BOARD_TUH_RHPORT, &host_init);
 | |
| 
 | |
|   if (board_init_after_tusb) {
 | |
|     board_init_after_tusb();
 | |
|   }
 | |
| 
 | |
|   while (1) {
 | |
|     tud_task(); // tinyusb device task
 | |
|     tuh_task(); // tinyusb host task
 | |
|     led_blinking_task();
 | |
|   }
 | |
| 
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| //--------------------------------------------------------------------+
 | |
| // Device CDC
 | |
| //--------------------------------------------------------------------+
 | |
| 
 | |
| // Invoked when device is mounted
 | |
| void tud_mount_cb(void) {
 | |
|   blink_interval_ms = BLINK_MOUNTED;
 | |
| }
 | |
| 
 | |
| // Invoked when device is unmounted
 | |
| void tud_umount_cb(void) {
 | |
|   blink_interval_ms = BLINK_NOT_MOUNTED;
 | |
| }
 | |
| 
 | |
| // Invoked when usb bus is suspended
 | |
| // remote_wakeup_en : if host allow us  to perform remote wakeup
 | |
| // Within 7ms, device must draw an average of current less than 2.5 mA from bus
 | |
| void tud_suspend_cb(bool remote_wakeup_en) {
 | |
|   (void) remote_wakeup_en;
 | |
|   blink_interval_ms = BLINK_SUSPENDED;
 | |
| }
 | |
| 
 | |
| // Invoked when usb bus is resumed
 | |
| void tud_resume_cb(void) {
 | |
|   blink_interval_ms = tud_mounted() ? BLINK_MOUNTED : BLINK_NOT_MOUNTED;
 | |
| }
 | |
| 
 | |
| // Invoked when CDC interface received data from host
 | |
| void tud_cdc_rx_cb(uint8_t itf) {
 | |
|   (void) itf;
 | |
| 
 | |
|   char buf[64];
 | |
|   uint32_t count = tud_cdc_read(buf, sizeof(buf));
 | |
| 
 | |
|   // TODO control LED on keyboard of host stack
 | |
|   (void) count;
 | |
| }
 | |
| 
 | |
| //--------------------------------------------------------------------+
 | |
| // Host HID
 | |
| //--------------------------------------------------------------------+
 | |
| 
 | |
| // Invoked when device with hid interface is mounted
 | |
| // Report descriptor is also available for use. tuh_hid_parse_report_descriptor()
 | |
| // can be used to parse common/simple enough descriptor.
 | |
| // Note: if report descriptor length > CFG_TUH_ENUMERATION_BUFSIZE, it will be skipped
 | |
| // therefore report_desc = NULL, desc_len = 0
 | |
| void tuh_hid_mount_cb(uint8_t dev_addr, uint8_t instance, uint8_t const* desc_report, uint16_t desc_len) {
 | |
|   (void) desc_report;
 | |
|   (void) desc_len;
 | |
| 
 | |
|   // Interface protocol (hid_interface_protocol_enum_t)
 | |
|   const char* protocol_str[] = {"None", "Keyboard", "Mouse"};
 | |
|   uint8_t const itf_protocol = tuh_hid_interface_protocol(dev_addr, instance);
 | |
| 
 | |
|   uint16_t vid, pid;
 | |
|   tuh_vid_pid_get(dev_addr, &vid, &pid);
 | |
| 
 | |
|   char tempbuf[256];
 | |
|   int count = sprintf(
 | |
|       tempbuf, "[%04x:%04x][%u] HID Interface%u, Protocol = %s\r\n", vid, pid, dev_addr, instance,
 | |
|       protocol_str[itf_protocol]);
 | |
| 
 | |
|   tud_cdc_write(tempbuf, (uint32_t) count);
 | |
|   tud_cdc_write_flush();
 | |
| 
 | |
|   // Receive report from boot keyboard & mouse only
 | |
|   // tuh_hid_report_received_cb() will be invoked when report is available
 | |
|   if (itf_protocol == HID_ITF_PROTOCOL_KEYBOARD || itf_protocol == HID_ITF_PROTOCOL_MOUSE) {
 | |
|     if (!tuh_hid_receive_report(dev_addr, instance)) {
 | |
|       tud_cdc_write_str("Error: cannot request report\r\n");
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| // Invoked when device with hid interface is un-mounted
 | |
| void tuh_hid_umount_cb(uint8_t dev_addr, uint8_t instance) {
 | |
|   char tempbuf[256];
 | |
|   int count = sprintf(tempbuf, "[%u] HID Interface%u is unmounted\r\n", dev_addr, instance);
 | |
|   tud_cdc_write(tempbuf, (uint32_t) count);
 | |
|   tud_cdc_write_flush();
 | |
| }
 | |
| 
 | |
| // look up new key in previous keys
 | |
| static inline bool find_key_in_report(hid_keyboard_report_t const* report, uint8_t keycode) {
 | |
|   for (uint8_t i = 0; i < 6; i++) {
 | |
|     if (report->keycode[i] == keycode) return true;
 | |
|   }
 | |
| 
 | |
|   return false;
 | |
| }
 | |
| 
 | |
| 
 | |
| // convert hid keycode to ascii and print via usb device CDC (ignore non-printable)
 | |
| static void process_kbd_report(uint8_t dev_addr, hid_keyboard_report_t const* report) {
 | |
|   (void) dev_addr;
 | |
|   static hid_keyboard_report_t prev_report = {0, 0, {0}}; // previous report to check key released
 | |
|   bool flush = false;
 | |
| 
 | |
|   for (uint8_t i = 0; i < 6; i++) {
 | |
|     uint8_t keycode = report->keycode[i];
 | |
|     if (keycode) {
 | |
|       if (find_key_in_report(&prev_report, keycode)) {
 | |
|         // exist in previous report means the current key is holding
 | |
|       } else {
 | |
|         // not existed in previous report means the current key is pressed
 | |
| 
 | |
|         // remap the key code for Colemak layout
 | |
|         #ifdef KEYBOARD_COLEMAK
 | |
|         uint8_t colemak_key_code = colemak[keycode];
 | |
|         if (colemak_key_code != 0) keycode = colemak_key_code;
 | |
|         #endif
 | |
| 
 | |
|         bool const is_shift = report->modifier & (KEYBOARD_MODIFIER_LEFTSHIFT | KEYBOARD_MODIFIER_RIGHTSHIFT);
 | |
|         uint8_t ch = keycode2ascii[keycode][is_shift ? 1 : 0];
 | |
| 
 | |
|         if (ch) {
 | |
|           if (ch == '\n') tud_cdc_write("\r", 1);
 | |
|           tud_cdc_write(&ch, 1);
 | |
|           flush = true;
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|     // TODO example skips key released
 | |
|   }
 | |
| 
 | |
|   if (flush) tud_cdc_write_flush();
 | |
| 
 | |
|   prev_report = *report;
 | |
| }
 | |
| 
 | |
| // send mouse report to usb device CDC
 | |
| static void process_mouse_report(uint8_t dev_addr, hid_mouse_report_t const* report) {
 | |
|   //------------- button state  -------------//
 | |
|   //uint8_t button_changed_mask = report->buttons ^ prev_report.buttons;
 | |
|   char l = report->buttons & MOUSE_BUTTON_LEFT ? 'L' : '-';
 | |
|   char m = report->buttons & MOUSE_BUTTON_MIDDLE ? 'M' : '-';
 | |
|   char r = report->buttons & MOUSE_BUTTON_RIGHT ? 'R' : '-';
 | |
| 
 | |
|   char tempbuf[32];
 | |
|   int count = sprintf(tempbuf, "[%u] %c%c%c %d %d %d\r\n", dev_addr, l, m, r, report->x, report->y, report->wheel);
 | |
| 
 | |
|   tud_cdc_write(tempbuf, (uint32_t) count);
 | |
|   tud_cdc_write_flush();
 | |
| }
 | |
| 
 | |
| // Invoked when received report from device via interrupt endpoint
 | |
| void tuh_hid_report_received_cb(uint8_t dev_addr, uint8_t instance, uint8_t const* report, uint16_t len) {
 | |
|   (void) len;
 | |
|   uint8_t const itf_protocol = tuh_hid_interface_protocol(dev_addr, instance);
 | |
| 
 | |
|   switch (itf_protocol) {
 | |
|     case HID_ITF_PROTOCOL_KEYBOARD:
 | |
|       process_kbd_report(dev_addr, (hid_keyboard_report_t const*) report);
 | |
|       break;
 | |
| 
 | |
|     case HID_ITF_PROTOCOL_MOUSE:
 | |
|       process_mouse_report(dev_addr, (hid_mouse_report_t const*) report);
 | |
|       break;
 | |
| 
 | |
|     default:
 | |
|       break;
 | |
|   }
 | |
| 
 | |
|   // continue to request to receive report
 | |
|   if (!tuh_hid_receive_report(dev_addr, instance)) {
 | |
|     tud_cdc_write_str("Error: cannot request report\r\n");
 | |
|   }
 | |
| }
 | |
| 
 | |
| //--------------------------------------------------------------------+
 | |
| // Blinking Task
 | |
| //--------------------------------------------------------------------+
 | |
| void led_blinking_task(void) {
 | |
|   static uint32_t start_ms = 0;
 | |
|   static bool led_state = false;
 | |
| 
 | |
|   // Blink every interval ms
 | |
|   if (board_millis() - start_ms < blink_interval_ms) return; // not enough time
 | |
|   start_ms += blink_interval_ms;
 | |
| 
 | |
|   board_led_write(led_state);
 | |
|   led_state = 1 - led_state; // toggle
 | |
| }
 | 
