1 /*
2 * Copyright (c) 2015 - 2016, Freescale Semiconductor, Inc.
3 * Copyright 2016,2019 NXP
4 * All rights reserved.
5 *
6 * SPDX-License-Identifier: BSD-3-Clause
7 */
8
9 #include "usb_host_config.h"
10 #if ((defined USB_HOST_CONFIG_HUB) && (USB_HOST_CONFIG_HUB))
11 #include "usb_host.h"
12 #include "usb_host_hub.h"
13
14 /*******************************************************************************
15 * Definitions
16 ******************************************************************************/
17
18 /*******************************************************************************
19 * Prototypes
20 ******************************************************************************/
21 /*!
22 * @brief hub control transfer callback.
23 *
24 * @param param callback parameter.
25 * @param transfer callback transfer.
26 * @param status transfer status.
27 */
28 static void USB_HostHubControlCallback(void *param, usb_host_transfer_t *transfer, usb_status_t status);
29
30 /*!
31 * @brief hub interrupt in transfer callback.
32 *
33 * @param param callback parameter.
34 * @param transfer callback transfer.
35 * @param status transfer status.
36 */
37 static void USB_HostHubInPipeCallback(void *param, usb_host_transfer_t *transfer, usb_status_t status);
38
39 /*!
40 * @brief USB_HostHubSendPortReset's transfer callback.
41 *
42 * @param param callback parameter.
43 * @param transfer callback transfer.
44 * @param status transfer status.
45 */
46 static void USB_HostHubResetCallback(void *param, usb_host_transfer_t *transfer, usb_status_t status);
47
48 /*!
49 * @brief hub control transfer common code.
50 *
51 * @param classHandle the class handle.
52 * @param requestType request type.
53 * @param request setup packet request field.
54 * @param wvalue setup packet wValue field.
55 * @param windex setup packet wIndex field.
56 * @param buffer the buffer pointer.
57 * @param bufferLength the buffer length.
58 * @param callbackFn this callback is called after this function completes.
59 * @param callbackParam the first parameter in the callback function.
60 *
61 * @return kStatus_USB_Success or error codes.
62 */
63 static usb_status_t USB_HostHubClassRequestCommon(usb_host_class_handle classHandle,
64 uint8_t requestType,
65 uint8_t request,
66 uint16_t wvalue,
67 uint16_t windex,
68 uint8_t *buffer,
69 uint16_t bufferLength,
70 transfer_callback_t callbackFn,
71 void *callbackParam);
72
73 /*******************************************************************************
74 * Variables
75 ******************************************************************************/
76
77 /*******************************************************************************
78 * Code
79 ******************************************************************************/
80
USB_HostHubControlCallback(void * param,usb_host_transfer_t * transfer,usb_status_t status)81 static void USB_HostHubControlCallback(void *param, usb_host_transfer_t *transfer, usb_status_t status)
82 {
83 usb_host_hub_instance_t *hubInstance = (usb_host_hub_instance_t *)param;
84
85 hubInstance->controlTransfer = NULL;
86 if (NULL != hubInstance->controlCallbackFn)
87 {
88 /* callback to application, callback function is initialized in the USB_HostPrinterControl,
89 USB_HostPrinterSetInterface
90 or USB_HostHubClassRequestCommon, but is the same function */
91 hubInstance->controlCallbackFn(hubInstance->controlCallbackParam, transfer->transferBuffer,
92 transfer->transferSofar, status); /* callback to application */
93 }
94 (void)USB_HostFreeTransfer(hubInstance->hostHandle, transfer);
95 }
96
USB_HostHubInPipeCallback(void * param,usb_host_transfer_t * transfer,usb_status_t status)97 static void USB_HostHubInPipeCallback(void *param, usb_host_transfer_t *transfer, usb_status_t status)
98 {
99 usb_host_hub_instance_t *hubInstance = (usb_host_hub_instance_t *)param;
100
101 if (NULL != hubInstance->inCallbackFn)
102 {
103 /* callback to application, callback function is initialized in the USB_HostHubInterruptRecv */
104 hubInstance->inCallbackFn(hubInstance->inCallbackParam, transfer->transferBuffer, transfer->transferSofar,
105 status); /* callback to application */
106 }
107 (void)USB_HostFreeTransfer(hubInstance->hostHandle, transfer);
108 }
109
USB_HostHubResetCallback(void * param,usb_host_transfer_t * transfer,usb_status_t status)110 static void USB_HostHubResetCallback(void *param, usb_host_transfer_t *transfer, usb_status_t status)
111 {
112 usb_host_hub_instance_t *hubInstance = (usb_host_hub_instance_t *)param;
113
114 /* note: there is not callback to application, the re-enumeration will start automatically after reset. */
115 (void)USB_HostFreeTransfer(hubInstance->hostHandle, transfer);
116 }
117
USB_HostHubClassRequestCommon(usb_host_class_handle classHandle,uint8_t requestType,uint8_t request,uint16_t wvalue,uint16_t windex,uint8_t * buffer,uint16_t bufferLength,transfer_callback_t callbackFn,void * callbackParam)118 static usb_status_t USB_HostHubClassRequestCommon(usb_host_class_handle classHandle,
119 uint8_t requestType,
120 uint8_t request,
121 uint16_t wvalue,
122 uint16_t windex,
123 uint8_t *buffer,
124 uint16_t bufferLength,
125 transfer_callback_t callbackFn,
126 void *callbackParam)
127 {
128 usb_host_hub_instance_t *hubInstance = (usb_host_hub_instance_t *)classHandle;
129 usb_host_transfer_t *transfer;
130
131 if (hubInstance->controlTransfer != NULL)
132 {
133 return kStatus_USB_Busy;
134 }
135
136 /* get transfer */
137 if (USB_HostMallocTransfer(hubInstance->hostHandle, &transfer) != kStatus_USB_Success)
138 {
139 #ifdef HOST_ECHO
140 usb_echo("error to get transfer\r\n");
141 #endif
142 return kStatus_USB_Error;
143 }
144
145 /* save hub application callback */
146 hubInstance->controlCallbackFn = callbackFn;
147 hubInstance->controlCallbackParam = callbackParam;
148
149 /* initialize transfer */
150 transfer->transferBuffer = buffer;
151 transfer->transferLength = bufferLength;
152 transfer->callbackFn = USB_HostHubControlCallback;
153 transfer->callbackParam = hubInstance;
154 transfer->setupPacket->bmRequestType = requestType;
155 transfer->setupPacket->bRequest = request;
156 transfer->setupPacket->wValue = USB_SHORT_TO_LITTLE_ENDIAN(wvalue);
157 transfer->setupPacket->wIndex = USB_SHORT_TO_LITTLE_ENDIAN(windex);
158 transfer->setupPacket->wLength = USB_SHORT_TO_LITTLE_ENDIAN(bufferLength);
159
160 /* send transfer */
161 if (USB_HostSendSetup(hubInstance->hostHandle, hubInstance->controlPipe, transfer) != kStatus_USB_Success)
162 {
163 #ifdef HOST_ECHO
164 usb_echo("Error in hid get report descriptor\r\n");
165 #endif
166 (void)USB_HostFreeTransfer(hubInstance->hostHandle, transfer);
167 return kStatus_USB_Error;
168 }
169 hubInstance->controlTransfer = transfer; /* record the on-going setup transfer */
170 return kStatus_USB_Success;
171 }
172
USB_HostHubInit(usb_device_handle deviceHandle,usb_host_class_handle * classHandle)173 usb_status_t USB_HostHubInit(usb_device_handle deviceHandle, usb_host_class_handle *classHandle)
174 {
175 /* malloc the hub instance */
176 usb_host_hub_instance_t *hubInstance =
177 (usb_host_hub_instance_t *)OSA_MemoryAllocate(sizeof(usb_host_hub_instance_t));
178 uint32_t *temp;
179 uint32_t infoValue = 0U;
180
181 if (hubInstance == NULL)
182 {
183 return kStatus_USB_AllocFail;
184 }
185
186 #if ((defined(USB_HOST_CONFIG_BUFFER_PROPERTY_CACHEABLE)) && (USB_HOST_CONFIG_BUFFER_PROPERTY_CACHEABLE > 0U))
187 hubInstance->hubDescriptor = (uint8_t *)OSA_MemoryAllocateAlign(7 + (USB_HOST_HUB_MAX_PORT >> 3) + 1, USB_CACHE_LINESIZE);
188 hubInstance->portStatusBuffer = (uint8_t *)OSA_MemoryAllocateAlign(4, USB_CACHE_LINESIZE);
189 hubInstance->hubStatusBuffer = (uint8_t *)OSA_MemoryAllocateAlign(4, USB_CACHE_LINESIZE);
190 hubInstance->hubBitmapBuffer = (uint8_t *)OSA_MemoryAllocateAlign((USB_HOST_HUB_MAX_PORT >> 3) + 1, USB_CACHE_LINESIZE);
191 #endif
192
193 /* initialize hub instance structure */
194 hubInstance->deviceHandle = deviceHandle;
195 hubInstance->interfaceHandle = NULL;
196 (void)USB_HostHelperGetPeripheralInformation(deviceHandle, (uint32_t)kUSB_HostGetHostHandle, &infoValue);
197 temp = (uint32_t *)infoValue;
198 hubInstance->hostHandle = (usb_host_handle)temp;
199 (void)USB_HostHelperGetPeripheralInformation(deviceHandle, (uint32_t)kUSB_HostGetDeviceControlPipe, &infoValue);
200 temp = (uint32_t *)infoValue;
201 hubInstance->controlPipe = (usb_host_pipe_handle)temp;
202 (void)USB_HostHelperGetPeripheralInformation(deviceHandle, (uint32_t)kUSB_HostGetDeviceLevel, &infoValue);
203 hubInstance->hubLevel = (uint8_t)infoValue;
204
205 *classHandle = hubInstance; /* return the hub class handle */
206 return kStatus_USB_Success;
207 }
208
USB_HostHubSetInterface(usb_host_class_handle classHandle,usb_host_interface_handle interfaceHandle,uint8_t alternateSetting,transfer_callback_t callbackFn,void * callbackParam)209 usb_status_t USB_HostHubSetInterface(usb_host_class_handle classHandle,
210 usb_host_interface_handle interfaceHandle,
211 uint8_t alternateSetting,
212 transfer_callback_t callbackFn,
213 void *callbackParam)
214 {
215 usb_status_t status;
216 usb_host_hub_instance_t *hubInstance = (usb_host_hub_instance_t *)classHandle;
217 usb_host_interface_t *interface = (usb_host_interface_t *)interfaceHandle;
218 usb_descriptor_endpoint_t *epDesc = NULL;
219 usb_host_pipe_init_t pipeInit;
220 uint8_t epIndex;
221
222 if (classHandle == NULL)
223 {
224 return kStatus_USB_InvalidHandle;
225 }
226
227 hubInstance->interfaceHandle = interfaceHandle; /* save the interface handle */
228
229 /* notify the host driver that the interface is used by class */
230 status = USB_HostOpenDeviceInterface(hubInstance->deviceHandle, interfaceHandle);
231 if (status != kStatus_USB_Success)
232 {
233 return status;
234 }
235
236 /* close opened hub interrupt pipe */
237 if (hubInstance->interruptPipe != NULL)
238 {
239 status = USB_HostCancelTransfer(hubInstance->hostHandle, hubInstance->interruptPipe, NULL);
240 if (status != kStatus_USB_Success)
241 {
242 #ifdef HOST_ECHO
243 usb_echo("error when Cancel pipe\r\n");
244 #endif
245 }
246 status = USB_HostClosePipe(hubInstance->hostHandle, hubInstance->interruptPipe);
247 if (status != kStatus_USB_Success)
248 {
249 #ifdef HOST_ECHO
250 usb_echo("error when close pipe\r\n");
251 #endif
252 }
253 hubInstance->interruptPipe = NULL;
254 }
255
256 /* open hub interrupt pipe */
257 for (epIndex = 0; epIndex < interface->epCount; ++epIndex)
258 {
259 epDesc = interface->epList[epIndex].epDesc;
260 if (((epDesc->bEndpointAddress & USB_DESCRIPTOR_ENDPOINT_ADDRESS_DIRECTION_MASK) ==
261 USB_DESCRIPTOR_ENDPOINT_ADDRESS_DIRECTION_IN) &&
262 ((epDesc->bmAttributes & USB_DESCRIPTOR_ENDPOINT_ATTRIBUTE_TYPE_MASK) == USB_ENDPOINT_INTERRUPT))
263 {
264 /* get pipe information from endpoint descriptor */
265 pipeInit.devInstance = hubInstance->deviceHandle;
266 pipeInit.pipeType = USB_ENDPOINT_INTERRUPT;
267 pipeInit.direction = USB_IN;
268 pipeInit.endpointAddress = (epDesc->bEndpointAddress & USB_DESCRIPTOR_ENDPOINT_ADDRESS_NUMBER_MASK);
269 pipeInit.interval = epDesc->bInterval;
270 pipeInit.maxPacketSize = (uint16_t)((USB_SHORT_FROM_LITTLE_ENDIAN_ADDRESS(epDesc->wMaxPacketSize) &
271 USB_DESCRIPTOR_ENDPOINT_MAXPACKETSIZE_SIZE_MASK));
272 pipeInit.numberPerUframe = (uint8_t)((USB_SHORT_FROM_LITTLE_ENDIAN_ADDRESS(epDesc->wMaxPacketSize) &
273 USB_DESCRIPTOR_ENDPOINT_MAXPACKETSIZE_MULT_TRANSACTIONS_MASK));
274 pipeInit.nakCount = USB_HOST_CONFIG_MAX_NAK;
275
276 /* open hub interrupt in pipe */
277 status = USB_HostOpenPipe(hubInstance->hostHandle, &hubInstance->interruptPipe, &pipeInit);
278 if (status != kStatus_USB_Success)
279 {
280 #ifdef HOST_ECHO
281 usb_echo("usb_host_hid_set_interface fail to open pipe\r\n");
282 #endif
283 return kStatus_USB_Error;
284 }
285 break;
286 }
287 }
288
289 /* hub don't support alternatesetting that is not 0 */
290 if (alternateSetting == 0U)
291 {
292 if (callbackFn != NULL)
293 {
294 callbackFn(callbackParam, NULL, 0, kStatus_USB_Success);
295 }
296 }
297 else
298 {
299 #ifdef HOST_ECHO
300 usb_echo("host don't support alternate setting\r\n");
301 #endif
302 return kStatus_USB_Error;
303 }
304
305 return status;
306 }
307
USB_HostHubDeinit(usb_device_handle deviceHandle,usb_host_class_handle classHandle)308 usb_status_t USB_HostHubDeinit(usb_device_handle deviceHandle, usb_host_class_handle classHandle)
309 {
310 usb_host_hub_instance_t *hubInstance = (usb_host_hub_instance_t *)classHandle;
311 usb_status_t status = kStatus_USB_Success;
312 if (deviceHandle == NULL)
313 {
314 return kStatus_USB_InvalidHandle;
315 }
316
317 if (classHandle != NULL)
318 {
319 /* close opened hub interrupt pipe */
320 if (hubInstance->interruptPipe != NULL)
321 {
322 status = USB_HostCancelTransfer(hubInstance->hostHandle, hubInstance->interruptPipe, NULL);
323 if (status != kStatus_USB_Success)
324 {
325 #ifdef HOST_ECHO
326 usb_echo("error when Cancel pipe\r\n");
327 #endif
328 }
329 status = USB_HostClosePipe(hubInstance->hostHandle, hubInstance->interruptPipe);
330 if (status != kStatus_USB_Success)
331 {
332 #ifdef HOST_ECHO
333 usb_echo("hub close interrupt pipe error\r\n");
334 #endif
335 }
336 hubInstance->interruptPipe = NULL;
337 }
338
339 /* cancel control transfer if exist */
340 if ((hubInstance->controlPipe != NULL) && (hubInstance->controlTransfer != NULL))
341 {
342 status =
343 USB_HostCancelTransfer(hubInstance->hostHandle, hubInstance->controlPipe, hubInstance->controlTransfer);
344 }
345
346 /* notify host driver that the interface will not be used */
347 (void)USB_HostCloseDeviceInterface(deviceHandle, hubInstance->interfaceHandle);
348 #if ((defined(USB_HOST_CONFIG_BUFFER_PROPERTY_CACHEABLE)) && (USB_HOST_CONFIG_BUFFER_PROPERTY_CACHEABLE > 0U))
349 OSA_MemoryFreeAlign(hubInstance->hubDescriptor);
350 OSA_MemoryFreeAlign(hubInstance->portStatusBuffer);
351 OSA_MemoryFreeAlign(hubInstance->hubStatusBuffer);
352 OSA_MemoryFreeAlign(hubInstance->hubBitmapBuffer);
353 #endif
354 OSA_MemoryFree(hubInstance);
355 }
356 else
357 {
358 /* notify host driver that the interface will not be used */
359 (void)USB_HostCloseDeviceInterface(deviceHandle, NULL);
360 }
361
362 return status;
363 }
364
USB_HostHubInterruptRecv(usb_host_class_handle classHandle,uint8_t * buffer,uint16_t bufferLength,transfer_callback_t callbackFn,void * callbackParam)365 usb_status_t USB_HostHubInterruptRecv(usb_host_class_handle classHandle,
366 uint8_t *buffer,
367 uint16_t bufferLength,
368 transfer_callback_t callbackFn,
369 void *callbackParam)
370 {
371 usb_host_hub_instance_t *hubInstance = (usb_host_hub_instance_t *)classHandle;
372 usb_host_transfer_t *transfer;
373
374 if (classHandle == NULL)
375 {
376 return kStatus_USB_InvalidHandle;
377 }
378
379 /* get transfer */
380 if (USB_HostMallocTransfer(hubInstance->hostHandle, &transfer) != kStatus_USB_Success)
381 {
382 #ifdef HOST_ECHO
383 usb_echo("error to get transfer\r\n");
384 #endif
385 return kStatus_USB_Error;
386 }
387
388 /* save hub application callback */
389 hubInstance->inCallbackFn = callbackFn;
390 hubInstance->inCallbackParam = callbackParam;
391
392 /* initialize transfer */
393 transfer->transferBuffer = buffer;
394 transfer->transferLength = bufferLength;
395 transfer->callbackFn = USB_HostHubInPipeCallback;
396 transfer->callbackParam = hubInstance;
397
398 /* call host driver API to receive data */
399 if (USB_HostRecv(hubInstance->hostHandle, hubInstance->interruptPipe, transfer) != kStatus_USB_Success)
400 {
401 #ifdef HOST_ECHO
402 usb_echo("failed to USB_HostRecv\r\n");
403 #endif
404 (void)USB_HostFreeTransfer(hubInstance->hostHandle, transfer);
405 return kStatus_USB_Error;
406 }
407
408 return kStatus_USB_Success;
409 }
410
USB_HostHubSendPortReset(usb_host_class_handle classHandle,uint8_t portNumber)411 usb_status_t USB_HostHubSendPortReset(usb_host_class_handle classHandle, uint8_t portNumber)
412 {
413 usb_host_hub_instance_t *hubInstance = (usb_host_hub_instance_t *)classHandle;
414 usb_host_transfer_t *transfer;
415
416 if (classHandle == NULL)
417 {
418 return kStatus_USB_InvalidHandle;
419 }
420
421 /* get transfer */
422 if (USB_HostMallocTransfer(hubInstance->hostHandle, &transfer) != kStatus_USB_Success)
423 {
424 #ifdef HOST_ECHO
425 usb_echo("error to get transfer\r\n");
426 #endif
427 return kStatus_USB_Busy;
428 }
429
430 /* initialize transfer */
431 transfer->transferBuffer = NULL;
432 transfer->transferLength = 0;
433 transfer->callbackFn = USB_HostHubResetCallback;
434 transfer->callbackParam = hubInstance;
435 transfer->setupPacket->bmRequestType =
436 USB_REQUEST_TYPE_DIR_OUT | USB_REQUEST_TYPE_TYPE_CLASS | USB_REQUEST_TYPE_RECIPIENT_OTHER;
437 transfer->setupPacket->bRequest = USB_REQUEST_STANDARD_SET_FEATURE;
438 transfer->setupPacket->wValue = USB_SHORT_TO_LITTLE_ENDIAN(PORT_RESET);
439 transfer->setupPacket->wIndex = USB_SHORT_TO_LITTLE_ENDIAN(portNumber);
440 transfer->setupPacket->wLength = 0;
441
442 /* send the transfer */
443 if (USB_HostSendSetup(hubInstance->hostHandle, hubInstance->controlPipe, transfer) != kStatus_USB_Success)
444 {
445 #ifdef HOST_ECHO
446 usb_echo("Error in hid get report descriptor\r\n");
447 #endif
448 (void)USB_HostFreeTransfer(hubInstance->hostHandle, transfer);
449 return kStatus_USB_Error;
450 }
451 return kStatus_USB_Success;
452 }
453
454 /*!
455 * @brief hub get descriptor.
456 *
457 * This function implements get hub descriptor specific request.
458 *
459 * @param classHandle the class handle.
460 * @param buffer the buffer pointer.
461 * @param bufferLength the buffer length.
462 * @param callbackFn this callback is called after this function completes.
463 * @param callbackParam the first parameter in the callback function.
464 *
465 * @retval kStatus_USB_Success send successfully.
466 * @retval kStatus_USB_InvalidHandle The classHandle is NULL pointer.
467 * @retval kStatus_USB_Busy There is no idle transfer.
468 * @retval kStatus_USB_Error pipe is not initialized.
469 * Or, send transfer fail, please reference to USB_HostSendSetup.
470 */
USB_HostHubGetDescriptor(usb_host_class_handle classHandle,uint8_t * buffer,uint16_t bufferLength,transfer_callback_t callbackFn,void * callbackParam)471 usb_status_t USB_HostHubGetDescriptor(usb_host_class_handle classHandle,
472 uint8_t *buffer,
473 uint16_t bufferLength,
474 transfer_callback_t callbackFn,
475 void *callbackParam)
476 {
477 return USB_HostHubClassRequestCommon(
478 classHandle, USB_REQUEST_TYPE_DIR_IN | USB_REQUEST_TYPE_TYPE_CLASS | USB_REQUEST_TYPE_RECIPIENT_DEVICE,
479 USB_REQUEST_STANDARD_GET_DESCRIPTOR, 0x00, 0, buffer, bufferLength, callbackFn, callbackParam);
480 }
481
482 /*!
483 * @brief hub clear feature.
484 *
485 * This function implements clear hub feature specific request.
486 *
487 * @param classHandle the class handle.
488 * @param buffer the buffer pointer.
489 * @param bufferLength the buffer length.
490 * @param callbackFn this callback is called after this function completes.
491 * @param callbackParam the first parameter in the callback function.
492 *
493 * @retval kStatus_USB_Success send successfully.
494 * @retval kStatus_USB_InvalidHandle The classHandle is NULL pointer.
495 * @retval kStatus_USB_Busy There is no idle transfer.
496 * @retval kStatus_USB_Error pipe is not initialized.
497 * Or, send transfer fail, please reference to USB_HostSendSetup.
498 */
USB_HostHubClearFeature(usb_host_class_handle classHandle,uint8_t feature,transfer_callback_t callbackFn,void * callbackParam)499 usb_status_t USB_HostHubClearFeature(usb_host_class_handle classHandle,
500 uint8_t feature,
501 transfer_callback_t callbackFn,
502 void *callbackParam)
503 {
504 return USB_HostHubClassRequestCommon(classHandle, USB_REQUEST_TYPE_DIR_OUT | USB_REQUEST_TYPE_TYPE_CLASS,
505 USB_REQUEST_STANDARD_CLEAR_FEATURE, feature, 0, NULL, 0, callbackFn,
506 callbackParam);
507 }
508
509 /*!
510 * @brief hub get status.
511 *
512 * This function implements get hub status specific request.
513 *
514 * @param classHandle the class handle.
515 * @param buffer the buffer pointer.
516 * @param bufferLength the buffer length.
517 * @param callbackFn this callback is called after this function completes.
518 * @param callbackParam the first parameter in the callback function.
519 *
520 * @retval kStatus_USB_Success send successfully.
521 * @retval kStatus_USB_InvalidHandle The classHandle is NULL pointer.
522 * @retval kStatus_USB_Busy There is no idle transfer.
523 * @retval kStatus_USB_Error pipe is not initialized.
524 * Or, send transfer fail, please reference to USB_HostSendSetup.
525 */
USB_HostHubGetStatus(usb_host_class_handle classHandle,uint8_t * buffer,uint16_t bufferLength,transfer_callback_t callbackFn,void * callbackParam)526 usb_status_t USB_HostHubGetStatus(usb_host_class_handle classHandle,
527 uint8_t *buffer,
528 uint16_t bufferLength,
529 transfer_callback_t callbackFn,
530 void *callbackParam)
531 {
532 return USB_HostHubClassRequestCommon(classHandle, USB_REQUEST_TYPE_DIR_IN | USB_REQUEST_TYPE_TYPE_CLASS,
533 USB_REQUEST_STANDARD_GET_STATUS, 0, 0, buffer, bufferLength, callbackFn,
534 callbackParam);
535 }
536
537 /*!
538 * @brief hub set feature.
539 *
540 * This function implements set hub feature specific request.
541 *
542 * @param classHandle the class handle.
543 * @param buffer the buffer pointer.
544 * @param bufferLength the buffer length.
545 * @param callbackFn this callback is called after this function completes.
546 * @param callbackParam the first parameter in the callback function.
547 *
548 * @retval kStatus_USB_Success send successfully.
549 * @retval kStatus_USB_InvalidHandle The classHandle is NULL pointer.
550 * @retval kStatus_USB_Busy There is no idle transfer.
551 * @retval kStatus_USB_Error pipe is not initialized.
552 * Or, send transfer fail, please reference to USB_HostSendSetup.
553 */
USB_HostHubSetPortFeature(usb_host_class_handle classHandle,uint8_t portNumber,uint8_t feature,transfer_callback_t callbackFn,void * callbackParam)554 usb_status_t USB_HostHubSetPortFeature(usb_host_class_handle classHandle,
555 uint8_t portNumber,
556 uint8_t feature,
557 transfer_callback_t callbackFn,
558 void *callbackParam)
559 {
560 return USB_HostHubClassRequestCommon(
561 classHandle, USB_REQUEST_TYPE_DIR_OUT | USB_REQUEST_TYPE_TYPE_CLASS | USB_REQUEST_TYPE_RECIPIENT_OTHER,
562 USB_REQUEST_STANDARD_SET_FEATURE, feature, portNumber, NULL, 0, callbackFn, callbackParam);
563 }
564
565 /*!
566 * @brief hub clear port feature.
567 *
568 * This function implements clear hub port feature specific request.
569 *
570 * @param classHandle the class handle.
571 * @param buffer the buffer pointer.
572 * @param bufferLength the buffer length.
573 * @param callbackFn this callback is called after this function completes.
574 * @param callbackParam the first parameter in the callback function.
575 *
576 * @retval kStatus_USB_Success send successfully.
577 * @retval kStatus_USB_InvalidHandle The classHandle is NULL pointer.
578 * @retval kStatus_USB_Busy There is no idle transfer.
579 * @retval kStatus_USB_Error pipe is not initialized.
580 * Or, send transfer fail, please reference to USB_HostSendSetup.
581 */
USB_HostHubClearPortFeature(usb_host_class_handle classHandle,uint8_t portNumber,uint8_t feature,transfer_callback_t callbackFn,void * callbackParam)582 usb_status_t USB_HostHubClearPortFeature(usb_host_class_handle classHandle,
583 uint8_t portNumber,
584 uint8_t feature,
585 transfer_callback_t callbackFn,
586 void *callbackParam)
587 {
588 return USB_HostHubClassRequestCommon(
589 classHandle, USB_REQUEST_TYPE_DIR_OUT | USB_REQUEST_TYPE_TYPE_CLASS | USB_REQUEST_TYPE_RECIPIENT_OTHER,
590 USB_REQUEST_STANDARD_CLEAR_FEATURE, feature, portNumber, NULL, 0, callbackFn, callbackParam);
591 }
592
593 /*!
594 * @brief hub port get status.
595 *
596 * This function implements get hub port status specific request.
597 *
598 * @param classHandle the class handle.
599 * @param buffer the buffer pointer.
600 * @param bufferLength the buffer length.
601 * @param callbackFn this callback is called after this function completes.
602 * @param callbackParam the first parameter in the callback function.
603 *
604 * @retval kStatus_USB_Success send successfully.
605 * @retval kStatus_USB_InvalidHandle The classHandle is NULL pointer.
606 * @retval kStatus_USB_Busy There is no idle transfer.
607 * @retval kStatus_USB_Error pipe is not initialized.
608 * Or, send transfer fail, please reference to USB_HostSendSetup.
609 */
USB_HostHubGetPortStatus(usb_host_class_handle classHandle,uint8_t portNumber,uint8_t * buffer,uint16_t bufferLength,transfer_callback_t callbackFn,void * callbackParam)610 usb_status_t USB_HostHubGetPortStatus(usb_host_class_handle classHandle,
611 uint8_t portNumber,
612 uint8_t *buffer,
613 uint16_t bufferLength,
614 transfer_callback_t callbackFn,
615 void *callbackParam)
616 {
617 return USB_HostHubClassRequestCommon(
618 classHandle, USB_REQUEST_TYPE_DIR_IN | USB_REQUEST_TYPE_TYPE_CLASS | USB_REQUEST_TYPE_RECIPIENT_OTHER,
619 USB_REQUEST_STANDARD_GET_STATUS, 0, portNumber, buffer, bufferLength, callbackFn, callbackParam);
620 }
621
622 #endif /* USB_HOST_CONFIG_HUB */
623