2015-04-09 16:32:04 +00:00
/*
Copyright 2011 Jun Wako < wakojun @ gmail . com >
This program is free software : you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation , either version 2 of the License , or
( at your option ) any later version .
This program is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
You should have received a copy of the GNU General Public License
along with this program . If not , see < http : //www.gnu.org/licenses/>.
*/
# include <stdint.h>
2020-06-09 20:30:37 +00:00
# include <avr/wdt.h>
2020-05-30 20:14:59 +00:00
# include <usbdrv/usbdrv.h>
2020-06-09 20:30:37 +00:00
2015-04-09 16:32:04 +00:00
# include "usbconfig.h"
# include "host.h"
# include "report.h"
# include "host_driver.h"
# include "vusb.h"
2020-05-03 00:25:39 +00:00
# include "print.h"
# include "debug.h"
2020-06-09 20:30:37 +00:00
# include "wait.h"
2020-05-21 10:31:42 +00:00
# include "usb_descriptor_common.h"
2015-04-09 16:32:04 +00:00
2020-04-05 22:25:07 +00:00
# ifdef RAW_ENABLE
2020-03-30 20:15:05 +00:00
# include "raw_hid.h"
# endif
2022-11-26 16:14:45 +00:00
# ifdef JOYSTICK_ENABLE
# include "joystick.h"
# endif
2022-03-18 21:37:23 +00:00
# ifdef XAP_ENABLE
# include "xap.h"
# include <string.h>
# endif
2020-05-03 00:25:39 +00:00
# if defined(CONSOLE_ENABLE)
# define RBUF_SIZE 128
# include "ring_buffer.h"
# endif
2022-12-08 16:45:30 +00:00
# ifdef OS_DETECTION_ENABLE
# include "os_detection.h"
# endif
2020-05-03 00:25:39 +00:00
# define NEXT_INTERFACE __COUNTER__
/*
* Interface indexes
*/
enum usb_interfaces {
2021-01-11 08:46:30 +00:00
# ifndef KEYBOARD_SHARED_EP
2020-05-03 00:25:39 +00:00
KEYBOARD_INTERFACE = NEXT_INTERFACE ,
2021-01-11 08:46:30 +00:00
# else
SHARED_INTERFACE = NEXT_INTERFACE ,
# define KEYBOARD_INTERFACE SHARED_INTERFACE
# endif
2020-05-14 00:24:18 +00:00
// It is important that the Raw HID interface is at a constant
// interface number, to support Linux/OSX platforms and chrome.hid
// If Raw HID is enabled, let it be always 1.
2020-05-03 00:25:39 +00:00
# ifdef RAW_ENABLE
RAW_INTERFACE = NEXT_INTERFACE ,
# endif
2021-01-11 08:46:30 +00:00
# if defined(SHARED_EP_ENABLE) && !defined(KEYBOARD_SHARED_EP)
SHARED_INTERFACE = NEXT_INTERFACE ,
2020-05-14 00:24:18 +00:00
# endif
2021-01-11 08:46:30 +00:00
2020-05-03 00:25:39 +00:00
# ifdef CONSOLE_ENABLE
CONSOLE_INTERFACE = NEXT_INTERFACE ,
# endif
2021-01-11 08:46:30 +00:00
2023-01-16 02:03:41 +00:00
# ifdef XAP_ENABLE
XAP_INTERFACE = NEXT_INTERFACE ,
# endif
2020-06-09 20:30:37 +00:00
TOTAL_INTERFACES = NEXT_INTERFACE
2020-05-03 00:25:39 +00:00
} ;
2020-08-19 01:16:35 +00:00
# define MAX_INTERFACES 3
2020-05-03 00:25:39 +00:00
# if (NEXT_INTERFACE - 1) > MAX_INTERFACES
# error There are not enough available interfaces to support all functions. Please disable one or more of the following: Mouse Keys, Extra Keys, Raw HID, Console
2020-03-30 20:15:05 +00:00
# endif
2020-08-19 01:16:35 +00:00
# if (defined(MOUSE_ENABLE) || defined(EXTRAKEY_ENABLE)) && CONSOLE_ENABLE
# error Mouse / Extra Keys share an endpoint with Console. Please disable one of the two.
# endif
2020-06-07 05:00:21 +00:00
static uint8_t keyboard_led_state = 0 ;
2019-08-30 18:19:03 +00:00
static uint8_t vusb_idle_rate = 0 ;
2023-10-23 04:43:46 +00:00
uint8_t keyboard_protocol = 1 ;
2015-04-09 16:32:04 +00:00
/* Keyboard report send buffer */
# define KBUF_SIZE 16
static report_keyboard_t kbuf [ KBUF_SIZE ] ;
2019-08-30 18:19:03 +00:00
static uint8_t kbuf_head = 0 ;
static uint8_t kbuf_tail = 0 ;
2015-04-09 16:32:04 +00:00
2020-06-07 05:00:21 +00:00
static report_keyboard_t keyboard_report_sent ;
2015-04-09 16:32:04 +00:00
2018-04-07 21:07:26 +00:00
# define VUSB_TRANSFER_KEYBOARD_MAX_TRIES 10
2015-04-09 16:32:04 +00:00
/* transfer keyboard report from buffer */
2019-08-30 18:19:03 +00:00
void vusb_transfer_keyboard ( void ) {
2018-04-07 21:07:26 +00:00
for ( int i = 0 ; i < VUSB_TRANSFER_KEYBOARD_MAX_TRIES ; i + + ) {
if ( usbInterruptIsReady ( ) ) {
if ( kbuf_head ! = kbuf_tail ) {
2021-01-11 08:46:30 +00:00
# ifndef KEYBOARD_SHARED_EP
2018-04-07 21:07:26 +00:00
usbSetInterrupt ( ( void * ) & kbuf [ kbuf_tail ] , sizeof ( report_keyboard_t ) ) ;
2021-01-11 08:46:30 +00:00
# else
// Ugly hack! :(
usbSetInterrupt ( ( void * ) & kbuf [ kbuf_tail ] , sizeof ( report_keyboard_t ) - 1 ) ;
while ( ! usbInterruptIsReady ( ) ) {
usbPoll ( ) ;
}
usbSetInterrupt ( ( void * ) ( & ( kbuf [ kbuf_tail ] . keys [ 5 ] ) ) , 1 ) ;
# endif
2018-04-07 21:07:26 +00:00
kbuf_tail = ( kbuf_tail + 1 ) % KBUF_SIZE ;
if ( debug_keyboard ) {
2020-06-09 20:30:37 +00:00
dprintf ( " V-USB: kbuf[%d->%d](%02X) \n " , kbuf_tail , kbuf_head , ( kbuf_head < kbuf_tail ) ? ( KBUF_SIZE - kbuf_tail + kbuf_head ) : ( kbuf_head - kbuf_tail ) ) ;
2018-04-07 21:07:26 +00:00
}
2015-04-09 16:32:04 +00:00
}
2018-04-07 21:07:26 +00:00
break ;
2015-04-09 16:32:04 +00:00
}
2018-04-07 21:07:26 +00:00
usbPoll ( ) ;
2020-06-09 20:30:37 +00:00
wait_ms ( 1 ) ;
2015-04-09 16:32:04 +00:00
}
}
2020-03-30 20:15:05 +00:00
/*------------------------------------------------------------------*
* RAW HID
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
# ifdef RAW_ENABLE
# define RAW_BUFFER_SIZE 32
# define RAW_EPSIZE 8
static uint8_t raw_output_buffer [ RAW_BUFFER_SIZE ] ;
static uint8_t raw_output_received_bytes = 0 ;
void raw_hid_send ( uint8_t * data , uint8_t length ) {
if ( length ! = RAW_BUFFER_SIZE ) {
return ;
}
uint8_t * temp = data ;
2022-03-18 21:37:23 +00:00
for ( uint8_t i = 0 ; i < ( RAW_BUFFER_SIZE / RAW_EPSIZE ) ; i + + ) {
2020-08-19 01:16:35 +00:00
while ( ! usbInterruptIsReady4 ( ) ) {
2020-03-30 20:15:05 +00:00
usbPoll ( ) ;
}
2020-08-19 01:16:35 +00:00
usbSetInterrupt4 ( temp , 8 ) ;
2020-03-30 20:15:05 +00:00
temp + = 8 ;
}
2020-08-19 01:16:35 +00:00
while ( ! usbInterruptIsReady4 ( ) ) {
2020-03-30 20:15:05 +00:00
usbPoll ( ) ;
}
2020-08-19 01:16:35 +00:00
usbSetInterrupt4 ( 0 , 0 ) ;
2020-03-30 20:15:05 +00:00
}
__attribute__ ( ( weak ) ) void raw_hid_receive ( uint8_t * data , uint8_t length ) {
// Users should #include "raw_hid.h" in their own code
// and implement this function there. Leave this as weak linkage
// so users can opt to not handle data coming in.
}
void raw_hid_task ( void ) {
if ( raw_output_received_bytes = = RAW_BUFFER_SIZE ) {
raw_hid_receive ( raw_output_buffer , RAW_BUFFER_SIZE ) ;
raw_output_received_bytes = 0 ;
}
}
2020-05-03 00:25:39 +00:00
# endif
2022-03-18 21:37:23 +00:00
/*------------------------------------------------------------------*
* XAP
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
# ifdef XAP_ENABLE
# define XAP_BUFFER_SIZE 64
# define XAP_EPSIZE 8
static uint8_t xap_output_buffer [ XAP_BUFFER_SIZE ] ;
static uint8_t xap_output_received_bytes = 0 ;
extern void xap_receive ( xap_token_t token , const uint8_t * data , size_t length ) ;
void xap_send_base ( uint8_t * data , uint8_t length ) {
if ( length ! = XAP_BUFFER_SIZE ) {
return ;
}
uint8_t * temp = data ;
for ( uint8_t i = 0 ; i < ( XAP_BUFFER_SIZE / XAP_EPSIZE ) ; i + + ) {
while ( ! usbInterruptIsReady4 ( ) ) {
usbPoll ( ) ;
}
usbSetInterrupt4 ( temp , 8 ) ;
temp + = 8 ;
}
while ( ! usbInterruptIsReady4 ( ) ) {
usbPoll ( ) ;
}
usbSetInterrupt4 ( 0 , 0 ) ;
}
2022-04-05 17:54:28 +00:00
void xap_send ( xap_token_t token , xap_response_flags_t response_flags , const void * data , size_t length ) {
2022-03-30 01:09:19 +00:00
uint8_t rdata [ XAP_BUFFER_SIZE ] = { 0 } ;
xap_response_header_t * header = ( xap_response_header_t * ) & rdata [ 0 ] ;
header - > token = token ;
if ( length > ( XAP_BUFFER_SIZE - sizeof ( xap_response_header_t ) ) ) response_flags & = ~ ( XAP_RESPONSE_FLAG_SUCCESS ) ;
header - > flags = response_flags ;
2022-03-18 21:37:23 +00:00
if ( response_flags & ( XAP_RESPONSE_FLAG_SUCCESS ) ) {
2022-03-30 01:09:19 +00:00
header - > length = ( uint8_t ) length ;
2022-03-18 21:37:23 +00:00
if ( data ! = NULL ) {
2022-03-30 01:09:19 +00:00
memcpy ( & rdata [ sizeof ( xap_response_header_t ) ] , data , length ) ;
2022-03-18 21:37:23 +00:00
}
}
xap_send_base ( rdata , sizeof ( rdata ) ) ;
}
2022-04-05 17:54:28 +00:00
void xap_broadcast ( uint8_t type , const void * data , size_t length ) {
2022-04-05 18:17:48 +00:00
uint8_t rdata [ XAP_BUFFER_SIZE ] = { 0 } ;
xap_broadcast_header_t * header = ( xap_broadcast_header_t * ) & rdata [ 0 ] ;
header - > token = XAP_BROADCAST_TOKEN ;
header - > type = type ;
2022-04-05 17:54:28 +00:00
if ( length > ( XAP_BUFFER_SIZE - sizeof ( xap_broadcast_header_t ) ) ) return ;
header - > length = ( uint8_t ) length ;
if ( data ! = NULL ) {
memcpy ( & rdata [ sizeof ( xap_broadcast_header_t ) ] , data , length ) ;
}
xap_send_base ( rdata , sizeof ( rdata ) ) ;
}
2022-03-18 21:37:23 +00:00
void xap_receive_base ( const void * data ) {
2022-03-30 01:28:03 +00:00
const uint8_t * u8data = ( const uint8_t * ) data ;
2022-03-30 01:09:19 +00:00
xap_request_header_t * header = ( xap_request_header_t * ) & u8data [ 0 ] ;
if ( header - > length < = ( XAP_BUFFER_SIZE - sizeof ( xap_request_header_t ) ) ) {
xap_receive ( header - > token , & u8data [ sizeof ( xap_request_header_t ) ] , header - > length ) ;
2022-03-18 21:37:23 +00:00
}
}
void xap_task ( void ) {
if ( xap_output_received_bytes = = XAP_BUFFER_SIZE ) {
xap_receive_base ( xap_output_buffer ) ;
xap_output_received_bytes = 0 ;
}
}
# endif
2020-05-03 00:25:39 +00:00
/*------------------------------------------------------------------*
* Console
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
# ifdef CONSOLE_ENABLE
# define CONSOLE_BUFFER_SIZE 32
# define CONSOLE_EPSIZE 8
int8_t sendchar ( uint8_t c ) {
rbuf_enqueue ( c ) ;
return 0 ;
}
static inline bool usbSendData3 ( char * data , uint8_t len ) {
uint8_t retries = 5 ;
while ( ! usbInterruptIsReady3 ( ) ) {
if ( ! ( retries - - ) ) {
return false ;
}
usbPoll ( ) ;
}
usbSetInterrupt3 ( ( unsigned char * ) data , len ) ;
return true ;
}
void console_task ( void ) {
if ( ! usbConfiguration ) {
return ;
}
if ( ! rbuf_has_data ( ) ) {
return ;
}
// Send in chunks of 8 padded to 32
char send_buf [ CONSOLE_BUFFER_SIZE ] = { 0 } ;
uint8_t send_buf_count = 0 ;
while ( rbuf_has_data ( ) & & send_buf_count < CONSOLE_EPSIZE ) {
send_buf [ send_buf_count + + ] = rbuf_dequeue ( ) ;
}
2020-03-30 20:15:05 +00:00
2020-05-03 00:25:39 +00:00
char * temp = send_buf ;
for ( uint8_t i = 0 ; i < 4 ; i + + ) {
if ( ! usbSendData3 ( temp , 8 ) ) {
break ;
}
temp + = 8 ;
}
usbSendData3 ( 0 , 0 ) ;
usbPoll ( ) ;
}
2020-03-30 20:15:05 +00:00
# endif
2015-04-09 16:32:04 +00:00
/*------------------------------------------------------------------*
* Host driver
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
static uint8_t keyboard_leds ( void ) ;
2019-08-30 18:19:03 +00:00
static void send_keyboard ( report_keyboard_t * report ) ;
2023-10-23 04:43:46 +00:00
static void send_nkro ( report_nkro_t * report ) ;
2019-08-30 18:19:03 +00:00
static void send_mouse ( report_mouse_t * report ) ;
2022-10-07 02:35:01 +00:00
static void send_extra ( report_extra_t * report ) ;
2015-04-09 16:32:04 +00:00
2023-10-23 04:43:46 +00:00
static host_driver_t driver = { keyboard_leds , send_keyboard , send_nkro , send_mouse , send_extra } ;
2015-04-09 16:32:04 +00:00
2022-02-12 18:29:31 +00:00
host_driver_t * vusb_driver ( void ) {
return & driver ;
}
2019-08-30 18:19:03 +00:00
2022-02-12 18:29:31 +00:00
static uint8_t keyboard_leds ( void ) {
return keyboard_led_state ;
}
2015-04-09 16:32:04 +00:00
2019-08-30 18:19:03 +00:00
static void send_keyboard ( report_keyboard_t * report ) {
2015-04-09 16:32:04 +00:00
uint8_t next = ( kbuf_head + 1 ) % KBUF_SIZE ;
if ( next ! = kbuf_tail ) {
kbuf [ kbuf_head ] = * report ;
2019-08-30 18:19:03 +00:00
kbuf_head = next ;
2015-04-09 16:32:04 +00:00
} else {
2020-06-09 20:30:37 +00:00
dprint ( " kbuf: full \n " ) ;
2015-04-09 16:32:04 +00:00
}
// NOTE: send key strokes of Macro
usbPoll ( ) ;
vusb_transfer_keyboard ( ) ;
2020-06-07 05:00:21 +00:00
keyboard_report_sent = * report ;
2015-04-09 16:32:04 +00:00
}
2023-10-23 04:43:46 +00:00
static void send_nkro ( report_nkro_t * report ) {
// TODO: Implement NKRO
}
2021-01-11 08:46:30 +00:00
# ifndef KEYBOARD_SHARED_EP
# define usbInterruptIsReadyShared usbInterruptIsReady3
# define usbSetInterruptShared usbSetInterrupt3
# else
# define usbInterruptIsReadyShared usbInterruptIsReady
# define usbSetInterruptShared usbSetInterrupt
# endif
2019-08-30 18:19:03 +00:00
static void send_mouse ( report_mouse_t * report ) {
2020-04-05 22:25:07 +00:00
# ifdef MOUSE_ENABLE
2021-01-11 08:46:30 +00:00
if ( usbInterruptIsReadyShared ( ) ) {
usbSetInterruptShared ( ( void * ) report , sizeof ( report_mouse_t ) ) ;
2015-04-09 16:32:04 +00:00
}
2020-04-01 20:06:22 +00:00
# endif
2015-04-09 16:32:04 +00:00
}
2022-10-07 02:35:01 +00:00
static void send_extra ( report_extra_t * report ) {
2022-09-02 20:50:36 +00:00
# ifdef EXTRAKEY_ENABLE
2021-01-11 08:46:30 +00:00
if ( usbInterruptIsReadyShared ( ) ) {
2022-10-07 02:35:01 +00:00
usbSetInterruptShared ( ( void * ) report , sizeof ( report_extra_t ) ) ;
2015-04-09 16:32:04 +00:00
}
2020-02-12 06:19:39 +00:00
# endif
2022-09-02 20:50:36 +00:00
}
2015-04-09 16:32:04 +00:00
2022-11-26 16:14:45 +00:00
void send_joystick ( report_joystick_t * report ) {
# ifdef JOYSTICK_ENABLE
if ( usbInterruptIsReadyShared ( ) ) {
usbSetInterruptShared ( ( void * ) report , sizeof ( report_joystick_t ) ) ;
}
# endif
}
2021-08-17 18:52:44 +00:00
void send_digitizer ( report_digitizer_t * report ) {
# ifdef DIGITIZER_ENABLE
if ( usbInterruptIsReadyShared ( ) ) {
usbSetInterruptShared ( ( void * ) report , sizeof ( report_digitizer_t ) ) ;
2021-11-01 18:57:37 +00:00
}
2021-09-15 15:40:22 +00:00
# endif
}
2022-10-04 22:19:12 +00:00
void send_programmable_button ( report_programmable_button_t * report ) {
2021-09-15 15:40:22 +00:00
# ifdef PROGRAMMABLE_BUTTON_ENABLE
if ( usbInterruptIsReadyShared ( ) ) {
2022-10-04 22:19:12 +00:00
usbSetInterruptShared ( ( void * ) report , sizeof ( report_programmable_button_t ) ) ;
2021-08-17 18:52:44 +00:00
}
# endif
}
2015-04-09 16:32:04 +00:00
/*------------------------------------------------------------------*
* Request from host *
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
static struct {
2019-08-30 18:19:03 +00:00
uint16_t len ;
2019-11-25 20:33:02 +00:00
enum { NONE , SET_LED } kind ;
2015-04-09 16:32:04 +00:00
} last_req ;
2019-08-30 18:19:03 +00:00
usbMsgLen_t usbFunctionSetup ( uchar data [ 8 ] ) {
usbRequest_t * rq = ( void * ) data ;
2015-04-09 16:32:04 +00:00
2019-08-30 18:19:03 +00:00
if ( ( rq - > bmRequestType & USBRQ_TYPE_MASK ) = = USBRQ_TYPE_CLASS ) { /* class request type */
if ( rq - > bRequest = = USBRQ_HID_GET_REPORT ) {
2020-06-09 20:30:37 +00:00
dprint ( " GET_REPORT: " ) ;
2020-06-07 05:00:21 +00:00
if ( rq - > wIndex . word = = KEYBOARD_INTERFACE ) {
usbMsgPtr = ( usbMsgPtr_t ) & keyboard_report_sent ;
return sizeof ( keyboard_report_sent ) ;
}
2019-08-30 18:19:03 +00:00
} else if ( rq - > bRequest = = USBRQ_HID_GET_IDLE ) {
2020-06-09 20:30:37 +00:00
dprint ( " GET_IDLE: " ) ;
2020-05-30 20:14:59 +00:00
usbMsgPtr = ( usbMsgPtr_t ) & vusb_idle_rate ;
2015-04-09 16:32:04 +00:00
return 1 ;
2019-08-30 18:19:03 +00:00
} else if ( rq - > bRequest = = USBRQ_HID_SET_IDLE ) {
2015-04-09 16:32:04 +00:00
vusb_idle_rate = rq - > wValue . bytes [ 1 ] ;
2020-06-09 20:30:37 +00:00
dprintf ( " SET_IDLE: %02X " , vusb_idle_rate ) ;
2019-08-30 18:19:03 +00:00
} else if ( rq - > bRequest = = USBRQ_HID_SET_REPORT ) {
2020-06-09 20:30:37 +00:00
dprint ( " SET_REPORT: " ) ;
2015-04-09 16:32:04 +00:00
// Report Type: 0x02(Out)/ReportID: 0x00(none) && Interface: 0(keyboard)
2020-06-07 05:00:21 +00:00
if ( rq - > wValue . word = = 0x0200 & & rq - > wIndex . word = = KEYBOARD_INTERFACE ) {
2020-06-09 20:30:37 +00:00
dprint ( " SET_LED: " ) ;
2015-04-09 16:32:04 +00:00
last_req . kind = SET_LED ;
2019-08-30 18:19:03 +00:00
last_req . len = rq - > wLength . word ;
2015-04-09 16:32:04 +00:00
}
2022-02-12 18:29:31 +00:00
return USB_NO_MSG ; // to get data in usbFunctionWrite
2015-04-09 16:32:04 +00:00
} else {
2020-06-09 20:30:37 +00:00
dprint ( " UNKNOWN: " ) ;
2015-04-09 16:32:04 +00:00
}
2019-08-30 18:19:03 +00:00
} else {
2020-06-09 20:30:37 +00:00
dprint ( " VENDOR: " ) ;
2015-04-09 16:32:04 +00:00
/* no vendor specific requests implemented */
}
2020-06-09 20:30:37 +00:00
dprint ( " \n " ) ;
2019-08-30 18:19:03 +00:00
return 0 ; /* default for not implemented requests: return no data back to host */
2015-04-09 16:32:04 +00:00
}
2019-08-30 18:19:03 +00:00
uchar usbFunctionWrite ( uchar * data , uchar len ) {
2015-04-09 16:32:04 +00:00
if ( last_req . len = = 0 ) {
return - 1 ;
}
switch ( last_req . kind ) {
case SET_LED :
2020-06-09 20:30:37 +00:00
dprintf ( " SET_LED: %02X \n " , data [ 0 ] ) ;
2020-06-07 05:00:21 +00:00
keyboard_led_state = data [ 0 ] ;
2019-08-30 18:19:03 +00:00
last_req . len = 0 ;
2015-04-09 16:32:04 +00:00
return 1 ;
break ;
case NONE :
default :
return - 1 ;
break ;
}
return 1 ;
}
2020-03-30 20:15:05 +00:00
void usbFunctionWriteOut ( uchar * data , uchar len ) {
# ifdef RAW_ENABLE
// Data from host must be divided every 8bytes
if ( len ! = 8 ) {
2020-06-09 20:30:37 +00:00
dprint ( " RAW: invalid length \n " ) ;
2020-03-30 20:15:05 +00:00
raw_output_received_bytes = 0 ;
return ;
}
if ( raw_output_received_bytes + len > RAW_BUFFER_SIZE ) {
2020-06-09 20:30:37 +00:00
dprint ( " RAW: buffer full \n " ) ;
2020-03-30 20:15:05 +00:00
raw_output_received_bytes = 0 ;
} else {
for ( uint8_t i = 0 ; i < 8 ; i + + ) {
raw_output_buffer [ raw_output_received_bytes + i ] = data [ i ] ;
}
raw_output_received_bytes + = len ;
}
# endif
2022-03-18 21:37:23 +00:00
# ifdef XAP_ENABLE
// Data from host must be divided every 8bytes
if ( len ! = 8 ) {
dprint ( " XAP: invalid length \n " ) ;
xap_output_received_bytes = 0 ;
return ;
}
if ( xap_output_received_bytes + len > XAP_BUFFER_SIZE ) {
dprint ( " XAP: buffer full \n " ) ;
xap_output_received_bytes = 0 ;
} else {
for ( uint8_t i = 0 ; i < 8 ; i + + ) {
xap_output_buffer [ xap_output_received_bytes + i ] = data [ i ] ;
}
xap_output_received_bytes + = len ;
}
# endif
2020-03-30 20:15:05 +00:00
}
2015-04-09 16:32:04 +00:00
/*------------------------------------------------------------------*
* Descriptors *
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
2021-01-11 08:46:30 +00:00
# ifdef KEYBOARD_SHARED_EP
const PROGMEM uchar shared_hid_report [ ] = {
# define SHARED_REPORT_STARTED
# else
2015-04-09 16:32:04 +00:00
const PROGMEM uchar keyboard_hid_report [ ] = {
2021-01-11 08:46:30 +00:00
# endif
2022-02-12 18:29:31 +00:00
0x05 , 0x01 , // Usage Page (Generic Desktop)
0x09 , 0x06 , // Usage (Keyboard)
0xA1 , 0x01 , // Collection (Application)
2021-01-11 08:46:30 +00:00
# ifdef KEYBOARD_SHARED_EP
2022-02-12 18:29:31 +00:00
0x85 , REPORT_ID_KEYBOARD , // Report ID
2021-01-11 08:46:30 +00:00
# endif
2020-02-12 06:19:39 +00:00
// Modifiers (8 bits)
2022-02-12 18:29:31 +00:00
0x05 , 0x07 , // Usage Page (Keyboard/Keypad)
0x19 , 0xE0 , // Usage Minimum (Keyboard Left Control)
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, Variable, Absolute)
2020-02-12 06:19:39 +00:00
// Reserved (1 byte)
2022-02-12 18:29:31 +00:00
0x95 , 0x01 , // Report Count (1)
0x75 , 0x08 , // Report Size (8)
0x81 , 0x03 , // Input (Constant)
2020-02-12 06:19:39 +00:00
// Keycodes (6 bytes)
2022-02-12 18:29:31 +00:00
0x05 , 0x07 , // Usage Page (Keyboard/Keypad)
0x19 , 0x00 , // Usage Minimum (0)
0x29 , 0xFF , // Usage Maximum (255)
0x15 , 0x00 , // Logical Minimum (0)
0x26 , 0xFF , 0x00 , // Logical Maximum (255)
0x95 , 0x06 , // Report Count (6)
0x75 , 0x08 , // Report Size (8)
0x81 , 0x00 , // Input (Data, Array, Absolute)
2020-02-12 06:19:39 +00:00
// Status LEDs (5 bits)
2022-02-12 18:29:31 +00:00
0x05 , 0x08 , // Usage Page (LED)
0x19 , 0x01 , // Usage Minimum (Num Lock)
0x29 , 0x05 , // Usage Maximum (Kana)
2023-07-28 00:05:01 +00:00
0x15 , 0x00 , // Logical Minimum (0)
0x25 , 0x01 , // Logical Maximum (1)
2022-02-12 18:29:31 +00:00
0x95 , 0x05 , // Report Count (5)
0x75 , 0x01 , // Report Size (1)
0x91 , 0x02 , // Output (Data, Variable, Absolute)
2020-02-12 06:19:39 +00:00
// LED padding (3 bits)
2022-02-12 18:29:31 +00:00
0x95 , 0x01 , // Report Count (1)
0x75 , 0x03 , // Report Size (3)
0x91 , 0x03 , // Output (Constant)
0xC0 , // End Collection
2021-01-11 08:46:30 +00:00
# ifndef KEYBOARD_SHARED_EP
2015-04-09 16:32:04 +00:00
} ;
2021-01-11 08:46:30 +00:00
# endif
2015-04-09 16:32:04 +00:00
2021-01-11 08:46:30 +00:00
# if defined(SHARED_EP_ENABLE) && !defined(SHARED_REPORT_STARTED)
const PROGMEM uchar shared_hid_report [ ] = {
# define SHARED_REPORT_STARTED
# endif
# ifdef MOUSE_ENABLE
2020-02-12 06:19:39 +00:00
// Mouse report descriptor
2022-02-12 18:29:31 +00:00
0x05 , 0x01 , // Usage Page (Generic Desktop)
0x09 , 0x02 , // Usage (Mouse)
0xA1 , 0x01 , // Collection (Application)
0x85 , REPORT_ID_MOUSE , // Report ID
0x09 , 0x01 , // Usage (Pointer)
0xA1 , 0x00 , // Collection (Physical)
2021-01-27 17:38:34 +00:00
// Buttons (8 bits)
2022-02-12 18:29:31 +00:00
0x05 , 0x09 , // Usage Page (Button)
0x19 , 0x01 , // Usage Minimum (Button 1)
0x29 , 0x08 , // Usage Maximum (Button 8)
0x15 , 0x00 , // Logical Minimum (0)
0x25 , 0x01 , // Logical Maximum (1)
0x95 , 0x08 , // Report Count (8)
0x75 , 0x01 , // Report Size (1)
0x81 , 0x02 , // Input (Data, Variable, Absolute)
2020-02-12 06:19:39 +00:00
2022-06-09 01:39:16 +00:00
# ifdef MOUSE_EXTENDED_REPORT
// Boot protocol XY ignored in Report protocol
0x95 , 0x02 , // Report Count (2)
0x75 , 0x08 , // Report Size (8)
0x81 , 0x03 , // Input (Constant)
# endif
// X/Y position (2 or 4 bytes)
2022-02-12 18:29:31 +00:00
0x05 , 0x01 , // Usage Page (Generic Desktop)
0x09 , 0x30 , // Usage (X)
0x09 , 0x31 , // Usage (Y)
2022-06-09 01:39:16 +00:00
# ifndef MOUSE_EXTENDED_REPORT
2022-02-12 18:29:31 +00:00
0x15 , 0x81 , // Logical Minimum (-127)
0x25 , 0x7F , // Logical Maximum (127)
0x95 , 0x02 , // Report Count (2)
0x75 , 0x08 , // Report Size (8)
2022-06-09 01:39:16 +00:00
# else
0x16 , 0x01 , 0x80 , // Logical Minimum (-32767)
0x26 , 0xFF , 0x7F , // Logical Maximum (32767)
0x95 , 0x02 , // Report Count (2)
0x75 , 0x10 , // Report Size (16)
# endif
2022-02-12 18:29:31 +00:00
0x81 , 0x06 , // Input (Data, Variable, Relative)
2020-02-12 06:19:39 +00:00
// Vertical wheel (1 byte)
2022-02-12 18:29:31 +00:00
0x09 , 0x38 , // Usage (Wheel)
0x15 , 0x81 , // Logical Minimum (-127)
0x25 , 0x7F , // Logical Maximum (127)
0x95 , 0x01 , // Report Count (1)
0x75 , 0x08 , // Report Size (8)
0x81 , 0x06 , // Input (Data, Variable, Relative)
2020-02-12 06:19:39 +00:00
// Horizontal wheel (1 byte)
2022-02-12 18:29:31 +00:00
0x05 , 0x0C , // Usage Page (Consumer)
0x0A , 0x38 , 0x02 , // Usage (AC Pan)
0x15 , 0x81 , // Logical Minimum (-127)
0x25 , 0x7F , // Logical Maximum (127)
0x95 , 0x01 , // Report Count (1)
0x75 , 0x08 , // Report Size (8)
0x81 , 0x06 , // Input (Data, Variable, Relative)
0xC0 , // End Collection
0xC0 , // End Collection
2021-01-11 08:46:30 +00:00
# endif
2020-02-12 06:19:39 +00:00
2021-01-11 08:46:30 +00:00
# ifdef EXTRAKEY_ENABLE
2020-02-12 06:19:39 +00:00
// Extrakeys report descriptor
2022-02-12 18:29:31 +00:00
0x05 , 0x01 , // Usage Page (Generic Desktop)
0x09 , 0x80 , // Usage (System Control)
0xA1 , 0x01 , // Collection (Application)
0x85 , REPORT_ID_SYSTEM , // Report ID
0x19 , 0x01 , // Usage Minimum (Pointer)
0x2A , 0xB7 , 0x00 , // Usage Maximum (System Display LCD Autoscale)
0x15 , 0x01 , // Logical Minimum
0x26 , 0xB7 , 0x00 , // Logical Maximum
0x95 , 0x01 , // Report Count (1)
0x75 , 0x10 , // Report Size (16)
0x81 , 0x00 , // Input (Data, Array, Absolute)
0xC0 , // End Collection
0x05 , 0x0C , // Usage Page (Consumer)
0x09 , 0x01 , // Usage (Consumer Control)
0xA1 , 0x01 , // Collection (Application)
0x85 , REPORT_ID_CONSUMER , // Report ID
0x19 , 0x01 , // Usage Minimum (Consumer Control)
0x2A , 0xA0 , 0x02 , // Usage Maximum (AC Desktop Show All Applications)
0x15 , 0x01 , // Logical Minimum
0x26 , 0xA0 , 0x02 , // Logical Maximum
0x95 , 0x01 , // Report Count (1)
0x75 , 0x10 , // Report Size (16)
0x81 , 0x00 , // Input (Data, Array, Absolute)
0xC0 , // End Collection
2021-08-17 18:52:44 +00:00
# endif
2022-11-26 16:14:45 +00:00
# ifdef JOYSTICK_ENABLE
// Joystick report descriptor
0x05 , 0x01 , // Usage Page (Generic Desktop)
0x09 , 0x04 , // Usage (Joystick)
0xA1 , 0x01 , // Collection (Application)
0x85 , REPORT_ID_JOYSTICK , // Report ID
0xA1 , 0x00 , // Collection (Physical)
# if JOYSTICK_AXIS_COUNT > 0
0x05 , 0x01 , // Usage Page (Generic Desktop)
0x09 , 0x30 , // Usage (X)
# if JOYSTICK_AXIS_COUNT > 1
0x09 , 0x31 , // Usage (Y)
# endif
# if JOYSTICK_AXIS_COUNT > 2
0x09 , 0x32 , // Usage (Z)
# endif
# if JOYSTICK_AXIS_COUNT > 3
0x09 , 0x33 , // Usage (Rx)
# endif
# if JOYSTICK_AXIS_COUNT > 4
0x09 , 0x34 , // Usage (Ry)
# endif
# if JOYSTICK_AXIS_COUNT > 5
0x09 , 0x35 , // Usage (Rz)
# endif
# if JOYSTICK_AXIS_RESOLUTION == 8
0x15 , - JOYSTICK_MAX_VALUE , // Logical Minimum
0x25 , JOYSTICK_MAX_VALUE , // Logical Maximum
0x95 , JOYSTICK_AXIS_COUNT , // Report Count
0x75 , 0x08 , // Report Size (8)
# else
0x16 , HID_VALUE_16 ( - JOYSTICK_MAX_VALUE ) , // Logical Minimum
0x26 , HID_VALUE_16 ( JOYSTICK_MAX_VALUE ) , // Logical Maximum
0x95 , JOYSTICK_AXIS_COUNT , // Report Count
0x75 , 0x10 , // Report Size (16)
# endif
0x81 , 0x02 , // Input (Data, Variable, Absolute)
# endif
# if JOYSTICK_BUTTON_COUNT > 0
0x05 , 0x09 , // Usage Page (Button)
0x19 , 0x01 , // Usage Minimum (Button 1)
0x29 , JOYSTICK_BUTTON_COUNT , // Usage Maximum
0x15 , 0x00 , // Logical Minimum (0)
0x25 , 0x01 , // Logical Maximum (1)
0x95 , JOYSTICK_BUTTON_COUNT , // Report Count
0x75 , 0x01 , // Report Size (1)
0x81 , 0x02 , // Input (Data, Variable, Absolute)
# if (JOYSTICK_BUTTON_COUNT % 8) != 0
0x95 , 8 - ( JOYSTICK_BUTTON_COUNT % 8 ) , // Report Count
0x75 , 0x01 , // Report Size (1)
0x81 , 0x03 , // Input (Constant)
# endif
# endif
0xC0 , // End Collection
0xC0 , // End Collection
# endif
2021-08-17 18:52:44 +00:00
# ifdef DIGITIZER_ENABLE
// Digitizer report descriptor
2022-02-12 18:29:31 +00:00
0x05 , 0x0D , // Usage Page (Digitizers)
0x09 , 0x01 , // Usage (Digitizer)
0xA1 , 0x01 , // Collection (Application)
0x85 , REPORT_ID_DIGITIZER , // Report ID
2022-11-12 23:28:11 +00:00
0x09 , 0x20 , // Usage (Stylus)
2022-02-12 18:29:31 +00:00
0xA1 , 0x00 , // Collection (Physical)
2022-11-12 23:28:11 +00:00
// In Range, Tip Switch & Barrel Switch (3 bits)
0x09 , 0x32 , // Usage (In Range)
2022-02-12 18:29:31 +00:00
0x09 , 0x42 , // Usage (Tip Switch)
2022-11-12 23:28:11 +00:00
0x09 , 0x44 , // Usage (Barrel Switch)
2022-02-12 18:29:31 +00:00
0x15 , 0x00 , // Logical Minimum
0x25 , 0x01 , // Logical Maximum
2022-11-12 23:28:11 +00:00
0x95 , 0x03 , // Report Count (3)
0x75 , 0x01 , // Report Size (1)
2022-02-12 18:29:31 +00:00
0x81 , 0x02 , // Input (Data, Variable, Absolute)
2022-11-12 23:28:11 +00:00
// Padding (5 bits)
0x95 , 0x05 , // Report Count (5)
2022-02-12 18:29:31 +00:00
0x81 , 0x03 , // Input (Constant)
2021-08-17 18:52:44 +00:00
// X/Y Position (4 bytes)
2022-02-12 18:29:31 +00:00
0x05 , 0x01 , // Usage Page (Generic Desktop)
2022-11-12 23:28:11 +00:00
0x09 , 0x30 , // Usage (X)
0x09 , 0x31 , // Usage (Y)
2022-02-12 18:29:31 +00:00
0x26 , 0xFF , 0x7F , // Logical Maximum (32767)
2022-11-12 23:28:11 +00:00
0x95 , 0x02 , // Report Count (2)
2022-02-12 18:29:31 +00:00
0x75 , 0x10 , // Report Size (16)
0x65 , 0x33 , // Unit (Inch, English Linear)
0x55 , 0x0E , // Unit Exponent (-2)
0x81 , 0x02 , // Input (Data, Variable, Absolute)
0xC0 , // End Collection
0xC0 , // End Collection
2021-01-11 08:46:30 +00:00
# endif
2021-08-17 18:52:44 +00:00
2021-09-15 15:40:22 +00:00
# ifdef PROGRAMMABLE_BUTTON_ENABLE
// Programmable buttons report descriptor
2022-02-12 18:29:31 +00:00
0x05 , 0x0C , // Usage Page (Consumer)
0x09 , 0x01 , // Usage (Consumer Control)
0xA1 , 0x01 , // Collection (Application)
0x85 , REPORT_ID_PROGRAMMABLE_BUTTON , // Report ID
0x09 , 0x03 , // Usage (Programmable Buttons)
0xA1 , 0x04 , // Collection (Named Array)
0x05 , 0x09 , // Usage Page (Button)
0x19 , 0x01 , // Usage Minimum (Button 1)
0x29 , 0x20 , // Usage Maximum (Button 32)
0x15 , 0x00 , // Logical Minimum (0)
0x25 , 0x01 , // Logical Maximum (1)
0x95 , 0x20 , // Report Count (32)
0x75 , 0x01 , // Report Size (1)
0x81 , 0x02 , // Input (Data, Variable, Absolute)
0xC0 , // End Collection
0xC0 , // End Collection
2021-09-15 15:40:22 +00:00
# endif
2021-01-11 08:46:30 +00:00
# ifdef SHARED_EP_ENABLE
2015-04-09 16:32:04 +00:00
} ;
2020-02-25 01:18:11 +00:00
# endif
2015-04-09 16:32:04 +00:00
2020-06-09 20:30:37 +00:00
# ifdef RAW_ENABLE
const PROGMEM uchar raw_hid_report [ ] = {
2022-11-26 16:14:45 +00:00
0x06 , HID_VALUE_16 ( RAW_USAGE_PAGE ) , // Usage Page (Vendor Defined)
0x09 , RAW_USAGE_ID , // Usage (Vendor Defined)
0xA1 , 0x01 , // Collection (Application)
2020-06-09 20:30:37 +00:00
// Data to host
2022-02-12 18:29:31 +00:00
0x09 , 0x62 , // Usage (Vendor Defined)
0x15 , 0x00 , // Logical Minimum (0)
0x26 , 0xFF , 0x00 , // Logical Maximum (255)
0x95 , RAW_BUFFER_SIZE , // Report Count
0x75 , 0x08 , // Report Size (8)
0x81 , 0x02 , // Input (Data, Variable, Absolute)
2020-06-09 20:30:37 +00:00
// Data from host
2022-02-12 18:29:31 +00:00
0x09 , 0x63 , // Usage (Vendor Defined)
0x15 , 0x00 , // Logical Minimum (0)
0x26 , 0xFF , 0x00 , // Logical Maximum (255)
0x95 , RAW_BUFFER_SIZE , // Report Count
0x75 , 0x08 , // Report Size (8)
0x91 , 0x02 , // Output (Data, Variable, Absolute)
0xC0 // End Collection
2020-06-09 20:30:37 +00:00
} ;
# endif
2020-05-03 00:25:39 +00:00
# if defined(CONSOLE_ENABLE)
const PROGMEM uchar console_hid_report [ ] = {
2022-02-12 18:29:31 +00:00
0x06 , 0x31 , 0xFF , // Usage Page (Vendor Defined - PJRC Teensy compatible)
0x09 , 0x74 , // Usage (Vendor Defined - PJRC Teensy compatible)
0xA1 , 0x01 , // Collection (Application)
2020-05-03 00:25:39 +00:00
// Data to host
2022-02-12 18:29:31 +00:00
0x09 , 0x75 , // Usage (Vendor Defined)
0x15 , 0x00 , // Logical Minimum (0x00)
0x26 , 0xFF , 0x00 , // Logical Maximum (0x00FF)
0x95 , CONSOLE_BUFFER_SIZE , // Report Count
0x75 , 0x08 , // Report Size (8)
0x81 , 0x02 , // Input (Data, Variable, Absolute)
2020-05-03 00:25:39 +00:00
// Data from host
2022-02-12 18:29:31 +00:00
0x09 , 0x76 , // Usage (Vendor Defined)
0x15 , 0x00 , // Logical Minimum (0x00)
0x26 , 0xFF , 0x00 , // Logical Maximum (0x00FF)
0x95 , CONSOLE_BUFFER_SIZE , // Report Count
0x75 , 0x08 , // Report Size (8)
0x91 , 0x02 , // Output (Data)
0xC0 // End Collection
2020-03-30 20:15:05 +00:00
} ;
# endif
2023-01-16 02:03:41 +00:00
# ifdef XAP_ENABLE
const PROGMEM uchar xap_report [ ] = {
0x06 , 0x51 , 0xFF , // Usage Page (Vendor Defined)
0x09 , 0x58 , // Usage (Vendor Defined)
0xA1 , 0x01 , // Collection (Application)
// Data to host
0x09 , 0x62 , // Usage (Vendor Defined)
0x15 , 0x00 , // Logical Minimum (0)
0x26 , 0xFF , 0x00 , // Logical Maximum (255)
0x95 , XAP_BUFFER_SIZE , // Report Count
0x75 , 0x08 , // Report Size (8)
0x81 , 0x02 , // Input (Data, Variable, Absolute)
// Data from host
0x09 , 0x63 , // Usage (Vendor Defined)
0x15 , 0x00 , // Logical Minimum (0)
0x26 , 0xFF , 0x00 , // Logical Maximum (255)
0x95 , XAP_BUFFER_SIZE , // Report Count
0x75 , 0x08 , // Report Size (8)
0x91 , 0x02 , // Output (Data, Variable, Absolute)
0xC0 // End Collection
} ;
# endif
2019-11-14 23:21:22 +00:00
# ifndef USB_MAX_POWER_CONSUMPTION
# define USB_MAX_POWER_CONSUMPTION 500
# endif
# ifndef USB_POLLING_INTERVAL_MS
# define USB_POLLING_INTERVAL_MS 1
# endif
2020-03-26 13:11:32 +00:00
// clang-format off
2020-03-28 02:02:25 +00:00
const PROGMEM usbStringDescriptor_t usbStringDescriptorZero = {
2020-03-26 13:11:32 +00:00
. header = {
2022-08-28 16:59:40 +00:00
. bLength = 4 ,
2020-03-26 13:11:32 +00:00
. bDescriptorType = USBDESCR_STRING
} ,
2020-03-28 02:02:25 +00:00
. bString = { 0x0409 } // US English
2020-03-26 13:11:32 +00:00
} ;
2020-03-28 02:02:25 +00:00
const PROGMEM usbStringDescriptor_t usbStringDescriptorManufacturer = {
2020-03-26 13:11:32 +00:00
. header = {
2022-08-28 16:59:40 +00:00
. bLength = sizeof ( USBSTR ( MANUFACTURER ) ) ,
2020-03-26 13:11:32 +00:00
. bDescriptorType = USBDESCR_STRING
} ,
2022-08-28 16:59:40 +00:00
. bString = USBSTR ( MANUFACTURER )
2020-03-26 13:11:32 +00:00
} ;
2020-03-28 02:02:25 +00:00
const PROGMEM usbStringDescriptor_t usbStringDescriptorProduct = {
2020-03-26 13:11:32 +00:00
. header = {
2022-08-28 16:59:40 +00:00
. bLength = sizeof ( USBSTR ( PRODUCT ) ) ,
2020-03-26 13:11:32 +00:00
. bDescriptorType = USBDESCR_STRING
} ,
2022-08-28 16:59:40 +00:00
. bString = USBSTR ( PRODUCT )
2020-03-26 13:11:32 +00:00
} ;
2020-12-05 10:14:19 +00:00
# if defined(SERIAL_NUMBER)
2020-03-28 02:02:25 +00:00
const PROGMEM usbStringDescriptor_t usbStringDescriptorSerial = {
2020-03-26 13:11:32 +00:00
. header = {
2022-08-28 16:59:40 +00:00
. bLength = sizeof ( USBSTR ( SERIAL_NUMBER ) ) ,
2020-03-26 13:11:32 +00:00
. bDescriptorType = USBDESCR_STRING
} ,
2021-07-16 09:43:18 +00:00
. bString = USBSTR ( SERIAL_NUMBER )
2020-03-26 13:11:32 +00:00
} ;
2020-12-05 10:14:19 +00:00
# endif
2020-03-26 13:11:32 +00:00
2019-08-30 18:19:03 +00:00
/*
2020-03-28 02:02:25 +00:00
* Device descriptor
2015-04-09 16:32:04 +00:00
*/
2020-03-28 02:02:25 +00:00
const PROGMEM usbDeviceDescriptor_t usbDeviceDescriptor = {
. header = {
. bLength = sizeof ( usbDeviceDescriptor_t ) ,
. bDescriptorType = USBDESCR_DEVICE
} ,
. bcdUSB = 0x0110 ,
2020-04-05 22:25:07 +00:00
. bDeviceClass = 0x00 ,
. bDeviceSubClass = 0x00 ,
2020-03-28 02:02:25 +00:00
. bDeviceProtocol = 0x00 ,
. bMaxPacketSize0 = 8 ,
. idVendor = VENDOR_ID ,
. idProduct = PRODUCT_ID ,
. bcdDevice = DEVICE_VER ,
. iManufacturer = 0x01 ,
. iProduct = 0x02 ,
2020-12-05 10:14:19 +00:00
# if defined(SERIAL_NUMBER)
2020-03-28 02:02:25 +00:00
. iSerialNumber = 0x03 ,
2020-12-05 10:14:19 +00:00
# else
. iSerialNumber = 0x00 ,
# endif
2020-03-28 02:02:25 +00:00
. bNumConfigurations = 1
} ;
/*
* Configuration descriptors
*/
const PROGMEM usbConfigurationDescriptor_t usbConfigurationDescriptor = {
. header = {
. header = {
. bLength = sizeof ( usbConfigurationDescriptorHeader_t ) ,
. bDescriptorType = USBDESCR_CONFIG
} ,
. wTotalLength = sizeof ( usbConfigurationDescriptor_t ) ,
2020-05-03 00:25:39 +00:00
. bNumInterfaces = TOTAL_INTERFACES ,
2020-03-28 02:02:25 +00:00
. bConfigurationValue = 0x01 ,
. iConfiguration = 0x00 ,
2020-04-05 08:18:19 +00:00
. bmAttributes = ( 1 < < 7 ) | USBATTR_REMOTEWAKE ,
2020-03-28 02:02:25 +00:00
. bMaxPower = USB_MAX_POWER_CONSUMPTION / 2
} ,
2015-04-09 16:32:04 +00:00
2021-01-11 08:46:30 +00:00
# ifndef KEYBOARD_SHARED_EP
2015-04-09 16:32:04 +00:00
/*
2020-03-28 02:02:25 +00:00
* Keyboard
2015-04-09 16:32:04 +00:00
*/
2020-03-28 02:02:25 +00:00
. keyboardInterface = {
. header = {
. bLength = sizeof ( usbInterfaceDescriptor_t ) ,
. bDescriptorType = USBDESCR_INTERFACE
} ,
2020-05-03 00:25:39 +00:00
. bInterfaceNumber = KEYBOARD_INTERFACE ,
2020-03-28 02:02:25 +00:00
. bAlternateSetting = 0x00 ,
2020-04-05 22:25:07 +00:00
. bNumEndpoints = 1 ,
. bInterfaceClass = 0x03 ,
. bInterfaceSubClass = 0x01 ,
. bInterfaceProtocol = 0x01 ,
2020-03-28 02:02:25 +00:00
. iInterface = 0x00
} ,
. keyboardHID = {
. header = {
. bLength = sizeof ( usbHIDDescriptor_t ) ,
. bDescriptorType = USBDESCR_HID
} ,
. bcdHID = 0x0101 ,
. bCountryCode = 0x00 ,
. bNumDescriptors = 1 ,
. bDescriptorType = USBDESCR_HID_REPORT ,
. wDescriptorLength = sizeof ( keyboard_hid_report )
} ,
. keyboardINEndpoint = {
. header = {
. bLength = sizeof ( usbEndpointDescriptor_t ) ,
. bDescriptorType = USBDESCR_ENDPOINT
} ,
. bEndpointAddress = ( USBRQ_DIR_DEVICE_TO_HOST | 1 ) ,
. bmAttributes = 0x03 ,
. wMaxPacketSize = 8 ,
. bInterval = USB_POLLING_INTERVAL_MS
} ,
2021-01-11 08:46:30 +00:00
# endif
2015-04-09 16:32:04 +00:00
2020-05-14 00:24:18 +00:00
# if defined(RAW_ENABLE)
2015-04-09 16:32:04 +00:00
/*
2020-05-14 00:24:18 +00:00
* RAW HID
2015-04-09 16:32:04 +00:00
*/
2020-05-14 00:24:18 +00:00
. rawInterface = {
2020-03-28 02:02:25 +00:00
. header = {
. bLength = sizeof ( usbInterfaceDescriptor_t ) ,
. bDescriptorType = USBDESCR_INTERFACE
} ,
2020-05-14 00:24:18 +00:00
. bInterfaceNumber = RAW_INTERFACE ,
2020-03-28 02:02:25 +00:00
. bAlternateSetting = 0x00 ,
2020-05-14 00:24:18 +00:00
. bNumEndpoints = 2 ,
2020-03-28 02:02:25 +00:00
. bInterfaceClass = 0x03 ,
. bInterfaceSubClass = 0x00 ,
. bInterfaceProtocol = 0x00 ,
. iInterface = 0x00
} ,
2020-05-14 00:24:18 +00:00
. rawHID = {
2020-03-28 02:02:25 +00:00
. header = {
. bLength = sizeof ( usbHIDDescriptor_t ) ,
. bDescriptorType = USBDESCR_HID
} ,
. bcdHID = 0x0101 ,
. bCountryCode = 0x00 ,
2020-12-21 02:12:07 +00:00
. bNumDescriptors = 1 ,
2020-03-28 02:02:25 +00:00
. bDescriptorType = USBDESCR_HID_REPORT ,
2020-05-14 00:24:18 +00:00
. wDescriptorLength = sizeof ( raw_hid_report )
2020-03-28 02:02:25 +00:00
} ,
2020-05-14 00:24:18 +00:00
. rawINEndpoint = {
2020-03-28 02:02:25 +00:00
. header = {
. bLength = sizeof ( usbEndpointDescriptor_t ) ,
. bDescriptorType = USBDESCR_ENDPOINT
} ,
2020-08-19 01:16:35 +00:00
. bEndpointAddress = ( USBRQ_DIR_DEVICE_TO_HOST | USB_CFG_EP4_NUMBER ) ,
2020-03-28 02:02:25 +00:00
. bmAttributes = 0x03 ,
2020-05-14 00:24:18 +00:00
. wMaxPacketSize = RAW_EPSIZE ,
. bInterval = USB_POLLING_INTERVAL_MS
} ,
. rawOUTEndpoint = {
. header = {
. bLength = sizeof ( usbEndpointDescriptor_t ) ,
. bDescriptorType = USBDESCR_ENDPOINT
} ,
2020-08-19 01:16:35 +00:00
. bEndpointAddress = ( USBRQ_DIR_HOST_TO_DEVICE | USB_CFG_EP4_NUMBER ) ,
2020-05-14 00:24:18 +00:00
. bmAttributes = 0x03 ,
. wMaxPacketSize = RAW_EPSIZE ,
2020-03-28 02:02:25 +00:00
. bInterval = USB_POLLING_INTERVAL_MS
2020-05-03 00:25:39 +00:00
} ,
# endif
2021-01-11 08:46:30 +00:00
# ifdef SHARED_EP_ENABLE
2020-05-14 00:24:18 +00:00
/*
2021-01-11 08:46:30 +00:00
* Shared
2020-05-14 00:24:18 +00:00
*/
2021-01-11 08:46:30 +00:00
. sharedInterface = {
2020-03-30 20:15:05 +00:00
. header = {
. bLength = sizeof ( usbInterfaceDescriptor_t ) ,
. bDescriptorType = USBDESCR_INTERFACE
} ,
2021-01-11 08:46:30 +00:00
. bInterfaceNumber = SHARED_INTERFACE ,
2020-03-30 20:15:05 +00:00
. bAlternateSetting = 0x00 ,
2020-05-14 00:24:18 +00:00
. bNumEndpoints = 1 ,
2020-03-30 20:15:05 +00:00
. bInterfaceClass = 0x03 ,
2021-01-11 08:46:30 +00:00
# ifdef KEYBOARD_SHARED_EP
. bInterfaceSubClass = 0x01 ,
. bInterfaceProtocol = 0x01 ,
# else
2020-03-30 20:15:05 +00:00
. bInterfaceSubClass = 0x00 ,
. bInterfaceProtocol = 0x00 ,
2021-01-11 08:46:30 +00:00
# endif
2020-03-30 20:15:05 +00:00
. iInterface = 0x00
} ,
2021-01-11 08:46:30 +00:00
. sharedHID = {
2020-03-30 20:15:05 +00:00
. header = {
. bLength = sizeof ( usbHIDDescriptor_t ) ,
. bDescriptorType = USBDESCR_HID
} ,
. bcdHID = 0x0101 ,
. bCountryCode = 0x00 ,
2020-05-14 00:24:18 +00:00
. bNumDescriptors = 1 ,
2020-03-30 20:15:05 +00:00
. bDescriptorType = USBDESCR_HID_REPORT ,
2021-01-11 08:46:30 +00:00
. wDescriptorLength = sizeof ( shared_hid_report )
2020-03-30 20:15:05 +00:00
} ,
2021-01-11 08:46:30 +00:00
. sharedINEndpoint = {
2020-03-30 20:15:05 +00:00
. header = {
. bLength = sizeof ( usbEndpointDescriptor_t ) ,
. bDescriptorType = USBDESCR_ENDPOINT
} ,
2021-01-11 08:46:30 +00:00
# ifdef KEYBOARD_SHARED_EP
. bEndpointAddress = ( USBRQ_DIR_DEVICE_TO_HOST | 1 ) ,
# else
2020-03-30 20:15:05 +00:00
. bEndpointAddress = ( USBRQ_DIR_DEVICE_TO_HOST | USB_CFG_EP3_NUMBER ) ,
2021-01-11 08:46:30 +00:00
# endif
2020-03-30 20:15:05 +00:00
. bmAttributes = 0x03 ,
2020-05-14 00:24:18 +00:00
. wMaxPacketSize = 8 ,
2020-03-30 20:15:05 +00:00
. bInterval = USB_POLLING_INTERVAL_MS
2020-05-03 00:25:39 +00:00
} ,
# endif
2021-01-11 08:46:30 +00:00
2020-05-03 00:25:39 +00:00
# if defined(CONSOLE_ENABLE)
/*
* Console
*/
. consoleInterface = {
. header = {
. bLength = sizeof ( usbInterfaceDescriptor_t ) ,
. bDescriptorType = USBDESCR_INTERFACE
} ,
. bInterfaceNumber = CONSOLE_INTERFACE ,
. bAlternateSetting = 0x00 ,
. bNumEndpoints = 2 ,
. bInterfaceClass = 0x03 ,
. bInterfaceSubClass = 0x00 ,
. bInterfaceProtocol = 0x00 ,
. iInterface = 0x00
} ,
. consoleHID = {
. header = {
. bLength = sizeof ( usbHIDDescriptor_t ) ,
. bDescriptorType = USBDESCR_HID
} ,
. bcdHID = 0x0111 ,
. bCountryCode = 0x00 ,
. bNumDescriptors = 1 ,
. bDescriptorType = USBDESCR_HID_REPORT ,
. wDescriptorLength = sizeof ( console_hid_report )
} ,
. consoleINEndpoint = {
. header = {
. bLength = sizeof ( usbEndpointDescriptor_t ) ,
. bDescriptorType = USBDESCR_ENDPOINT
} ,
. bEndpointAddress = ( USBRQ_DIR_DEVICE_TO_HOST | USB_CFG_EP3_NUMBER ) ,
. bmAttributes = 0x03 ,
. wMaxPacketSize = CONSOLE_EPSIZE ,
. bInterval = 0x01
} ,
. consoleOUTEndpoint = {
. header = {
. bLength = sizeof ( usbEndpointDescriptor_t ) ,
. bDescriptorType = USBDESCR_ENDPOINT
} ,
. bEndpointAddress = ( USBRQ_DIR_HOST_TO_DEVICE | USB_CFG_EP3_NUMBER ) ,
. bmAttributes = 0x03 ,
. wMaxPacketSize = CONSOLE_EPSIZE ,
. bInterval = 0x01
2023-01-16 02:03:41 +00:00
} ,
# endif
# if defined(XAP_ENABLE)
/*
* XAP
*/
. xapInterface = {
. header = {
. bLength = sizeof ( usbInterfaceDescriptor_t ) ,
. bDescriptorType = USBDESCR_INTERFACE
} ,
. bInterfaceNumber = XAP_INTERFACE ,
. bAlternateSetting = 0x00 ,
. bNumEndpoints = 2 ,
. bInterfaceClass = 0x03 ,
. bInterfaceSubClass = 0x00 ,
. bInterfaceProtocol = 0x00 ,
. iInterface = 0x00
} ,
. xapHID = {
. header = {
. bLength = sizeof ( usbHIDDescriptor_t ) ,
. bDescriptorType = USBDESCR_HID
} ,
. bcdHID = 0x0101 ,
. bCountryCode = 0x00 ,
. bNumDescriptors = 1 ,
. bDescriptorType = USBDESCR_HID_REPORT ,
. wDescriptorLength = sizeof ( xap_report )
} ,
. xapINEndpoint = {
. header = {
. bLength = sizeof ( usbEndpointDescriptor_t ) ,
. bDescriptorType = USBDESCR_ENDPOINT
} ,
. bEndpointAddress = ( USBRQ_DIR_DEVICE_TO_HOST | USB_CFG_EP4_NUMBER ) ,
. bmAttributes = 0x03 ,
. wMaxPacketSize = XAP_EPSIZE ,
. bInterval = USB_POLLING_INTERVAL_MS
} ,
. xapOUTEndpoint = {
. header = {
. bLength = sizeof ( usbEndpointDescriptor_t ) ,
. bDescriptorType = USBDESCR_ENDPOINT
} ,
. bEndpointAddress = ( USBRQ_DIR_HOST_TO_DEVICE | USB_CFG_EP4_NUMBER ) ,
. bmAttributes = 0x03 ,
. wMaxPacketSize = XAP_EPSIZE ,
. bInterval = USB_POLLING_INTERVAL_MS
} ,
2019-08-30 18:19:03 +00:00
# endif
2015-04-09 16:32:04 +00:00
} ;
2020-03-28 02:02:25 +00:00
// clang-format on
2019-08-30 18:19:03 +00:00
USB_PUBLIC usbMsgLen_t usbFunctionDescriptor ( struct usbRequest * rq ) {
2015-04-09 16:32:04 +00:00
usbMsgLen_t len = 0 ;
switch ( rq - > wValue . bytes [ 1 ] ) {
2020-03-28 02:02:25 +00:00
case USBDESCR_DEVICE :
2020-05-30 20:14:59 +00:00
usbMsgPtr = ( usbMsgPtr_t ) & usbDeviceDescriptor ;
2020-03-28 02:02:25 +00:00
len = sizeof ( usbDeviceDescriptor_t ) ;
break ;
2015-04-09 16:32:04 +00:00
case USBDESCR_CONFIG :
2020-05-30 20:14:59 +00:00
usbMsgPtr = ( usbMsgPtr_t ) & usbConfigurationDescriptor ;
2020-03-28 02:02:25 +00:00
len = sizeof ( usbConfigurationDescriptor_t ) ;
2015-04-09 16:32:04 +00:00
break ;
2020-03-26 13:11:32 +00:00
case USBDESCR_STRING :
switch ( rq - > wValue . bytes [ 0 ] ) {
case 0 :
2020-05-30 20:14:59 +00:00
usbMsgPtr = ( usbMsgPtr_t ) & usbStringDescriptorZero ;
2020-03-28 02:02:25 +00:00
len = usbStringDescriptorZero . header . bLength ;
2020-03-26 13:11:32 +00:00
break ;
2022-02-12 18:29:31 +00:00
case 1 : // iManufacturer
2020-05-30 20:14:59 +00:00
usbMsgPtr = ( usbMsgPtr_t ) & usbStringDescriptorManufacturer ;
2020-03-28 02:02:25 +00:00
len = usbStringDescriptorManufacturer . header . bLength ;
2020-03-26 13:11:32 +00:00
break ;
2022-02-12 18:29:31 +00:00
case 2 : // iProduct
2020-05-30 20:14:59 +00:00
usbMsgPtr = ( usbMsgPtr_t ) & usbStringDescriptorProduct ;
2020-03-28 02:02:25 +00:00
len = usbStringDescriptorProduct . header . bLength ;
2020-03-26 13:11:32 +00:00
break ;
2020-12-05 10:14:19 +00:00
# if defined(SERIAL_NUMBER)
2022-02-12 18:29:31 +00:00
case 3 : // iSerialNumber
2020-05-30 20:14:59 +00:00
usbMsgPtr = ( usbMsgPtr_t ) & usbStringDescriptorSerial ;
2020-03-28 02:02:25 +00:00
len = usbStringDescriptorSerial . header . bLength ;
2020-03-26 13:11:32 +00:00
break ;
2020-12-05 10:14:19 +00:00
# endif
2020-03-26 13:11:32 +00:00
}
2022-12-08 16:45:30 +00:00
# ifdef OS_DETECTION_ENABLE
process_wlength ( rq - > wLength . word ) ;
# endif
2020-03-26 13:11:32 +00:00
break ;
2015-04-09 16:32:04 +00:00
case USBDESCR_HID :
switch ( rq - > wValue . bytes [ 0 ] ) {
2021-01-11 08:46:30 +00:00
# ifndef KEYBOARD_SHARED_EP
2020-05-03 00:25:39 +00:00
case KEYBOARD_INTERFACE :
2020-05-30 20:14:59 +00:00
usbMsgPtr = ( usbMsgPtr_t ) & usbConfigurationDescriptor . keyboardHID ;
2020-03-28 02:02:25 +00:00
len = sizeof ( usbHIDDescriptor_t ) ;
2015-04-09 16:32:04 +00:00
break ;
2021-01-11 08:46:30 +00:00
# endif
2020-05-03 00:25:39 +00:00
# if defined(RAW_ENABLE)
case RAW_INTERFACE :
2020-05-30 20:14:59 +00:00
usbMsgPtr = ( usbMsgPtr_t ) & usbConfigurationDescriptor . rawHID ;
2020-03-30 20:15:05 +00:00
len = sizeof ( usbHIDDescriptor_t ) ;
break ;
2020-05-03 00:25:39 +00:00
# endif
2021-01-11 08:46:30 +00:00
# ifdef SHARED_EP_ENABLE
case SHARED_INTERFACE :
usbMsgPtr = ( usbMsgPtr_t ) & usbConfigurationDescriptor . sharedHID ;
2020-05-14 00:24:18 +00:00
len = sizeof ( usbHIDDescriptor_t ) ;
break ;
# endif
2021-01-11 08:46:30 +00:00
2020-05-03 00:25:39 +00:00
# if defined(CONSOLE_ENABLE)
case CONSOLE_INTERFACE :
2020-05-30 20:14:59 +00:00
usbMsgPtr = ( usbMsgPtr_t ) & usbConfigurationDescriptor . consoleHID ;
2020-05-03 00:25:39 +00:00
len = sizeof ( usbHIDDescriptor_t ) ;
break ;
2020-02-25 01:18:11 +00:00
# endif
2023-01-16 02:03:41 +00:00
# if defined(XAP_ENABLE)
case XAP_INTERFACE :
usbMsgPtr = ( usbMsgPtr_t ) & usbConfigurationDescriptor . xapHID ;
len = sizeof ( usbHIDDescriptor_t ) ;
break ;
# endif
2015-04-09 16:32:04 +00:00
}
break ;
case USBDESCR_HID_REPORT :
/* interface index */
switch ( rq - > wIndex . word ) {
2021-01-11 08:46:30 +00:00
# ifndef KEYBOARD_SHARED_EP
2020-05-03 00:25:39 +00:00
case KEYBOARD_INTERFACE :
2020-05-30 20:14:59 +00:00
usbMsgPtr = ( usbMsgPtr_t ) keyboard_hid_report ;
2019-08-30 18:19:03 +00:00
len = sizeof ( keyboard_hid_report ) ;
2015-04-09 16:32:04 +00:00
break ;
2021-01-11 08:46:30 +00:00
# endif
2020-05-03 00:25:39 +00:00
# if defined(RAW_ENABLE)
case RAW_INTERFACE :
2020-05-30 20:14:59 +00:00
usbMsgPtr = ( usbMsgPtr_t ) raw_hid_report ;
2020-03-30 20:15:05 +00:00
len = sizeof ( raw_hid_report ) ;
break ;
2020-05-03 00:25:39 +00:00
# endif
2021-01-11 08:46:30 +00:00
# ifdef SHARED_EP_ENABLE
case SHARED_INTERFACE :
usbMsgPtr = ( usbMsgPtr_t ) shared_hid_report ;
len = sizeof ( shared_hid_report ) ;
2020-05-14 00:24:18 +00:00
break ;
# endif
2021-01-11 08:46:30 +00:00
2020-05-03 00:25:39 +00:00
# if defined(CONSOLE_ENABLE)
case CONSOLE_INTERFACE :
2020-05-30 20:14:59 +00:00
usbMsgPtr = ( usbMsgPtr_t ) console_hid_report ;
2020-05-03 00:25:39 +00:00
len = sizeof ( console_hid_report ) ;
break ;
2020-02-25 01:18:11 +00:00
# endif
2023-01-16 02:03:41 +00:00
# if defined(XAP_ENABLE)
case XAP_INTERFACE :
usbMsgPtr = ( usbMsgPtr_t ) xap_report ;
len = sizeof ( xap_report ) ;
break ;
# endif
2015-04-09 16:32:04 +00:00
}
break ;
}
return len ;
}