1
1
/*
2
2
Copyright (c) 2015, Arduino LLC
3
3
Original code (pre-library): Copyright (c) 2011, Peter Barrett
4
+ Modified code: Copyright (c) 2020, Aleksandr Bratchik
4
5
5
6
Permission to use, copy, modify, and/or distribute this software for
6
7
any purpose with or without fee is hereby granted, provided that the
@@ -30,18 +31,35 @@ int HID_::getInterface(uint8_t* interfaceCount)
30
31
{
31
32
*interfaceCount += 1 ; // uses 1
32
33
HIDDescriptor hidInterface = {
33
- D_INTERFACE (pluggedInterface, 1 , USB_DEVICE_CLASS_HUMAN_INTERFACE, HID_SUBCLASS_NONE, HID_PROTOCOL_NONE),
34
+ D_INTERFACE (pluggedInterface, 2 , USB_DEVICE_CLASS_HUMAN_INTERFACE, HID_SUBCLASS_NONE, HID_PROTOCOL_NONE),
34
35
D_HIDREPORT (descriptorSize),
35
- D_ENDPOINT (USB_ENDPOINT_IN (pluggedEndpoint), USB_ENDPOINT_TYPE_INTERRUPT, USB_EP_SIZE, 0x01 )
36
+ D_ENDPOINT (USB_ENDPOINT_IN (HID_TX), USB_ENDPOINT_TYPE_INTERRUPT, USB_EP_SIZE, 0x14 ),
37
+ D_ENDPOINT (USB_ENDPOINT_OUT (HID_RX), USB_ENDPOINT_TYPE_INTERRUPT, USB_EP_SIZE, 0x0A )
36
38
};
37
39
return USB_SendControl (0 , &hidInterface, sizeof (hidInterface));
38
40
}
39
41
40
42
int HID_::getDescriptor (USBSetup& setup)
41
43
{
44
+
45
+ u8 t = setup.wValueH ;
46
+
47
+ // HID-specific strings
48
+ if (USB_STRING_DESCRIPTOR_TYPE == t) {
49
+
50
+ // we place all strings in the 0xFF00-0xFFFE range
51
+ HIDReport* rep = GetFeature (0xFF00 | setup.wValueL );
52
+ if (rep) {
53
+ return USB_SendStringDescriptor ((u8*)rep->data , strlen_P ((char *)rep->data ), TRANSFER_PGM);
54
+ }
55
+ else {
56
+ return 0 ;
57
+ }
58
+ }
59
+
42
60
// Check if this is a HID Class Descriptor request
43
61
if (setup.bmRequestType != REQUEST_DEVICETOHOST_STANDARD_INTERFACE) { return 0 ; }
44
- if (setup. wValueH != HID_REPORT_DESCRIPTOR_TYPE ) { return 0 ; }
62
+ if (HID_REPORT_DESCRIPTOR_TYPE != t ) { return 0 ; }
45
63
46
64
// In a HID Class Descriptor wIndex cointains the interface number
47
65
if (setup.wIndex != pluggedInterface) { return 0 ; }
@@ -64,12 +82,24 @@ int HID_::getDescriptor(USBSetup& setup)
64
82
65
83
uint8_t HID_::getShortName (char *name)
66
84
{
85
+ if (serial) {
86
+ byte seriallen=min (strlen_P (serial), ISERIAL_MAX_LEN);
87
+ for (byte i=0 ; i< seriallen; i++) {
88
+ name[i] = pgm_read_byte_near (serial + i);
89
+ }
90
+ return seriallen;
91
+ }
92
+ else {
93
+
94
+ // default serial number
95
+
67
96
name[0 ] = ' H' ;
68
97
name[1 ] = ' I' ;
69
98
name[2 ] = ' D' ;
70
99
name[3 ] = ' A' + (descriptorSize & 0x0F );
71
100
name[4 ] = ' A' + ((descriptorSize >> 4 ) & 0x0F );
72
101
return 5 ;
102
+ }
73
103
}
74
104
75
105
void HID_::AppendDescriptor (HIDSubDescriptor *node)
@@ -86,28 +116,102 @@ void HID_::AppendDescriptor(HIDSubDescriptor *node)
86
116
descriptorSize += node->length ;
87
117
}
88
118
89
- int HID_::SendReport ( uint8_t id, const void * data, int len)
119
+ int HID_::SetFeature ( uint16_t id, const void * data, int len)
90
120
{
91
- auto ret = USB_Send (pluggedEndpoint, &id, 1 );
121
+ if (!rootReport) {
122
+ rootReport = new HIDReport (id, data, len);
123
+ } else {
124
+ HIDReport* current;
125
+ int i=0 ;
126
+ for ( current = rootReport; current; current = current->next , i++) {
127
+ if (current->id == id) {
128
+ return i;
129
+ }
130
+ // check if we are on the last report
131
+ if (!current->next ) {
132
+ current->next = new HIDReport (id, data, len);
133
+ break ;
134
+ }
135
+ }
136
+
137
+ }
138
+
139
+ reportCount++;
140
+ return reportCount;
141
+ }
142
+
143
+ int HID_::SetStringFeature (uint8_t id, const uint8_t * index, const char * data) {
144
+
145
+ int res = SetFeature (id, index , 1 );
146
+
147
+ if (res == 0 ) return 0 ;
148
+
149
+ res += SetFeature (0xFF00 | *index , data, strlen_P (data));
150
+
151
+ return res;
152
+ }
153
+
154
+ bool HID_::LockFeature (uint16_t id, bool lock) {
155
+ if (rootReport) {
156
+ HIDReport* current;
157
+ for (current = rootReport;current; current=current->next ) {
158
+ if (current->id == id) {
159
+ current->lock = lock;
160
+ return true ;
161
+ }
162
+ }
163
+ }
164
+ return false ;
165
+ }
166
+
167
+
168
+ int HID_::SendReport (uint16_t id, const void * data, int len)
169
+ {
170
+ auto ret = USB_Send (HID_TX, &id, 1 );
92
171
if (ret < 0 ) return ret;
93
- auto ret2 = USB_Send (pluggedEndpoint | TRANSFER_RELEASE, data, len);
172
+ auto ret2 = USB_Send (HID_TX | TRANSFER_RELEASE, data, len);
94
173
if (ret2 < 0 ) return ret2;
95
174
return ret + ret2;
96
175
}
97
176
98
- bool HID_::setup (USBSetup& setup )
177
+ HIDReport* HID_::GetFeature ( uint16_t id )
99
178
{
179
+ HIDReport* current;
180
+ int i=0 ;
181
+ for (current=rootReport; current && i<reportCount; current=current->next , i++) {
182
+ if (id == current->id ) {
183
+ return current;
184
+ }
185
+ }
186
+ return (HIDReport*) NULL ;
187
+ }
188
+
189
+ bool HID_::setup (USBSetup& setup)
190
+ {
100
191
if (pluggedInterface != setup.wIndex ) {
101
192
return false ;
102
193
}
103
194
104
195
uint8_t request = setup.bRequest ;
105
196
uint8_t requestType = setup.bmRequestType ;
106
-
197
+
107
198
if (requestType == REQUEST_DEVICETOHOST_CLASS_INTERFACE)
108
- {
199
+ {
109
200
if (request == HID_GET_REPORT) {
110
- // TODO: HID_GetReport();
201
+
202
+ if (setup.wValueH == HID_REPORT_TYPE_FEATURE)
203
+ {
204
+
205
+ HIDReport* current = GetFeature (setup.wValueL );
206
+ if (current){
207
+ if (USB_SendControl (0 , &(current->id ), 1 )>0 &&
208
+ USB_SendControl (0 , current->data , current->length )>0 )
209
+ return true ;
210
+ }
211
+
212
+ return false ;
213
+
214
+ }
111
215
return true ;
112
216
}
113
217
if (request == HID_GET_PROTOCOL) {
@@ -120,7 +224,7 @@ bool HID_::setup(USBSetup& setup)
120
224
}
121
225
122
226
if (requestType == REQUEST_HOSTTODEVICE_CLASS_INTERFACE)
123
- {
227
+ {
124
228
if (request == HID_SET_PROTOCOL) {
125
229
// The USB Host tells us if we are in boot or report mode.
126
230
// This only works with a real boot compatible device.
@@ -133,24 +237,33 @@ bool HID_::setup(USBSetup& setup)
133
237
}
134
238
if (request == HID_SET_REPORT)
135
239
{
136
- // uint8_t reportID = setup.wValueL;
137
- // uint16_t length = setup.wLength;
138
- // uint8_t data[length];
139
- // Make sure to not read more data than USB_EP_SIZE.
140
- // You can read multiple times through a loop.
141
- // The first byte (may!) contain the reportID on a multreport.
142
- // USB_RecvControl(data, length);
240
+ if (setup.wValueH == HID_REPORT_TYPE_FEATURE)
241
+ {
242
+
243
+ HIDReport* current = GetFeature (setup.wValueL );
244
+ if (!current) return false ;
245
+ if (setup.wLength != current->length + 1 ) return false ;
246
+ uint8_t * data = new uint8_t [setup.wLength ];
247
+ USB_RecvControl (data, setup.wLength );
248
+ if (*data != current->id ) return false ;
249
+ memcpy ((uint8_t *)current->data , data+1 , current->length );
250
+ delete[] data;
251
+ return true ;
252
+
253
+ }
254
+
143
255
}
144
256
}
145
257
146
258
return false ;
147
259
}
148
260
149
- HID_::HID_ (void ) : PluggableUSBModule(1 , 1 , epType),
261
+ HID_::HID_ (void ) : PluggableUSBModule(2 , 1 , epType),
150
262
rootNode(NULL ), descriptorSize(0 ),
151
263
protocol(HID_REPORT_PROTOCOL), idle(1 )
152
264
{
153
265
epType[0 ] = EP_TYPE_INTERRUPT_IN;
266
+ epType[1 ] = EP_TYPE_INTERRUPT_OUT;
154
267
PluggableUSB ().plug (this );
155
268
}
156
269
0 commit comments