rusEFI
The most advanced open source ECU
Loading...
Searching...
No Matches
usbcfg.cpp
Go to the documentation of this file.
1/*
2 ChibiOS - Copyright (C) 2006..2016 Giovanni Di Sirio
3 Modified for RusEFI
4
5 Licensed under the Apache License, Version 2.0 (the "License");
6 you may not use this file except in compliance with the License.
7 You may obtain a copy of the License at
8
9 http://www.apache.org/licenses/LICENSE-2.0
10
11 Unless required by applicable law or agreed to in writing, software
12 distributed under the License is distributed on an "AS IS" BASIS,
13 WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 See the License for the specific language governing permissions and
15 limitations under the License.
16*/
17
18#include "pch.h"
19#include "mmc_card.h"
20
21#if HAL_USE_SERIAL_USB
22
23#include "usbcfg.h"
24
25/* Virtual serial port over USB.*/
26SerialUSBDriver SDU1;
27
28/*
29 * Endpoints to be used for USBD1.
30 */
31#define USBD1_DATA_REQUEST_EP 2
32#define USBD1_DATA_AVAILABLE_EP 2
33#define USBD1_INTERRUPT_REQUEST_EP 3
34
35#if HAL_USE_USB_MSD
36 // Descriptor that includes MSD is larger and has an extra interface
37 #define DESCRIPTOR_SIZE 98
38 #define NUM_INTERFACES 3
39 #define USB_MSD_EP_SIZE 64
40
41 #define MSD_IF 0
42 #define CDC_INT_IF 1
43 #define CDC_DATA_IF 2
44#else
45 // Alternate descriptor is smaller, no MSD
46 #define DESCRIPTOR_SIZE 75
47 #define NUM_INTERFACES 2
48
49 // only two interfaces
50 #define CDC_INT_IF 0
51 #define CDC_DATA_IF 1
52#endif
53
54/*
55 * USB Device Descriptor.
56 */
57static const uint8_t vcom_device_descriptor_data[18] = {
58 USB_DESC_DEVICE (0x0110, /* bcdUSB (1.1). */
59 0xEF, /* bDeviceClass ( misc ). */
60 0x02, /* bDeviceSubClass. */
61 0x01, /* bDeviceProtocol. */
62 0x40, /* bMaxPacketSize. */
63 0x0483, /* idVendor (ST). */
64 0x5740, /* idProduct. */
65 0x0200, /* bcdDevice. */
66 1, /* iManufacturer. */
67 2, /* iProduct. */
68 3, /* iSerialNumber. */
69 1) /* bNumConfigurations. */
70};
71
72/*
73 * Device Descriptor wrapper.
74 */
75static const USBDescriptor vcom_device_descriptor = {
78};
79
80/* Configuration Descriptor tree for a CDC.*/
81static const uint8_t vcom_configuration_descriptor_data[DESCRIPTOR_SIZE] = {
82 /* Configuration Descriptor.*/
83 USB_DESC_CONFIGURATION(DESCRIPTOR_SIZE,/* wTotalLength. */
84 NUM_INTERFACES,/* bNumInterfaces. */
85 0x01, /* bConfigurationValue. */
86 0, /* iConfiguration. */
87 0x80, /* bmAttributes (bus powered). */
88 200), /* bMaxPower (400mA). */
89#if HAL_USE_USB_MSD
90 USB_DESC_INTERFACE (MSD_IF, /* bInterfaceNumber. */
91 0x00, /* bAlternateSetting. */
92 0x02, /* bNumEndpoints. */
93 0x08, /* bInterfaceClass (mass storage) */
94 0x06, /* bInterfaceSubClass (SCSI
95 transparent storage class). */
96 0x50, /* bInterfaceProtocol (Bulk Only). */
97 0), /* iInterface. */
98 /* Mass Storage Data In Endpoint Descriptor.*/
99 USB_DESC_ENDPOINT (USB_MSD_DATA_EP | 0x80,
100 0x02, /* bmAttributes (Bulk). */
101 64, /* wMaxPacketSize. */
102 0x00), /* bInterval. 1ms */
103 /* Mass Storage Data Out Endpoint Descriptor.*/
104 USB_DESC_ENDPOINT (USB_MSD_DATA_EP,
105 0x02, /* bmAttributes (Bulk). */
106 64, /* wMaxPacketSize. */
107 0x00), /* bInterval. 1ms */
108#endif // HAL_USE_USB_MSD
109 // CDC
110 /* IAD Descriptor - describes that EP2+3 belong to CDC */
111 USB_DESC_INTERFACE_ASSOCIATION(CDC_INT_IF, /* bFirstInterface. */
112 0x02, /* bInterfaceCount. */
113 0x02, /* bFunctionClass (CDC). */
114 0x02, /* bFunctionSubClass. (2) */
115 0x01, /* bFunctionProtocol (1) */
116 2), /* iInterface. */
117 /* Interface Descriptor.*/
118 USB_DESC_INTERFACE (CDC_INT_IF, /* bInterfaceNumber. */
119 0x00, /* bAlternateSetting. */
120 0x01, /* bNumEndpoints. */
121 0x02, /* bInterfaceClass (Communications
122 Interface Class, CDC section
123 4.2). */
124 0x02, /* bInterfaceSubClass (Abstract
125 Control Model, CDC section 4.3). */
126 0x01, /* bInterfaceProtocol (AT commands,
127 CDC section 4.4). */
128 0), /* iInterface. */
129 /* Header Functional Descriptor (CDC section 5.2.3).*/
130 USB_DESC_BYTE (5), /* bLength. */
131 USB_DESC_BYTE (0x24), /* bDescriptorType (CS_INTERFACE). */
132 USB_DESC_BYTE (0x00), /* bDescriptorSubtype (Header
133 Functional Descriptor. */
134 USB_DESC_BCD (0x0110), /* bcdCDC. */
135 /* Call Management Functional Descriptor. */
136 USB_DESC_BYTE (5), /* bFunctionLength. */
137 USB_DESC_BYTE (0x24), /* bDescriptorType (CS_INTERFACE). */
138 USB_DESC_BYTE (0x01), /* bDescriptorSubtype (Call Management
139 Functional Descriptor). */
140 USB_DESC_BYTE (0x00), /* bmCapabilities (D0+D1). */
141 USB_DESC_BYTE (CDC_DATA_IF), /* bDataInterface. */
142 /* ACM Functional Descriptor.*/
143 USB_DESC_BYTE (4), /* bFunctionLength. */
144 USB_DESC_BYTE (0x24), /* bDescriptorType (CS_INTERFACE). */
145 USB_DESC_BYTE (0x02), /* bDescriptorSubtype (Abstract
146 Control Management Descriptor). */
147 USB_DESC_BYTE (0x02), /* bmCapabilities. */
148 /* Union Functional Descriptor.*/
149 USB_DESC_BYTE (5), /* bFunctionLength. */
150 USB_DESC_BYTE (0x24), /* bDescriptorType (CS_INTERFACE). */
151 USB_DESC_BYTE (0x06), /* bDescriptorSubtype (Union
152 Functional Descriptor). */
153 USB_DESC_BYTE (CDC_INT_IF), /* bMasterInterface (Communication
154 Class Interface). */
155 USB_DESC_BYTE (CDC_DATA_IF), /* bSlaveInterface0 (Data Class
156 Interface). */
157 /* Endpoint 3 Descriptor.*/
158 USB_DESC_ENDPOINT (USBD1_INTERRUPT_REQUEST_EP|0x80,
159 0x03, /* bmAttributes (Interrupt). */
160 0x0008, /* wMaxPacketSize. */
161 0xFF), /* bInterval. */
162 /* Interface Descriptor.*/
163 USB_DESC_INTERFACE (CDC_DATA_IF, /* bInterfaceNumber. */
164 0x00, /* bAlternateSetting. */
165 0x02, /* bNumEndpoints. */
166 0x0A, /* bInterfaceClass (Data Class
167 Interface, CDC section 4.5). */
168 0x00, /* bInterfaceSubClass (CDC section
169 4.6). */
170 0x00, /* bInterfaceProtocol (CDC section
171 4.7). */
172 0x00), /* iInterface. */
173 /* Endpoint 2 Descriptor.*/
174 USB_DESC_ENDPOINT (USBD1_DATA_AVAILABLE_EP, /* bEndpointAddress.*/
175 0x02, /* bmAttributes (Bulk). */
176 0x0040, /* wMaxPacketSize. */
177 0x00), /* bInterval. */
178 /* Endpoint 2 Descriptor.*/
179 USB_DESC_ENDPOINT (USBD1_DATA_REQUEST_EP|0x80, /* bEndpointAddress.*/
180 0x02, /* bmAttributes (Bulk). */
181 0x0040, /* wMaxPacketSize. */
182 0x00) /* bInterval. */
183};
184
185/*
186 * Configuration Descriptor wrapper.
187 */
192
193/*
194 * U.S. English language identifier.
195 */
196static const uint8_t vcom_string0[] = {
197 USB_DESC_BYTE(4), /* bLength. */
198 USB_DESC_BYTE(USB_DESCRIPTOR_STRING), /* bDescriptorType. */
199 USB_DESC_WORD(0x0409) /* wLANGID (U.S. English). */
200};
201
202/*
203 * Vendor string.
204 */
205static const uint8_t vcom_string1[] = {
206 USB_DESC_BYTE(22), /* bLength. */
207 USB_DESC_BYTE(USB_DESCRIPTOR_STRING), /* bDescriptorType. */
208 'r', 0, 'u', 0, 's', 0, 'E', 0, 'F', 0, 'I', 0, ' ', 0, 'L', 0,
209 'L', 0, 'C', 0
210};
211
212// STRING_CONTENT length + 2?
213#ifndef USB_DESCRIPTOR_B_LENGTH
214#define USB_DESCRIPTOR_B_LENGTH (56 + 2)
215#endif
216
217#ifndef USB_DESCRIPTOR_STRING_CONTENT
218#define USB_DESCRIPTOR_STRING_CONTENT 'r', 0, 'u', 0, 's', 0, 'E', 0, 'F', 0, 'I', 0, ' ', 0, 'E', 0, \
219 'n', 0, 'g', 0, 'i', 0, 'n', 0, 'e', 0, ' ', 0, 'M', 0, 'a', 0, \
220 'n', 0, 'a', 0, 'g', 0, 'e', 0, 'm', 0, 'e', 0, 'n', 0, 't', 0, \
221 ' ', 0, 'E', 0, 'C', 0, 'U', 0
222#endif
223
224/*
225 * Device Description string.
226 */
227static const uint8_t vcom_string2[] = {
228 USB_DESC_BYTE(USB_DESCRIPTOR_B_LENGTH), /* bLength. */
229 USB_DESC_BYTE(USB_DESCRIPTOR_STRING), /* bDescriptorType. */
230 USB_DESCRIPTOR_STRING_CONTENT, 0
231};
232
233/*
234 * Serial Number string.
235 */
236static uint8_t vcom_string3[] = {
237 USB_DESC_BYTE(50), /* bLength. */
238 USB_DESC_BYTE(USB_DESCRIPTOR_STRING), /* bDescriptorType. */
239 '0', 0, '1', 0, '2', 0, '3', 0, '4', 0, '5', 0, '6', 0, '7', 0,
240 '8', 0, '9', 0, 'A', 0, 'B', 0, 'C', 0, 'D', 0, 'E', 0, 'F', 0,
241 '0', 0, '1', 0, '2', 0, '3', 0, '4', 0, '5', 0, '6', 0, '7', 0,
242 0
243};
244
245/*
246 * Strings wrappers array.
247 */
248static const USBDescriptor vcom_strings[] = {
249 {sizeof vcom_string0, vcom_string0},
250 {sizeof vcom_string1, vcom_string1},
251 {sizeof vcom_string2, vcom_string2},
252 {sizeof vcom_string3, vcom_string3}
253};
254
255static char nib2char(uint8_t nibble) {
256 if (nibble > 0x9) {
257 return nibble - 0xA + 'A';
258 } else {
259 return nibble + '0';
260 }
261}
262
263void usbPopulateSerialNumber(const uint8_t* serialNumber, size_t bytes) {
264 if (bytes > 12) {
265 bytes = 12;
266 }
267
268 // Skip the first two bytes (metadata)
269 uint8_t* dst = &vcom_string3[2];
270
271 for (size_t i = 0; i < bytes; i++) {
272 uint8_t byte = serialNumber[i];
273
274 uint8_t lowNibble = byte & 0xF;
275 uint8_t highNibble = byte >> 4;
276
277 // Descriptor strings are UCS16, so write every other byte
278 dst[4 * i] = nib2char(highNibble);
279 dst[4 * i + 2] = nib2char(lowNibble);
280 }
281}
282
283/*
284 * Handles the GET_DESCRIPTOR callback. All required descriptors must be
285 * handled here.
286 */
287static const USBDescriptor *get_descriptor(USBDriver *usbp,
288 uint8_t dtype,
289 uint8_t dindex,
290 uint16_t lang) {
291
292 (void)usbp;
293 (void)lang;
294 switch (dtype) {
295 case USB_DESCRIPTOR_DEVICE:
297 case USB_DESCRIPTOR_CONFIGURATION:
299 case USB_DESCRIPTOR_STRING:
300 if (dindex < 4)
301 return &vcom_strings[dindex];
302 }
303 return NULL;
304}
305
306#if HAL_USE_USB_MSD
307 /**
308 * @brief IN MSD state
309 */
310 static USBInEndpointState msdInstate;
311
312 /**
313 * @brief OUT MSD state
314 */
315 static USBOutEndpointState msdOutstate;
316
317 /**
318 * @brief MSD initialization structure (both IN and OUT).
319 */
320 static const USBEndpointConfig msdEpConfig = {
321 USB_EP_MODE_TYPE_BULK,
322 NULL,
323 NULL,
324 NULL,
325 USB_MSD_EP_SIZE,
326 USB_MSD_EP_SIZE,
327 &msdInstate,
329 4,
330 NULL
331 };
332#endif //HAL_USE_MSD
333
334// IN CDC data state.
335static USBInEndpointState cdcDataInstate;
336// OUT CDC data state.
337static USBOutEndpointState cdcDataOutstate;
338// CDC data initialization structure (both IN and OUT).
339static const USBEndpointConfig cdcDataEpConfig = {
340 USB_EP_MODE_TYPE_BULK,
341 NULL,
342 sduDataTransmitted,
343 sduDataReceived,
344 0x0040,
345 0x0040,
348 4,
349 NULL
350};
351
352// IN CDC interrupt state.
353static USBInEndpointState cdcInterruptInstate;
354// CDC interrupt initialization structure (IN only).
355static const USBEndpointConfig cdcInterruptEpConfig = {
356 USB_EP_MODE_TYPE_INTR,
357 NULL,
358 sduInterruptTransmitted,
359 NULL,
360 0x0010,
361 0x0000,
363 NULL,
364 1,
365 NULL
366};
367
368/*
369 * Handles the USB driver global events.
370 */
371static void usb_event(USBDriver *usbp, usbevent_t event) {
372 switch (event) {
373 case USB_EVENT_ADDRESS:
374 return;
375 case USB_EVENT_CONFIGURED:
376 chSysLockFromISR();
377
378 /* Enables the endpoints specified into the configuration.
379 Note, this callback is invoked from an ISR so I-Class functions
380 must be used.*/
381#if HAL_USE_USB_MSD
382 usbInitEndpointI(usbp, USB_MSD_DATA_EP, &msdEpConfig);
383#endif
384
385 usbInitEndpointI(usbp, USBD1_DATA_REQUEST_EP, &cdcDataEpConfig);
386 usbInitEndpointI(usbp, USBD1_INTERRUPT_REQUEST_EP, &cdcInterruptEpConfig);
387
388 /* Resetting the state of the CDC subsystem.*/
389 sduConfigureHookI(&SDU1);
390
391 #if HAL_USE_USB_MSD
392 // Tell the MMC thread to wake up and mount the card as a USB device
394 #endif
395
396 chSysUnlockFromISR();
397 return;
398 case USB_EVENT_RESET:
399 /* Falls into.*/
400 case USB_EVENT_UNCONFIGURED:
401 /* Falls into.*/
402 case USB_EVENT_SUSPEND:
403 chSysLockFromISR();
404
405 /* Disconnection event on suspend.*/
406 sduSuspendHookI(&SDU1);
407
408 chSysUnlockFromISR();
409 return;
410 case USB_EVENT_WAKEUP:
411 chSysLockFromISR();
412
413 /* Disconnection event on suspend.*/
414 sduWakeupHookI(&SDU1);
415
416 chSysUnlockFromISR();
417 return;
418 case USB_EVENT_STALLED:
419 return;
420 }
421 return;
422}
423
424/*
425 * Handles the USB driver global events.
426 */
427static void sof_handler(USBDriver *usbp) {
428
429 (void)usbp;
430
431 osalSysLockFromISR();
432 sduSOFHookI(&SDU1);
433 osalSysUnlockFromISR();
434}
435
436// We need a custom hook to handle both MSD and CDC at the same time
437static bool hybridRequestHook(USBDriver *usbp) {
438#if HAL_USE_USB_MSD
439 // Try the MSD driver first
440 if (msd_request_hook_new(usbp)) {
441 return true;
442 }
443#endif // HAL_USE_USB_MSD
444
445 // if not MSD, it must be serial
446 return sduRequestsHook(usbp);
447}
448
449/*
450 * USB driver configuration.
451 */
452const USBConfig usbcfg = {
453 usb_event,
457};
458
459/*
460 * Serial over USB driver configuration.
461 */
462const SerialUSBConfig serusbcfg = {
463#if STM32_USB_USE_OTG1
464 .usbp = &USBD1,
465#elif STM32_USB_USE_OTG2
466 .usbp = &USBD2,
467#else
468 #error Serial over USB needs OTG1 or OTG2 to be enabled
469#endif
470 .bulk_in = USBD1_DATA_REQUEST_EP,
471 .bulk_out = USBD1_DATA_AVAILABLE_EP,
472 .int_in = USBD1_INTERRUPT_REQUEST_EP
473};
474
475#endif /* EFI_USB_SERIAL */
bool msd_request_hook_new(USBDriver *usbp)
void onUsbConnectedNotifyMmcI()
Definition mmc_card.cpp:426
static void usb_event(USBDriver *usbp, usbevent_t event)
Definition usbcfg.cpp:371
static uint8_t vcom_string3[]
Definition usbcfg.cpp:236
static const uint8_t vcom_string0[]
Definition usbcfg.cpp:196
SerialUSBDriver SDU1
Definition usbcfg.cpp:26
static const uint8_t vcom_configuration_descriptor_data[DESCRIPTOR_SIZE]
Definition usbcfg.cpp:81
static const uint8_t vcom_device_descriptor_data[18]
Definition usbcfg.cpp:57
const USBConfig usbcfg
Definition usbcfg.cpp:452
static bool hybridRequestHook(USBDriver *usbp)
Definition usbcfg.cpp:437
static USBInEndpointState cdcInterruptInstate
Definition usbcfg.cpp:353
static const uint8_t vcom_string2[]
Definition usbcfg.cpp:227
static const USBEndpointConfig msdEpConfig
MSD initialization structure (both IN and OUT).
Definition usbcfg.cpp:320
static USBInEndpointState cdcDataInstate
Definition usbcfg.cpp:335
static const USBDescriptor * get_descriptor(USBDriver *usbp, uint8_t dtype, uint8_t dindex, uint16_t lang)
Definition usbcfg.cpp:287
static void sof_handler(USBDriver *usbp)
Definition usbcfg.cpp:427
static const USBEndpointConfig cdcDataEpConfig
Definition usbcfg.cpp:339
static const USBDescriptor vcom_strings[]
Definition usbcfg.cpp:248
static const USBEndpointConfig cdcInterruptEpConfig
Definition usbcfg.cpp:355
static const uint8_t vcom_string1[]
Definition usbcfg.cpp:205
static USBOutEndpointState msdOutstate
OUT MSD state.
Definition usbcfg.cpp:315
static USBOutEndpointState cdcDataOutstate
Definition usbcfg.cpp:337
static const USBDescriptor vcom_device_descriptor
Definition usbcfg.cpp:75
const SerialUSBConfig serusbcfg
Definition usbcfg.cpp:462
void usbPopulateSerialNumber(const uint8_t *serialNumber, size_t bytes)
Definition usbcfg.cpp:263
static char nib2char(uint8_t nibble)
Definition usbcfg.cpp:255
static USBInEndpointState msdInstate
IN MSD state.
Definition usbcfg.cpp:310
static const USBDescriptor vcom_configuration_descriptor
Definition usbcfg.cpp:188