1 /*
2 * Copyright (c) 2022 Nordic Semiconductor ASA
3 *
4 * SPDX-License-Identifier: Apache-2.0
5 */
6
7 #include <zephyr/drivers/usb/udc.h>
8 #include <zephyr/usb/usbd.h>
9
10 #include "usbd_device.h"
11 #include "usbd_config.h"
12 #include "usbd_class.h"
13 #include "usbd_ch9.h"
14 #include "usbd_desc.h"
15
16 #include <zephyr/logging/log.h>
17 LOG_MODULE_REGISTER(usbd_dev, CONFIG_USBD_LOG_LEVEL);
18
19 /*
20 * All the functions below are part of public USB device support API.
21 */
22
usbd_bus_speed(const struct usbd_context * const uds_ctx)23 enum usbd_speed usbd_bus_speed(const struct usbd_context *const uds_ctx)
24 {
25 return uds_ctx->status.speed;
26 }
27
usbd_caps_speed(const struct usbd_context * const uds_ctx)28 enum usbd_speed usbd_caps_speed(const struct usbd_context *const uds_ctx)
29 {
30 struct udc_device_caps caps = udc_caps(uds_ctx->dev);
31
32 /* For now, either high speed is supported or not. */
33 if (caps.hs) {
34 return USBD_SPEED_HS;
35 }
36
37 return USBD_SPEED_FS;
38 }
39
40 static struct usb_device_descriptor *
get_device_descriptor(struct usbd_context * const uds_ctx,const enum usbd_speed speed)41 get_device_descriptor(struct usbd_context *const uds_ctx,
42 const enum usbd_speed speed)
43 {
44 switch (speed) {
45 case USBD_SPEED_FS:
46 return uds_ctx->fs_desc;
47 case USBD_SPEED_HS:
48 return uds_ctx->hs_desc;
49 default:
50 __ASSERT(false, "Not supported speed");
51 return NULL;
52 }
53 }
54
usbd_device_set_bcd_usb(struct usbd_context * const uds_ctx,const enum usbd_speed speed,const uint16_t bcd)55 int usbd_device_set_bcd_usb(struct usbd_context *const uds_ctx,
56 const enum usbd_speed speed, const uint16_t bcd)
57 {
58 struct usb_device_descriptor *desc;
59 int ret = 0;
60
61 usbd_device_lock(uds_ctx);
62
63 if (usbd_is_enabled(uds_ctx)) {
64 ret = -EALREADY;
65 goto set_bcd_exit;
66 }
67
68 desc = get_device_descriptor(uds_ctx, speed);
69 desc->bcdUSB = sys_cpu_to_le16(bcd);
70
71 set_bcd_exit:
72 usbd_device_unlock(uds_ctx);
73 return ret;
74 }
75
usbd_device_set_vid(struct usbd_context * const uds_ctx,const uint16_t vid)76 int usbd_device_set_vid(struct usbd_context *const uds_ctx,
77 const uint16_t vid)
78 {
79 struct usb_device_descriptor *fs_desc, *hs_desc;
80 int ret = 0;
81
82 usbd_device_lock(uds_ctx);
83
84 if (usbd_is_enabled(uds_ctx)) {
85 ret = -EALREADY;
86 goto set_vid_exit;
87 }
88
89 fs_desc = get_device_descriptor(uds_ctx, USBD_SPEED_FS);
90 fs_desc->idVendor = sys_cpu_to_le16(vid);
91
92 hs_desc = get_device_descriptor(uds_ctx, USBD_SPEED_HS);
93 hs_desc->idVendor = sys_cpu_to_le16(vid);
94
95 set_vid_exit:
96 usbd_device_unlock(uds_ctx);
97 return ret;
98 }
99
usbd_device_set_pid(struct usbd_context * const uds_ctx,const uint16_t pid)100 int usbd_device_set_pid(struct usbd_context *const uds_ctx,
101 const uint16_t pid)
102 {
103 struct usb_device_descriptor *fs_desc, *hs_desc;
104 int ret = 0;
105
106 usbd_device_lock(uds_ctx);
107
108 if (usbd_is_enabled(uds_ctx)) {
109 ret = -EALREADY;
110 goto set_pid_exit;
111 }
112
113 fs_desc = get_device_descriptor(uds_ctx, USBD_SPEED_FS);
114 fs_desc->idProduct = sys_cpu_to_le16(pid);
115
116 hs_desc = get_device_descriptor(uds_ctx, USBD_SPEED_HS);
117 hs_desc->idProduct = sys_cpu_to_le16(pid);
118
119 set_pid_exit:
120 usbd_device_unlock(uds_ctx);
121 return ret;
122 }
123
usbd_device_set_bcd_device(struct usbd_context * const uds_ctx,const uint16_t bcd)124 int usbd_device_set_bcd_device(struct usbd_context *const uds_ctx,
125 const uint16_t bcd)
126 {
127 struct usb_device_descriptor *fs_desc, *hs_desc;
128 int ret = 0;
129
130 usbd_device_lock(uds_ctx);
131
132 if (usbd_is_enabled(uds_ctx)) {
133 ret = -EALREADY;
134 goto set_bcd_device_exit;
135 }
136
137 fs_desc = get_device_descriptor(uds_ctx, USBD_SPEED_FS);
138 fs_desc->bcdDevice = sys_cpu_to_le16(bcd);
139
140 hs_desc = get_device_descriptor(uds_ctx, USBD_SPEED_HS);
141 hs_desc->bcdDevice = sys_cpu_to_le16(bcd);
142
143 set_bcd_device_exit:
144 usbd_device_unlock(uds_ctx);
145 return ret;
146 }
147
usbd_device_set_code_triple(struct usbd_context * const uds_ctx,const enum usbd_speed speed,const uint8_t base_class,const uint8_t subclass,const uint8_t protocol)148 int usbd_device_set_code_triple(struct usbd_context *const uds_ctx,
149 const enum usbd_speed speed,
150 const uint8_t base_class,
151 const uint8_t subclass, const uint8_t protocol)
152 {
153 struct usb_device_descriptor *desc;
154 int ret = 0;
155
156 usbd_device_lock(uds_ctx);
157
158 if (usbd_is_enabled(uds_ctx)) {
159 ret = -EALREADY;
160 goto set_code_triple_exit;
161 }
162
163 desc = get_device_descriptor(uds_ctx, speed);
164 desc->bDeviceClass = base_class;
165 desc->bDeviceSubClass = subclass;
166 desc->bDeviceProtocol = protocol;
167
168 set_code_triple_exit:
169 usbd_device_unlock(uds_ctx);
170 return ret;
171 }
172
usbd_wakeup_request(struct usbd_context * const uds_ctx)173 int usbd_wakeup_request(struct usbd_context *const uds_ctx)
174 {
175 struct udc_device_caps caps = udc_caps(uds_ctx->dev);
176 int ret = 0;
177
178 usbd_device_lock(uds_ctx);
179
180 if (!caps.rwup) {
181 LOG_ERR("Remote wakeup feature not supported");
182 ret = -ENOTSUP;
183 goto wakeup_request_error;
184 }
185
186 if (!uds_ctx->status.rwup || !usbd_is_suspended(uds_ctx)) {
187 LOG_WRN("Remote wakeup feature not enabled or not suspended");
188 ret = -EACCES;
189 goto wakeup_request_error;
190 }
191
192 ret = udc_host_wakeup(uds_ctx->dev);
193
194 wakeup_request_error:
195 usbd_device_unlock(uds_ctx);
196
197 return ret;
198 }
199
usbd_is_suspended(struct usbd_context * uds_ctx)200 bool usbd_is_suspended(struct usbd_context *uds_ctx)
201 {
202 return uds_ctx->status.suspended;
203 }
204
usbd_init(struct usbd_context * const uds_ctx)205 int usbd_init(struct usbd_context *const uds_ctx)
206 {
207 int ret;
208
209 /*
210 * Lock the scheduler to ensure that the context is not preempted
211 * before it is fully initialized.
212 */
213 k_sched_lock();
214 usbd_device_lock(uds_ctx);
215
216 if (uds_ctx->dev == NULL) {
217 ret = -ENODEV;
218 goto init_exit;
219 }
220
221 if (usbd_is_initialized(uds_ctx)) {
222 LOG_WRN("USB device support is already initialized");
223 ret = -EALREADY;
224 goto init_exit;
225 }
226
227 if (!device_is_ready(uds_ctx->dev)) {
228 LOG_ERR("USB device controller is not ready");
229 ret = -ENODEV;
230 goto init_exit;
231 }
232
233 ret = usbd_device_init_core(uds_ctx);
234 if (ret) {
235 goto init_exit;
236 }
237
238 memset(&uds_ctx->ch9_data, 0, sizeof(struct usbd_ch9_data));
239 uds_ctx->status.initialized = true;
240
241 init_exit:
242 usbd_device_unlock(uds_ctx);
243 k_sched_unlock();
244
245 return ret;
246 }
247
usbd_enable(struct usbd_context * const uds_ctx)248 int usbd_enable(struct usbd_context *const uds_ctx)
249 {
250 int ret;
251
252 usbd_device_lock(uds_ctx);
253
254 if (!usbd_is_initialized(uds_ctx)) {
255 LOG_WRN("USB device support is not initialized");
256 ret = -EPERM;
257 goto enable_exit;
258 }
259
260 if (usbd_is_enabled(uds_ctx)) {
261 LOG_WRN("USB device support is already enabled");
262 ret = -EALREADY;
263 goto enable_exit;
264 }
265
266 ret = udc_enable(uds_ctx->dev);
267 if (ret != 0) {
268 LOG_ERR("Failed to enable controller");
269 goto enable_exit;
270 }
271
272 ret = usbd_init_control_pipe(uds_ctx);
273 if (ret != 0) {
274 udc_disable(uds_ctx->dev);
275 goto enable_exit;
276 }
277
278 uds_ctx->status.enabled = true;
279
280 enable_exit:
281 usbd_device_unlock(uds_ctx);
282 return ret;
283 }
284
usbd_disable(struct usbd_context * const uds_ctx)285 int usbd_disable(struct usbd_context *const uds_ctx)
286 {
287 int ret;
288
289 if (!usbd_is_enabled(uds_ctx)) {
290 LOG_WRN("USB device support is already disabled");
291 return -EALREADY;
292 }
293
294 usbd_device_lock(uds_ctx);
295
296 ret = usbd_config_set(uds_ctx, 0);
297 if (ret) {
298 LOG_ERR("Failed to reset configuration");
299 }
300
301 ret = udc_disable(uds_ctx->dev);
302 if (ret) {
303 LOG_ERR("Failed to disable USB device");
304 }
305
306 uds_ctx->status.enabled = false;
307
308 usbd_device_unlock(uds_ctx);
309
310 return ret;
311 }
312
usbd_shutdown(struct usbd_context * const uds_ctx)313 int usbd_shutdown(struct usbd_context *const uds_ctx)
314 {
315 int ret;
316
317 usbd_device_lock(uds_ctx);
318
319 /* TODO: control request dequeue ? */
320 ret = usbd_device_shutdown_core(uds_ctx);
321 if (ret) {
322 LOG_ERR("Failed to shutdown USB device");
323 }
324
325 uds_ctx->status.initialized = false;
326 usbd_device_unlock(uds_ctx);
327
328 return 0;
329 }
330
usbd_can_detect_vbus(struct usbd_context * const uds_ctx)331 bool usbd_can_detect_vbus(struct usbd_context *const uds_ctx)
332 {
333 const struct udc_device_caps caps = udc_caps(uds_ctx->dev);
334
335 return caps.can_detect_vbus;
336 }
337
usbd_device_get_vreq(struct usbd_context * const uds_ctx,const uint8_t code)338 struct usbd_vreq_node *usbd_device_get_vreq(struct usbd_context *const uds_ctx,
339 const uint8_t code)
340 {
341 struct usbd_vreq_node *vreq_nd;
342
343 SYS_DLIST_FOR_EACH_CONTAINER(&uds_ctx->vreqs, vreq_nd, node) {
344 if (vreq_nd->code == code) {
345 return vreq_nd;
346 }
347 }
348
349 return NULL;
350 }
351
usbd_device_register_vreq(struct usbd_context * const uds_ctx,struct usbd_vreq_node * const vreq_nd)352 int usbd_device_register_vreq(struct usbd_context *const uds_ctx,
353 struct usbd_vreq_node *const vreq_nd)
354 {
355 int ret = 0;
356
357 usbd_device_lock(uds_ctx);
358
359 if (usbd_is_initialized(uds_ctx)) {
360 ret = -EPERM;
361 goto error;
362 }
363
364 if (vreq_nd->to_dev == NULL && vreq_nd->to_host == NULL) {
365 ret = -EINVAL;
366 goto error;
367 }
368
369 if (!sys_dnode_is_linked(&uds_ctx->vreqs)) {
370 LOG_DBG("Initialize vendor request list");
371 sys_dlist_init(&uds_ctx->vreqs);
372 }
373
374 if (sys_dnode_is_linked(&vreq_nd->node)) {
375 ret = -EALREADY;
376 goto error;
377 }
378
379 sys_dlist_append(&uds_ctx->vreqs, &vreq_nd->node);
380 LOG_DBG("Registered vendor request 0x%02x", vreq_nd->code);
381
382 error:
383 usbd_device_unlock(uds_ctx);
384 return ret;
385 }
386
usbd_device_unregister_all_vreq(struct usbd_context * const uds_ctx)387 void usbd_device_unregister_all_vreq(struct usbd_context *const uds_ctx)
388 {
389 struct usbd_vreq_node *tmp;
390 sys_dnode_t *node;
391
392 if (!sys_dnode_is_linked(&uds_ctx->vreqs)) {
393 return;
394 }
395
396 while ((node = sys_dlist_get(&uds_ctx->vreqs))) {
397 tmp = CONTAINER_OF(node, struct usbd_vreq_node, node);
398 LOG_DBG("Remove vendor request 0x%02x", tmp->code);
399 }
400 }
401