1 /*
2 * Copyright (c) 2017 Intel Corporation
3 *
4 * SPDX-License-Identifier: Apache-2.0
5 */
6
7 #include <zephyr/logging/log.h>
8 LOG_MODULE_REGISTER(usb_rndis, CONFIG_USB_DEVICE_NETWORK_LOG_LEVEL);
9
10 /* Enable verbose debug printing extra hexdumps */
11 #define VERBOSE_DEBUG 0
12
13 #include <zephyr/init.h>
14
15 #include <zephyr/net/ethernet.h>
16 #include <net_private.h>
17
18 #include <zephyr/usb/usb_device.h>
19 #include <zephyr/usb/class/usb_cdc.h>
20 #include <os_desc.h>
21
22 #include "netusb.h"
23 #include "function_rndis.h"
24
25 /* RNDIS handling */
26 #define CFG_RNDIS_TX_BUF_COUNT 5
27 #define CFG_RNDIS_TX_BUF_SIZE 512
28 NET_BUF_POOL_DEFINE(rndis_tx_pool, CFG_RNDIS_TX_BUF_COUNT,
29 CFG_RNDIS_TX_BUF_SIZE, 0, NULL);
30 static struct k_fifo rndis_tx_queue;
31
32 /* Serialize RNDIS command queue for later processing */
33 #define CFG_RNDIS_CMD_BUF_COUNT 2
34 #define CFG_RNDIS_CMD_BUF_SIZE CONFIG_USB_REQUEST_BUFFER_SIZE
35 NET_BUF_POOL_DEFINE(rndis_cmd_pool, CFG_RNDIS_CMD_BUF_COUNT,
36 CFG_RNDIS_CMD_BUF_SIZE, 0, NULL);
37 static struct k_fifo rndis_cmd_queue;
38
39 /*
40 * Stack for cmd thread
41 */
42 static K_KERNEL_STACK_DEFINE(cmd_stack, 2048);
43 static struct k_thread cmd_thread_data;
44
45 struct usb_rndis_config {
46 struct usb_association_descriptor iad;
47 struct usb_if_descriptor if0;
48 struct usb_ep_descriptor if0_int_ep;
49
50 struct usb_if_descriptor if1;
51 struct usb_ep_descriptor if1_in_ep;
52 struct usb_ep_descriptor if1_out_ep;
53 } __packed;
54
55 USBD_CLASS_DESCR_DEFINE(primary, 0) struct usb_rndis_config rndis_cfg = {
56 .iad = {
57 .bLength = sizeof(struct usb_association_descriptor),
58 .bDescriptorType = USB_DESC_INTERFACE_ASSOC,
59 .bFirstInterface = 0,
60 .bInterfaceCount = 0x02,
61 .bFunctionClass = USB_BCC_MISCELLANEOUS,
62 .bFunctionSubClass = 4,
63 .bFunctionProtocol = 1,
64 .iFunction = 0,
65 },
66 /* Interface descriptor 0 */
67 .if0 = {
68 .bLength = sizeof(struct usb_if_descriptor),
69 .bDescriptorType = USB_DESC_INTERFACE,
70 .bInterfaceNumber = 0,
71 .bAlternateSetting = 0,
72 .bNumEndpoints = 1,
73 .bInterfaceClass = USB_BCC_MISCELLANEOUS,
74 .bInterfaceSubClass = 4,
75 .bInterfaceProtocol = 1,
76 .iInterface = 0,
77 },
78 /* Notification EP Descriptor */
79 .if0_int_ep = {
80 .bLength = sizeof(struct usb_ep_descriptor),
81 .bDescriptorType = USB_DESC_ENDPOINT,
82 .bEndpointAddress = RNDIS_INT_EP_ADDR,
83 .bmAttributes = USB_DC_EP_INTERRUPT,
84 .wMaxPacketSize =
85 sys_cpu_to_le16(CONFIG_RNDIS_INTERRUPT_EP_MPS),
86 .bInterval = 0x09,
87 },
88 /* Interface descriptor 1 */
89 .if1 = {
90 .bLength = sizeof(struct usb_if_descriptor),
91 .bDescriptorType = USB_DESC_INTERFACE,
92 .bInterfaceNumber = 1,
93 .bAlternateSetting = 0,
94 .bNumEndpoints = 2,
95 .bInterfaceClass = USB_BCC_CDC_DATA,
96 .bInterfaceSubClass = 0,
97 .bInterfaceProtocol = 0,
98 .iInterface = 0,
99 },
100 /* Data Endpoint IN */
101 .if1_in_ep = {
102 .bLength = sizeof(struct usb_ep_descriptor),
103 .bDescriptorType = USB_DESC_ENDPOINT,
104 .bEndpointAddress = RNDIS_IN_EP_ADDR,
105 .bmAttributes = USB_DC_EP_BULK,
106 .wMaxPacketSize =
107 sys_cpu_to_le16(CONFIG_RNDIS_BULK_EP_MPS),
108 .bInterval = 0x00,
109 },
110 /* Data Endpoint OUT */
111 .if1_out_ep = {
112 .bLength = sizeof(struct usb_ep_descriptor),
113 .bDescriptorType = USB_DESC_ENDPOINT,
114 .bEndpointAddress = RNDIS_OUT_EP_ADDR,
115 .bmAttributes = USB_DC_EP_BULK,
116 .wMaxPacketSize =
117 sys_cpu_to_le16(CONFIG_RNDIS_BULK_EP_MPS),
118 .bInterval = 0x00,
119 },
120 };
121
122 /*
123 * TLV structure is used for data encapsulation parsing
124 */
125 struct tlv {
126 uint32_t type;
127 uint32_t len;
128 uint8_t data[];
129 } __packed;
130
131 static struct __rndis {
132 uint32_t net_filter;
133
134 enum {
135 UNINITIALIZED,
136 INITIALIZED,
137 } state;
138
139 struct net_pkt *in_pkt; /* Pointer to pkt assembling at the moment */
140 int in_pkt_len; /* Packet length to be assembled */
141 int skip_bytes; /* In case of low memory, skip bytes */
142
143 uint16_t mtu;
144 uint16_t speed; /* TODO: Calculate right speed */
145
146 /* Statistics */
147 uint32_t rx_err;
148 uint32_t tx_err;
149 uint32_t rx_no_buf;
150
151 atomic_t notify_count;
152
153 uint8_t mac[6];
154 uint8_t media_status;
155 } rndis = {
156 .mac = { 0x00, 0x00, 0x5E, 0x00, 0x53, 0x01 },
157 .mtu = NET_ETH_MTU, /* Ethernet frame */
158 .media_status = RNDIS_OBJECT_ID_MEDIA_DISCONNECTED,
159 .state = UNINITIALIZED,
160 .skip_bytes = 0,
161 .speed = 0,
162 };
163
164 static uint8_t manufacturer[] = CONFIG_USB_DEVICE_MANUFACTURER;
165 static uint32_t drv_version = 1U;
166
167 /**
168 * Assumes MaxPacketsPerTransfer of 1 and 802.2 (ethernet) medium.
169 */
170 #define RNDIS_BUF_SIZE (NET_ETH_MAX_FRAME_SIZE + sizeof(struct rndis_payload_packet))
171
172 static uint8_t tx_buf[RNDIS_BUF_SIZE];
173
174 /**
175 * TODO: package reception can be optimized to avoid rx_buf usage.
176 */
177 static uint8_t rx_buf[RNDIS_BUF_SIZE];
178
179 static uint32_t object_id_supported[] = {
180 RNDIS_OBJECT_ID_GEN_SUPP_LIST,
181 RNDIS_OBJECT_ID_GEN_HW_STATUS,
182 RNDIS_OBJECT_ID_GEN_SUPP_MEDIA,
183 RNDIS_OBJECT_ID_GEN_IN_USE_MEDIA,
184
185 RNDIS_OBJECT_ID_GEN_MAX_FRAME_SIZE,
186 RNDIS_OBJECT_ID_GEN_LINK_SPEED,
187 RNDIS_OBJECT_ID_GEN_BLOCK_TX_SIZE,
188 RNDIS_OBJECT_ID_GEN_BLOCK_RX_SIZE,
189
190 RNDIS_OBJECT_ID_GEN_VENDOR_ID,
191 RNDIS_OBJECT_ID_GEN_VENDOR_DESC,
192 RNDIS_OBJECT_ID_GEN_VENDOR_DRV_VER,
193
194 RNDIS_OBJECT_ID_GEN_PKT_FILTER,
195 RNDIS_OBJECT_ID_GEN_MAX_TOTAL_SIZE,
196 RNDIS_OBJECT_ID_GEN_CONN_MEDIA_STATUS,
197 RNDIS_OBJECT_ID_GEN_PHYSICAL_MEDIUM,
198 #if defined(USE_RNDIS_STATISTICS)
199 /* Using RNDIS statistics puts heavy load on
200 * USB bus, disable it for now
201 */
202 RNDIS_OBJECT_ID_GEN_TRANSMIT_OK,
203 RNDIS_OBJECT_ID_GEN_RECEIVE_OK,
204 RNDIS_OBJECT_ID_GEN_TRANSMIT_ERROR,
205 RNDIS_OBJECT_ID_GEN_RECEIVE_ERROR,
206 RNDIS_OBJECT_ID_GEN_RECEIVE_NO_BUF,
207 #endif /* USE_RNDIS_STATISTICS */
208 RNDIS_OBJECT_ID_802_3_PERMANENT_ADDRESS,
209 RNDIS_OBJECT_ID_802_3_CURR_ADDRESS,
210 RNDIS_OBJECT_ID_802_3_MCAST_LIST,
211 RNDIS_OBJECT_ID_802_3_MAX_LIST_SIZE,
212 RNDIS_OBJECT_ID_802_3_MAC_OPTIONS,
213 };
214
215 #define RNDIS_INT_EP_IDX 0
216 #define RNDIS_OUT_EP_IDX 1
217 #define RNDIS_IN_EP_IDX 2
218
219 static void rndis_bulk_out(uint8_t ep, enum usb_dc_ep_cb_status_code ep_status);
220
221 static struct usb_ep_cfg_data rndis_ep_data[] = {
222 {
223 .ep_cb = usb_transfer_ep_callback,
224 .ep_addr = RNDIS_INT_EP_ADDR
225 },
226 {
227 .ep_cb = rndis_bulk_out,
228 .ep_addr = RNDIS_OUT_EP_ADDR
229 },
230 {
231 .ep_cb = usb_transfer_ep_callback,
232 .ep_addr = RNDIS_IN_EP_ADDR
233 },
234 };
235
parse_rndis_header(const uint8_t * buffer,uint32_t buf_len)236 static int parse_rndis_header(const uint8_t *buffer, uint32_t buf_len)
237 {
238 struct rndis_payload_packet *hdr = (void *)buffer;
239 uint32_t len;
240
241 if (buf_len < sizeof(*hdr)) {
242 LOG_ERR("Too small packet len %u", buf_len);
243 return -EINVAL;
244 }
245
246 if (hdr->type != sys_cpu_to_le32(RNDIS_DATA_PACKET)) {
247 LOG_ERR("Wrong data packet type 0x%x",
248 sys_le32_to_cpu(hdr->type));
249 return -EINVAL;
250 }
251
252 len = sys_le32_to_cpu(hdr->len);
253 /*
254 * Calculate additional offset since payload_offset is calculated
255 * from the start of itself ;)
256 */
257 if (len < sys_le32_to_cpu(hdr->payload_offset) +
258 sys_le32_to_cpu(hdr->payload_len) +
259 offsetof(struct rndis_payload_packet, payload_offset)) {
260 LOG_ERR("Incorrect RNDIS packet");
261 return -EINVAL;
262 }
263
264 LOG_DBG("Parsing packet: len %u payload offset %u payload len %u",
265 len, sys_le32_to_cpu(hdr->payload_offset),
266 sys_le32_to_cpu(hdr->payload_len));
267
268 return len;
269 }
270
rndis_clean(void)271 void rndis_clean(void)
272 {
273 LOG_DBG("");
274
275 if (rndis.in_pkt) {
276 net_pkt_unref(rndis.in_pkt);
277
278 rndis.in_pkt = NULL;
279 rndis.in_pkt_len = 0;
280 }
281
282 rndis.skip_bytes = 0;
283 }
284
rndis_bulk_out(uint8_t ep,enum usb_dc_ep_cb_status_code ep_status)285 static void rndis_bulk_out(uint8_t ep, enum usb_dc_ep_cb_status_code ep_status)
286 {
287 uint32_t hdr_offset = 0U;
288 uint32_t len, read;
289
290 usb_read(ep, NULL, 0, &len);
291
292 LOG_DBG("EP 0x%x status %d len %u", ep, ep_status, len);
293
294 if (len > sizeof(rx_buf)) {
295 LOG_WRN("Trying to receive too much data, drop");
296 rndis_clean();
297 return;
298 }
299
300 usb_read(ep, rx_buf, len, &read);
301 if (len != read) {
302 LOG_ERR("Read %u instead of expected %u, skip the rest",
303 read, len);
304 rndis.skip_bytes = len - read;
305 return;
306 }
307
308 /* We already use frame keeping with len, warn here about
309 * receiving frame delimiter
310 */
311 if (len == 1U && !rx_buf[0]) {
312 LOG_DBG("Got frame delimiter, skip");
313 return;
314 }
315
316 /* Handle skip bytes */
317 if (rndis.skip_bytes) {
318 LOG_WRN("Skip %u bytes out of remaining %d bytes",
319 len, rndis.skip_bytes);
320
321 rndis.skip_bytes -= len;
322
323 if (rndis.skip_bytes < 0) {
324 LOG_ERR("Error skipping bytes");
325
326 rndis.skip_bytes = 0;
327 }
328
329 return;
330 }
331
332 /* Start new packet */
333 if (!rndis.in_pkt) {
334 struct net_pkt *pkt;
335
336 /* Append data only, skipping RNDIS header */
337 hdr_offset = sizeof(struct rndis_payload_packet);
338
339 rndis.in_pkt_len = parse_rndis_header(rx_buf, len);
340 if (rndis.in_pkt_len < 0) {
341 LOG_ERR("Error parsing RNDIS header");
342
343 rndis.rx_err++;
344 return;
345 }
346
347 pkt = net_pkt_rx_alloc_with_buffer(netusb_net_iface(),
348 rndis.in_pkt_len, AF_UNSPEC,
349 0, K_NO_WAIT);
350 if (!pkt) {
351 /* In case of low memory: skip the whole packet
352 * hoping to get buffers for later ones
353 */
354 rndis.skip_bytes = rndis.in_pkt_len - len;
355 rndis.rx_no_buf++;
356
357 LOG_ERR("Not enough pkt buffers, len %u, skip %u",
358 rndis.in_pkt_len, rndis.skip_bytes);
359
360 return;
361 }
362
363 rndis.in_pkt = pkt;
364 }
365
366 if (net_pkt_write(rndis.in_pkt,
367 rx_buf + hdr_offset, len - hdr_offset)) {
368 LOG_ERR("Error writing data to pkt: %p", rndis.in_pkt);
369 rndis_clean();
370 rndis.rx_err++;
371 return;
372 }
373
374 LOG_DBG("To assemble %d bytes, reading %u bytes",
375 rndis.in_pkt_len, len);
376
377 rndis.in_pkt_len -= len;
378 if (!rndis.in_pkt_len) {
379 LOG_DBG("Assembled full RNDIS packet");
380
381 if (VERBOSE_DEBUG) {
382 net_pkt_hexdump(rndis.in_pkt, ">");
383 }
384
385 /* Queue data to iface */
386 netusb_recv(rndis.in_pkt);
387
388 /* Start over for new packets */
389 rndis.in_pkt = NULL;
390 } else if (rndis.in_pkt_len < 0) {
391 LOG_ERR("Error assembling packet, drop and start over");
392 rndis_clean();
393 }
394 }
395
rndis_notify_cb(uint8_t ep,int size,void * priv)396 static void rndis_notify_cb(uint8_t ep, int size, void *priv)
397 {
398 LOG_DBG("ep %x size %u", ep, size);
399
400
401 atomic_dec(&rndis.notify_count);
402 }
403
rndis_queue_rsp(struct net_buf * rsp)404 static void rndis_queue_rsp(struct net_buf *rsp)
405 {
406 if (!k_fifo_is_empty(&rndis_tx_queue)) {
407 LOG_WRN("Transmit response queue is not empty");
408 }
409
410 LOG_DBG("Queued response pkt %p", rsp);
411
412 k_fifo_put(&rndis_tx_queue, rsp);
413 }
414
415 /* Notify host about available data */
rndis_notify_rsp(void)416 static void rndis_notify_rsp(void)
417 {
418 static uint32_t buf[2] = {
419 sys_cpu_to_le32(0x01),
420 sys_cpu_to_le32(0x00)
421 };
422 int ret;
423
424 LOG_DBG("count %lu", atomic_get(&rndis.notify_count));
425
426 if (atomic_get(&rndis.notify_count)) {
427 LOG_WRN("Notification is already sent");
428 return;
429 }
430
431 atomic_inc(&rndis.notify_count);
432
433 ret = usb_transfer(rndis_ep_data[RNDIS_INT_EP_IDX].ep_addr,
434 (uint8_t *)buf, sizeof(buf),
435 USB_TRANS_WRITE | USB_TRANS_NO_ZLP,
436 rndis_notify_cb, NULL);
437 if (ret < 0) {
438 LOG_ERR("Transfer failure, ret %d", ret);
439 }
440 }
441
rndis_init_handle(uint8_t * data,uint32_t len)442 static int rndis_init_handle(uint8_t *data, uint32_t len)
443 {
444 struct rndis_init_cmd *cmd = (void *)data;
445 struct rndis_init_cmd_complete *rsp;
446 struct net_buf *buf;
447
448 LOG_DBG("req_id 0x%x", cmd->req_id);
449
450 buf = net_buf_alloc(&rndis_tx_pool, K_NO_WAIT);
451 if (!buf) {
452 LOG_ERR("Cannot get free buffer");
453 return -ENOMEM;
454 }
455
456 rsp = net_buf_add(buf, sizeof(*rsp));
457 rsp->status = sys_cpu_to_le32(RNDIS_CMD_STATUS_SUCCESS);
458 rsp->type = sys_cpu_to_le32(RNDIS_CMD_INITIALIZE_COMPLETE);
459 rsp->len = sys_cpu_to_le32(sizeof(*rsp));
460 rsp->req_id = cmd->req_id;
461
462 rsp->major_ver = sys_cpu_to_le32(RNDIS_MAJOR_VERSION);
463 rsp->minor_ver = sys_cpu_to_le32(RNDIS_MINOR_VERSION);
464
465 rsp->flags = sys_cpu_to_le32(RNDIS_FLAG_CONNECTIONLESS);
466 rsp->medium = sys_cpu_to_le32(RNDIS_MEDIUM_WIRED_ETHERNET);
467 rsp->max_packets = sys_cpu_to_le32(1);
468 rsp->max_transfer_size = sys_cpu_to_le32(RNDIS_BUF_SIZE);
469
470 rsp->pkt_align_factor = sys_cpu_to_le32(0);
471 (void)memset(rsp->__reserved, 0, sizeof(rsp->__reserved));
472
473 rndis.state = INITIALIZED;
474
475 rndis_queue_rsp(buf);
476
477 /* Notify about ready reply */
478 rndis_notify_rsp();
479
480 return 0;
481 }
482
rndis_halt_handle(void)483 static int rndis_halt_handle(void)
484 {
485 LOG_DBG("");
486
487 rndis.state = UNINITIALIZED;
488
489 /* TODO: Stop networking */
490
491 return 0;
492 }
493
rndis_query_add_supp_list(struct net_buf * buf)494 static uint32_t rndis_query_add_supp_list(struct net_buf *buf)
495 {
496 for (int i = 0; i < ARRAY_SIZE(object_id_supported); i++) {
497 net_buf_add_le32(buf, object_id_supported[i]);
498 }
499
500 return sizeof(object_id_supported);
501 }
502
rndis_query_handle(uint8_t * data,uint32_t len)503 static int rndis_query_handle(uint8_t *data, uint32_t len)
504 {
505 struct rndis_query_cmd *cmd = (void *)data;
506 struct rndis_query_cmd_complete *rsp;
507 struct net_buf *buf;
508 uint32_t object_id, buf_len = 0U;
509
510 buf = net_buf_alloc(&rndis_tx_pool, K_NO_WAIT);
511 if (!buf) {
512 LOG_ERR("Cannot get free buffer");
513 return -ENOMEM;
514 }
515
516 object_id = sys_le32_to_cpu(cmd->object_id);
517
518 LOG_DBG("req_id 0x%x Object ID 0x%x buf_len %u buf_offset %u",
519 sys_le32_to_cpu(cmd->req_id),
520 object_id,
521 sys_le32_to_cpu(cmd->buf_len),
522 sys_le32_to_cpu(cmd->buf_offset));
523
524 rsp = net_buf_add(buf, sizeof(*rsp));
525 rsp->type = sys_cpu_to_le32(RNDIS_CMD_QUERY_COMPLETE);
526 rsp->req_id = cmd->req_id;
527
528 /* offset is from the beginning of the req_id field */
529 rsp->buf_offset = sys_cpu_to_le32(16);
530
531 switch (object_id) {
532 case RNDIS_OBJECT_ID_GEN_SUPP_LIST:
533 LOG_DBG("RNDIS_OBJECT_ID_GEN_SUPP_LIST");
534 rndis_query_add_supp_list(buf);
535 break;
536 case RNDIS_OBJECT_ID_GEN_PHYSICAL_MEDIUM:
537 LOG_DBG("RNDIS_OBJECT_ID_GEN_PHYSICAL_MEDIUM");
538 net_buf_add_le32(buf, RNDIS_PHYSICAL_MEDIUM_TYPE_UNSPECIFIED);
539 break;
540 case RNDIS_OBJECT_ID_GEN_MAX_FRAME_SIZE:
541 LOG_DBG("RNDIS_OBJECT_ID_GEN_MAX_FRAME_SIZE");
542 net_buf_add_le32(buf, rndis.mtu);
543 break;
544 case RNDIS_OBJECT_ID_GEN_LINK_SPEED:
545 LOG_DBG("RNDIS_OBJECT_ID_GEN_LINK_SPEED");
546 if (rndis.media_status == RNDIS_OBJECT_ID_MEDIA_DISCONNECTED) {
547 net_buf_add_le32(buf, 0);
548 } else {
549 net_buf_add_le32(buf, rndis.speed);
550 }
551 break;
552 case RNDIS_OBJECT_ID_GEN_CONN_MEDIA_STATUS:
553 LOG_DBG("RNDIS_OBJECT_ID_GEN_CONN_MEDIA_STATUS");
554 net_buf_add_le32(buf, rndis.media_status);
555 break;
556 case RNDIS_OBJECT_ID_GEN_MAX_TOTAL_SIZE:
557 LOG_DBG("RNDIS_OBJECT_ID_GEN_MAX_TOTAL_SIZE");
558 net_buf_add_le32(buf, RNDIS_GEN_MAX_TOTAL_SIZE);
559 break;
560
561 /* Statistics stuff */
562 case RNDIS_OBJECT_ID_GEN_TRANSMIT_ERROR:
563 LOG_DBG("RNDIS_OBJECT_ID_GEN_TRANSMIT_ERROR: %u", rndis.tx_err);
564 net_buf_add_le32(buf, rndis.tx_err);
565 break;
566 case RNDIS_OBJECT_ID_GEN_RECEIVE_ERROR:
567 LOG_DBG("RNDIS_OBJECT_ID_GEN_RECEIVE_ERROR: %u", rndis.rx_err);
568 net_buf_add_le32(buf, rndis.rx_err);
569 break;
570 case RNDIS_OBJECT_ID_GEN_RECEIVE_NO_BUF:
571 LOG_DBG("RNDIS_OBJECT_ID_GEN_RECEIVE_NO_BUF: %u",
572 rndis.rx_no_buf);
573 net_buf_add_le32(buf, rndis.rx_no_buf);
574 break;
575
576 /* IEEE 802.3 */
577 case RNDIS_OBJECT_ID_802_3_PERMANENT_ADDRESS:
578 LOG_DBG("RNDIS_OBJECT_ID_802_3_PERMANENT_ADDRESS");
579 memcpy(net_buf_add(buf, sizeof(rndis.mac)), rndis.mac,
580 sizeof(rndis.mac));
581 break;
582 case RNDIS_OBJECT_ID_802_3_CURR_ADDRESS:
583 LOG_DBG("RNDIS_OBJECT_ID_802_3_CURR_ADDRESS");
584 memcpy(net_buf_add(buf, sizeof(rndis.mac)), rndis.mac,
585 sizeof(rndis.mac));
586 break;
587 case RNDIS_OBJECT_ID_802_3_MCAST_LIST:
588 LOG_DBG("RNDIS_OBJECT_ID_802_3_MCAST_LIST");
589 net_buf_add_le32(buf, 0xE0000000); /* 224.0.0.0 */
590 break;
591 case RNDIS_OBJECT_ID_802_3_MAX_LIST_SIZE:
592 LOG_DBG("RNDIS_OBJECT_ID_802_3_MAX_LIST_SIZE");
593 net_buf_add_le32(buf, 1); /* one address */
594 break;
595
596 /* Vendor information */
597 case RNDIS_OBJECT_ID_GEN_VENDOR_ID:
598 LOG_DBG("RNDIS_OBJECT_ID_GEN_VENDOR_ID");
599 net_buf_add_le32(buf, CONFIG_USB_DEVICE_VID);
600 break;
601 case RNDIS_OBJECT_ID_GEN_VENDOR_DESC:
602 LOG_DBG("RNDIS_OBJECT_ID_GEN_VENDOR_DESC");
603 memcpy(net_buf_add(buf, sizeof(manufacturer) - 1), manufacturer,
604 sizeof(manufacturer) - 1);
605 break;
606 case RNDIS_OBJECT_ID_GEN_VENDOR_DRV_VER:
607 LOG_DBG("RNDIS_OBJECT_ID_GEN_VENDOR_DRV_VER");
608 net_buf_add_le32(buf, drv_version);
609 break;
610 default:
611 LOG_WRN("Unhandled query for Object ID 0x%x", object_id);
612 break;
613 }
614
615 buf_len = buf->len - sizeof(*rsp);
616
617 if (buf_len) {
618 rsp->status = sys_cpu_to_le32(RNDIS_CMD_STATUS_SUCCESS);
619 } else {
620 rsp->status = sys_cpu_to_le32(RNDIS_CMD_STATUS_NOT_SUPP);
621 }
622
623 /* Can be zero if object_id not handled / found */
624 rsp->buf_len = sys_cpu_to_le32(buf_len);
625
626 rsp->len = sys_cpu_to_le32(buf_len + sizeof(*rsp));
627
628 LOG_DBG("buf_len %u rsp->len %u buf->len %u",
629 buf_len, rsp->len, buf->len);
630
631 rndis_queue_rsp(buf);
632
633 /* Notify about ready reply */
634 rndis_notify_rsp();
635
636 return 0;
637 }
638
rndis_set_handle(uint8_t * data,uint32_t len)639 static int rndis_set_handle(uint8_t *data, uint32_t len)
640 {
641 struct rndis_set_cmd *cmd = (void *)data;
642 struct rndis_set_cmd_complete *rsp;
643 struct net_buf *buf;
644 uint32_t object_id;
645 uint8_t *param;
646
647 if (len < sizeof(*cmd)) {
648 LOG_ERR("Packet is shorter then header");
649 return -EINVAL;
650 }
651
652 /* Parameter starts at offset buf_offset of the req_id field ;) */
653 param = (uint8_t *)&cmd->req_id + sys_le32_to_cpu(cmd->buf_offset);
654
655 if (len - ((uintptr_t)param - (uintptr_t)cmd) !=
656 sys_le32_to_cpu(cmd->buf_len)) {
657 LOG_ERR("Packet parsing error");
658 return -EINVAL;
659 }
660
661 buf = net_buf_alloc(&rndis_tx_pool, K_NO_WAIT);
662 if (!buf) {
663 LOG_ERR("Cannot get free buffer");
664 return -ENOMEM;
665 }
666
667 object_id = sys_le32_to_cpu(cmd->object_id);
668
669 LOG_DBG("req_id 0x%x Object ID 0x%x buf_len %u buf_offset %u",
670 sys_le32_to_cpu(cmd->req_id), object_id,
671 sys_le32_to_cpu(cmd->buf_len),
672 sys_le32_to_cpu(cmd->buf_offset));
673
674 rsp = net_buf_add(buf, sizeof(*rsp));
675 rsp->type = sys_cpu_to_le32(RNDIS_CMD_SET_COMPLETE);
676 rsp->len = sys_cpu_to_le32(sizeof(*rsp));
677 rsp->req_id = cmd->req_id; /* same endianness */
678
679 switch (object_id) {
680 case RNDIS_OBJECT_ID_GEN_PKT_FILTER:
681 if (sys_le32_to_cpu(cmd->buf_len) < sizeof(rndis.net_filter)) {
682 LOG_ERR("Packet is too small");
683 rsp->status = RNDIS_CMD_STATUS_INVALID_DATA;
684 break;
685 }
686
687 rndis.net_filter = sys_get_le32(param);
688 LOG_DBG("RNDIS_OBJECT_ID_GEN_PKT_FILTER 0x%x",
689 rndis.net_filter);
690 /* TODO: Start / Stop networking here */
691 rsp->status = sys_cpu_to_le32(RNDIS_CMD_STATUS_SUCCESS);
692 break;
693 case RNDIS_OBJECT_ID_802_3_MCAST_LIST:
694 LOG_DBG("RNDIS_OBJECT_ID_802_3_MCAST_LIST");
695 /* ignore for now */
696 rsp->status = sys_cpu_to_le32(RNDIS_CMD_STATUS_SUCCESS);
697 break;
698 default:
699 LOG_ERR("Unhandled object_id 0x%x", object_id);
700 rsp->status = sys_cpu_to_le32(RNDIS_CMD_STATUS_NOT_SUPP);
701 break;
702 }
703
704 rndis_queue_rsp(buf);
705
706 /* Notify about ready reply */
707 rndis_notify_rsp();
708
709 return 0;
710 }
711
rndis_reset_handle(uint8_t * data,uint32_t len)712 static int rndis_reset_handle(uint8_t *data, uint32_t len)
713 {
714 struct rndis_reset_cmd_complete *rsp;
715 struct net_buf *buf;
716
717 buf = net_buf_alloc(&rndis_tx_pool, K_NO_WAIT);
718 if (!buf) {
719 LOG_ERR("Cannot get free buffer");
720 return -ENOMEM;
721 }
722
723 LOG_DBG("");
724
725 rsp = net_buf_add(buf, sizeof(*rsp));
726 rsp->type = sys_cpu_to_le32(RNDIS_CMD_RESET_COMPLETE);
727 rsp->len = sys_cpu_to_le32(sizeof(*rsp));
728 rsp->status = sys_cpu_to_le32(RNDIS_CMD_STATUS_SUCCESS);
729 rsp->addr_reset = sys_cpu_to_le32(1);
730
731 rndis_queue_rsp(buf);
732
733 /* Notify about ready reply */
734 rndis_notify_rsp();
735
736 return 0;
737 }
738
rndis_keepalive_handle(uint8_t * data,uint32_t len)739 static int rndis_keepalive_handle(uint8_t *data, uint32_t len)
740 {
741 struct rndis_keepalive_cmd *cmd = (void *)data;
742 struct rndis_keepalive_cmd_complete *rsp;
743 struct net_buf *buf;
744
745 buf = net_buf_alloc(&rndis_tx_pool, K_NO_WAIT);
746 if (!buf) {
747 LOG_ERR("Cannot get free buffer");
748 return -ENOMEM;
749 }
750
751 LOG_DBG("");
752
753 rsp = net_buf_add(buf, sizeof(*rsp));
754 rsp->type = sys_cpu_to_le32(RNDIS_CMD_KEEPALIVE_COMPLETE);
755 rsp->len = sys_cpu_to_le32(sizeof(*rsp));
756 rsp->req_id = cmd->req_id; /* same order */
757 rsp->status = sys_cpu_to_le32(RNDIS_CMD_STATUS_SUCCESS);
758
759 rndis_queue_rsp(buf);
760
761 /* Notify about ready reply */
762 rndis_notify_rsp();
763
764 return 0;
765 }
766
queue_encapsulated_cmd(uint8_t * data,uint32_t len)767 static int queue_encapsulated_cmd(uint8_t *data, uint32_t len)
768 {
769 struct net_buf *buf;
770
771 buf = net_buf_alloc(&rndis_cmd_pool, K_NO_WAIT);
772 if (!buf) {
773 LOG_ERR("Cannot get free buffer");
774 return -ENOMEM;
775 }
776
777 memcpy(net_buf_add(buf, len), data, len);
778
779 k_fifo_put(&rndis_cmd_queue, buf);
780
781 LOG_DBG("queued buf %p", buf);
782
783 return 0;
784 }
785
handle_encapsulated_cmd(uint8_t * data,uint32_t len)786 static int handle_encapsulated_cmd(uint8_t *data, uint32_t len)
787 {
788 struct tlv *msg = (void *)data;
789
790 if (VERBOSE_DEBUG) {
791 net_hexdump("CMD >", data, len);
792 }
793
794 if (len != msg->len) {
795 LOG_WRN("Total len is different then command len %u %u",
796 len, msg->len);
797 /* TODO: need actions? */
798 }
799
800 LOG_DBG("RNDIS type 0x%x len %u total len %u",
801 msg->type, msg->len, len);
802
803 switch (msg->type) {
804 case RNDIS_CMD_INITIALIZE:
805 return rndis_init_handle(data, len);
806 case RNDIS_CMD_HALT:
807 return rndis_halt_handle();
808 case RNDIS_CMD_QUERY:
809 return rndis_query_handle(data, len);
810 case RNDIS_CMD_SET:
811 return rndis_set_handle(data, len);
812 case RNDIS_CMD_RESET:
813 return rndis_reset_handle(data, len);
814 case RNDIS_CMD_KEEPALIVE:
815 return rndis_keepalive_handle(data, len);
816 default:
817 LOG_ERR("Message 0x%x unhandled", msg->type);
818 return -ENOTSUP;
819 }
820
821 return 0;
822 }
823
handle_encapsulated_rsp(uint8_t ** data,uint32_t * len)824 static int handle_encapsulated_rsp(uint8_t **data, uint32_t *len)
825 {
826 struct net_buf *buf;
827
828 LOG_DBG("");
829
830 buf = k_fifo_get(&rndis_tx_queue, K_NO_WAIT);
831 if (!buf) {
832 LOG_ERR("Error getting response buffer");
833 *len = 0U;
834 return -ENODATA;
835 }
836
837 *len = buf->len;
838 if (*len > CONFIG_USB_REQUEST_BUFFER_SIZE) {
839 LOG_ERR("Response too long %u, truncating to %u", buf->len,
840 CONFIG_USB_REQUEST_BUFFER_SIZE);
841 *len = CONFIG_USB_REQUEST_BUFFER_SIZE;
842 }
843
844 if (VERBOSE_DEBUG) {
845 net_hexdump("RSP <", buf->data, buf->len);
846 }
847
848 memcpy(*data, buf->data, *len);
849
850 net_buf_unref(buf);
851
852 return 0;
853 }
854
rndis_class_handler(struct usb_setup_packet * setup,int32_t * len,uint8_t ** data)855 static int rndis_class_handler(struct usb_setup_packet *setup, int32_t *len,
856 uint8_t **data)
857 {
858 LOG_DBG("len %d req_type 0x%x req 0x%x enabled %u",
859 *len, setup->bmRequestType, setup->bRequest,
860 netusb_enabled());
861
862 if (!netusb_enabled()) {
863 LOG_ERR("interface disabled");
864 return -ENODEV;
865 }
866
867 if (usb_reqtype_is_to_device(setup)) {
868 if (setup->bRequest == CDC_SEND_ENC_CMD) {
869 /*
870 * Instead of handling here, queue
871 * handle_encapsulated_cmd(*data, *len);
872 */
873 return queue_encapsulated_cmd(*data, *len);
874 }
875 } else {
876 if (setup->bRequest == CDC_GET_ENC_RSP) {
877 return handle_encapsulated_rsp(data, len);
878 }
879 }
880
881 LOG_WRN("Unknown USB packet req 0x%x type 0x%x",
882 setup->bRequest, setup->bmRequestType);
883 return -ENOTSUP;
884 }
885
cmd_thread(void * p1,void * p2,void * p3)886 static void cmd_thread(void *p1, void *p2, void *p3)
887 {
888 ARG_UNUSED(p1);
889 ARG_UNUSED(p2);
890 ARG_UNUSED(p3);
891
892 LOG_INF("Command thread started");
893
894 while (true) {
895 struct net_buf *buf;
896
897 buf = k_fifo_get(&rndis_cmd_queue, K_FOREVER);
898
899 LOG_DBG("got buf %p", buf);
900
901 handle_encapsulated_cmd(buf->data, buf->len);
902
903 net_buf_unref(buf);
904
905 k_yield();
906 }
907 }
908
909 /*
910 * RNDIS Send functions
911 */
912
rndis_hdr_add(uint8_t * buf,uint32_t len)913 static void rndis_hdr_add(uint8_t *buf, uint32_t len)
914 {
915 struct rndis_payload_packet *hdr = (void *)buf;
916 uint32_t offset = offsetof(struct rndis_payload_packet, payload_offset);
917
918 (void)memset(hdr, 0, sizeof(*hdr));
919
920 hdr->type = sys_cpu_to_le32(RNDIS_DATA_PACKET);
921 hdr->len = sys_cpu_to_le32(len + sizeof(*hdr));
922 hdr->payload_offset = sys_cpu_to_le32(sizeof(*hdr) - offset);
923 hdr->payload_len = sys_cpu_to_le32(len);
924
925 LOG_DBG("type %u len %u payload offset %u payload len %u",
926 hdr->type, hdr->len, hdr->payload_offset, hdr->payload_len);
927 }
928
rndis_send(struct net_pkt * pkt)929 static int rndis_send(struct net_pkt *pkt)
930 {
931 size_t len = net_pkt_get_len(pkt);
932 int ret;
933
934 LOG_DBG("send pkt %p len %u", pkt, len);
935
936 if (rndis.media_status == RNDIS_OBJECT_ID_MEDIA_DISCONNECTED) {
937 LOG_DBG("Media disconnected, drop pkt %p", pkt);
938 return -EPIPE;
939 }
940
941 if (VERBOSE_DEBUG) {
942 net_pkt_hexdump(pkt, "<");
943 }
944
945 if (len + sizeof(struct rndis_payload_packet) > sizeof(tx_buf)) {
946 LOG_WRN("Trying to send too large packet, drop");
947 return -ENOMEM;
948 }
949
950 rndis_hdr_add(tx_buf, len);
951
952 ret = net_pkt_read(pkt,
953 tx_buf + sizeof(struct rndis_payload_packet),
954 len);
955 if (ret < 0) {
956 return ret;
957 }
958
959 ret = usb_transfer_sync(rndis_ep_data[RNDIS_IN_EP_IDX].ep_addr, tx_buf,
960 len + sizeof(struct rndis_payload_packet),
961 USB_TRANS_WRITE);
962 if (ret != len + sizeof(struct rndis_payload_packet)) {
963 LOG_ERR("Transfer failure");
964 return ret;
965 }
966
967 return 0;
968 }
969
970 #if defined(CONFIG_USB_DEVICE_OS_DESC)
971 /* This string descriptor would be read the first time device is plugged in.
972 * It is Microsoft extension called an OS String Descriptor
973 */
974 #define MSOS_STRING_LENGTH 18
975 static struct string_desc {
976 uint8_t bLength;
977 uint8_t bDescriptorType;
978 uint8_t bString[MSOS_STRING_LENGTH - 4];
979 uint8_t bMS_VendorCode;
980 uint8_t bPad;
981 } __packed msosv1_string_descriptor = {
982 .bLength = MSOS_STRING_LENGTH,
983 .bDescriptorType = USB_DESC_STRING,
984 /* Signature MSFT100 */
985 .bString = {
986 'M', 0x00, 'S', 0x00, 'F', 0x00, 'T', 0x00,
987 '1', 0x00, '0', 0x00, '0', 0x00
988 },
989 .bMS_VendorCode = 0x03, /* Vendor Code, used for a control request */
990 .bPad = 0x00, /* Padding byte for VendorCode look as UTF16 */
991 };
992
993 static struct compat_id_desc {
994 /* MS OS 1.0 Header Section */
995 uint32_t dwLength;
996 uint16_t bcdVersion;
997 uint16_t wIndex;
998 uint8_t bCount;
999 uint8_t Reserved[7];
1000 /* MS OS 1.0 Function Section */
1001 struct compat_id_func {
1002 uint8_t bFirstInterfaceNumber;
1003 uint8_t Reserved1;
1004 uint8_t compatibleID[8];
1005 uint8_t subCompatibleID[8];
1006 uint8_t Reserved2[6];
1007 } __packed func[1];
1008 } __packed msosv1_compatid_descriptor = {
1009 .dwLength = sys_cpu_to_le32(40),
1010 .bcdVersion = sys_cpu_to_le16(0x0100),
1011 .wIndex = sys_cpu_to_le16(USB_OSDESC_EXTENDED_COMPAT_ID),
1012 .bCount = 0x01, /* One function section */
1013 .Reserved = {
1014 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00
1015 },
1016
1017 .func = {
1018 {
1019 .bFirstInterfaceNumber = 0x00,
1020 .Reserved1 = 0x01,
1021 .compatibleID = {
1022 'R', 'N', 'D', 'I', 'S', 0x00, 0x00, 0x00
1023 },
1024 .subCompatibleID = {
1025 '5', '1', '6', '2', '0', '0', '1', 0x00
1026 },
1027 .Reserved2 = {
1028 0x00, 0x00, 0x00, 0x00, 0x00, 0x00
1029 }
1030 },
1031 }
1032 };
1033
1034 static struct usb_os_descriptor os_desc = {
1035 .string = (uint8_t *)&msosv1_string_descriptor,
1036 .string_len = sizeof(msosv1_string_descriptor),
1037 .vendor_code = 0x03,
1038 .compat_id = (uint8_t *)&msosv1_compatid_descriptor,
1039 .compat_id_len = sizeof(msosv1_compatid_descriptor),
1040 };
1041 #endif /* CONFIG_USB_DEVICE_OS_DESC */
1042
rndis_init(void)1043 static int rndis_init(void)
1044 {
1045
1046 LOG_DBG("RNDIS initialization");
1047
1048 /* Transmit queue init */
1049 k_fifo_init(&rndis_tx_queue);
1050 /* Command queue init */
1051 k_fifo_init(&rndis_cmd_queue);
1052
1053 /* Register MS OS Descriptor */
1054 usb_register_os_desc(&os_desc);
1055
1056 k_thread_create(&cmd_thread_data, cmd_stack,
1057 K_KERNEL_STACK_SIZEOF(cmd_stack),
1058 cmd_thread,
1059 NULL, NULL, NULL, K_PRIO_COOP(8), 0, K_NO_WAIT);
1060
1061 k_thread_name_set(&cmd_thread_data, "usb_rndis");
1062
1063 return 0;
1064 }
1065
rndis_connect_media(bool status)1066 static int rndis_connect_media(bool status)
1067 {
1068 if (status) {
1069 rndis.media_status = RNDIS_OBJECT_ID_MEDIA_CONNECTED;
1070 } else {
1071 rndis.media_status = RNDIS_OBJECT_ID_MEDIA_DISCONNECTED;
1072 }
1073
1074 return 0;
1075 }
1076
1077 static struct netusb_function rndis_function = {
1078 .connect_media = rndis_connect_media,
1079 .send_pkt = rndis_send,
1080 };
1081
rndis_status_cb(struct usb_cfg_data * cfg,enum usb_dc_status_code status,const uint8_t * param)1082 static void rndis_status_cb(struct usb_cfg_data *cfg,
1083 enum usb_dc_status_code status,
1084 const uint8_t *param)
1085 {
1086 ARG_UNUSED(cfg);
1087
1088 /* Check the USB status and do needed action if required */
1089 switch (status) {
1090 case USB_DC_CONFIGURED:
1091 LOG_DBG("USB device configured");
1092 netusb_enable(&rndis_function);
1093 break;
1094
1095 case USB_DC_DISCONNECTED:
1096 LOG_DBG("USB device disconnected");
1097 netusb_disable();
1098 break;
1099
1100 case USB_DC_ERROR:
1101 case USB_DC_RESET:
1102 case USB_DC_CONNECTED:
1103 case USB_DC_SUSPEND:
1104 case USB_DC_RESUME:
1105 case USB_DC_INTERFACE:
1106 LOG_DBG("USB unhandled state: %d", status);
1107 break;
1108
1109 case USB_DC_SOF:
1110 break;
1111
1112 case USB_DC_UNKNOWN:
1113 default:
1114 LOG_DBG("USB unknown state %d", status);
1115 break;
1116 }
1117 }
1118
netusb_interface_config(struct usb_desc_header * head,uint8_t bInterfaceNumber)1119 static void netusb_interface_config(struct usb_desc_header *head,
1120 uint8_t bInterfaceNumber)
1121 {
1122 ARG_UNUSED(head);
1123
1124 rndis_cfg.if0.bInterfaceNumber = bInterfaceNumber;
1125 rndis_cfg.if1.bInterfaceNumber = bInterfaceNumber + 1;
1126 rndis_cfg.iad.bFirstInterface = bInterfaceNumber;
1127 }
1128
1129 USBD_DEFINE_CFG_DATA(rndis_config) = {
1130 .usb_device_description = NULL,
1131 .interface_config = netusb_interface_config,
1132 .interface_descriptor = &rndis_cfg.if0,
1133 .cb_usb_status = rndis_status_cb,
1134 .interface = {
1135 .class_handler = rndis_class_handler,
1136 .custom_handler = NULL,
1137 .vendor_handler = NULL,
1138 },
1139 .num_endpoints = ARRAY_SIZE(rndis_ep_data),
1140 .endpoint = rndis_ep_data,
1141 };
1142
1143 /* Initialize this before eth_netusb device init */
1144 SYS_INIT(rndis_init, POST_KERNEL, 0);
1145