1 /***************************************************************************
2  * Copyright (c) 2024 Microsoft Corporation
3  *
4  * This program and the accompanying materials are made available under the
5  * terms of the MIT License which is available at
6  * https://opensource.org/licenses/MIT.
7  *
8  * SPDX-License-Identifier: MIT
9  **************************************************************************/
10 
11 
12 /**************************************************************************/
13 /**************************************************************************/
14 /**                                                                       */
15 /** USBX Component                                                        */
16 /**                                                                       */
17 /**   Host Stack                                                          */
18 /**                                                                       */
19 /**************************************************************************/
20 /**************************************************************************/
21 
22 
23 /* Include necessary system files.  */
24 
25 #define UX_SOURCE_CODE
26 
27 #include "ux_api.h"
28 #include "ux_host_stack.h"
29 
30 
31 /**************************************************************************/
32 /*                                                                        */
33 /*  FUNCTION                                               RELEASE        */
34 /*                                                                        */
35 /*    _ux_host_stack_configuration_enumerate              PORTABLE C      */
36 /*                                                           6.1          */
37 /*  AUTHOR                                                                */
38 /*                                                                        */
39 /*    Chaoqiong Xiao, Microsoft Corporation                               */
40 /*                                                                        */
41 /*  DESCRIPTION                                                           */
42 /*                                                                        */
43 /*    This function reads the configuration descriptor, creates the       */
44 /*    configuration container(s) for the device, and enumerates all found */
45 /*    configurations.                                                     */
46 /*                                                                        */
47 /*    At this stage, only the containers for each subcomponents are       */
48 /*    linked. No configuration, interface or endpoints are active unless  */
49 /*    a class issues a SET_CONFIGURATION.                                 */
50 /*                                                                        */
51 /*  INPUT                                                                 */
52 /*                                                                        */
53 /*    device                                Pointer to device             */
54 /*                                                                        */
55 /*  OUTPUT                                                                */
56 /*                                                                        */
57 /*    Completion Status                                                   */
58 /*                                                                        */
59 /*  CALLS                                                                 */
60 /*                                                                        */
61 /*    _ux_host_stack_configuration_descriptor_parse                       */
62 /*                                          Parse configuration descriptor*/
63 /*    _ux_host_stack_configuration_instance_delete                        */
64 /*                                          Delete configuration instance */
65 /*    _ux_host_stack_new_configuration_create                             */
66 /*                                          Create new configuration      */
67 /*    _ux_host_stack_transfer_request       Process transfer request      */
68 /*    _ux_utility_descriptor_parse          Parse descriptor              */
69 /*    _ux_utility_memory_allocate           Allocate block of memory      */
70 /*    _ux_utility_memory_free               Free block of memory          */
71 /*                                                                        */
72 /*  CALLED BY                                                             */
73 /*                                                                        */
74 /*    USBX Components                                                     */
75 /*                                                                        */
76 /*  RELEASE HISTORY                                                       */
77 /*                                                                        */
78 /*    DATE              NAME                      DESCRIPTION             */
79 /*                                                                        */
80 /*  05-19-2020     Chaoqiong Xiao           Initial Version 6.0           */
81 /*  09-30-2020     Chaoqiong Xiao           Modified comment(s),          */
82 /*                                            resulting in version 6.1    */
83 /*                                                                        */
84 /**************************************************************************/
_ux_host_stack_configuration_enumerate(UX_DEVICE * device)85 UINT  _ux_host_stack_configuration_enumerate(UX_DEVICE *device)
86 {
87 
88 UX_TRANSFER         *transfer_request;
89 UINT                status =  UX_ERROR;
90 UCHAR *             descriptor;
91 UX_ENDPOINT         *control_endpoint;
92 UX_CONFIGURATION    *configuration;
93 ULONG               nb_configurations;
94 ULONG               configuration_index;
95 
96     /* If trace is enabled, insert this event into the trace buffer.  */
97     UX_TRACE_IN_LINE_INSERT(UX_TRACE_HOST_STACK_CONFIGURATION_ENUMERATE, device, 0, 0, 0, UX_TRACE_HOST_STACK_EVENTS, 0, 0)
98 
99     /* Retrieve the pointer to the control endpoint and its transfer_request.  */
100     control_endpoint =  &device -> ux_device_control_endpoint;
101     transfer_request =  &control_endpoint -> ux_endpoint_transfer_request;
102 
103     /* Need to allocate memory for the configuration descriptor the first time we read
104        only the configuration descriptor when we have the configuration descriptor, we have
105        the length of the entire configuration\interface\endpoint descriptors.  */
106     descriptor =  _ux_utility_memory_allocate(UX_SAFE_ALIGN, UX_CACHE_SAFE_MEMORY, UX_CONFIGURATION_DESCRIPTOR_LENGTH);
107     if (descriptor == UX_NULL)
108         return(UX_MEMORY_INSUFFICIENT);
109 
110     /* There maybe multiple configurations for this device.  */
111     nb_configurations =  device -> ux_device_descriptor.bNumConfigurations;
112 
113     /* Parse all the configurations attached to the device. We start with the first index.
114        The index and the actual configuration value may be different according to the USB specification!  */
115     for (configuration_index = 0; configuration_index < nb_configurations; configuration_index++)
116     {
117 
118         /* Create a transfer_request for the GET_DESCRIPTOR request.  */
119         transfer_request -> ux_transfer_request_data_pointer =      descriptor;
120         transfer_request -> ux_transfer_request_requested_length =  UX_CONFIGURATION_DESCRIPTOR_LENGTH;
121         transfer_request -> ux_transfer_request_function =          UX_GET_DESCRIPTOR;
122         transfer_request -> ux_transfer_request_type =              UX_REQUEST_IN | UX_REQUEST_TYPE_STANDARD | UX_REQUEST_TARGET_DEVICE;
123         transfer_request -> ux_transfer_request_value =             configuration_index | (UINT)(UX_CONFIGURATION_DESCRIPTOR_ITEM << 8);
124         transfer_request -> ux_transfer_request_index =             0;
125 
126         /* Send request to HCD layer.  */
127         status =  _ux_host_stack_transfer_request(transfer_request);
128 
129         /* Check for correct transfer and entire descriptor returned.  */
130         if ((status == UX_SUCCESS) && (transfer_request -> ux_transfer_request_actual_length == UX_CONFIGURATION_DESCRIPTOR_LENGTH))
131         {
132 
133             /* Allocate some memory for the container of this descriptor.  */
134             configuration =  _ux_utility_memory_allocate(UX_NO_ALIGN, UX_REGULAR_MEMORY, sizeof(UX_CONFIGURATION));
135 
136             /* Check to see if the block was allocated.  */
137             if (configuration != UX_NULL)
138             {
139 
140                 /* This configuration must be linked to the device.  */
141                 _ux_host_stack_new_configuration_create(device, configuration);
142 
143                 /* The descriptor is in a packed format, parse it locally.  */
144                 _ux_utility_descriptor_parse(descriptor, _ux_system_configuration_descriptor_structure,
145                                     UX_CONFIGURATION_DESCRIPTOR_ENTRIES, (UCHAR *) &configuration -> ux_configuration_descriptor);
146 
147                 /* Parse the device descriptor so that we can retrieve the length
148                     of the entire configuration.  */
149                 status =  _ux_host_stack_configuration_descriptor_parse(device, configuration, configuration_index);
150 
151                 /* Check the completion status.  */
152                 if (status != UX_SUCCESS)
153                 {
154                     /* Error, delete the configuration instance.  */
155                     _ux_host_stack_configuration_instance_delete(configuration);
156                 }
157             }
158             else
159             {
160 
161                 /* Cannot allocate configuration memory. Abort enumeration */
162                 status =  UX_MEMORY_INSUFFICIENT;
163 
164                 break;
165             }
166         }
167         else
168         {
169 
170             /* Error trap. */
171             _ux_system_error_handler(UX_SYSTEM_LEVEL_THREAD, UX_SYSTEM_CONTEXT_ENUMERATOR, UX_DESCRIPTOR_CORRUPTED);
172 
173             /* If trace is enabled, insert this event into the trace buffer.  */
174             UX_TRACE_IN_LINE_INSERT(UX_TRACE_ERROR, UX_DESCRIPTOR_CORRUPTED, descriptor, 0, 0, UX_TRACE_ERRORS, 0, 0)
175 
176             /* The device descriptor does not contain the right amount of data. Maybe corruption.  */
177             status =  UX_DESCRIPTOR_CORRUPTED;
178 
179             break;
180         }
181 
182     }
183 
184     /* Free all used resources.  */
185     _ux_utility_memory_free(descriptor);
186 
187     /* Return completion status.  */
188     return(status);
189 }
190 
191