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_interface_setting_select             PORTABLE C      */
36 /*                                                           6.1.12       */
37 /*  AUTHOR                                                                */
38 /*                                                                        */
39 /*    Chaoqiong Xiao, Microsoft Corporation                               */
40 /*                                                                        */
41 /*  DESCRIPTION                                                           */
42 /*                                                                        */
43 /*    This function selects a specific alternate setting for a given      */
44 /*    interface belonging to the selected configuration.                  */
45 /*                                                                        */
46 /*  INPUT                                                                 */
47 /*                                                                        */
48 /*    interface                             Pointer to interface          */
49 /*                                                                        */
50 /*  OUTPUT                                                                */
51 /*                                                                        */
52 /*    Completion Status                                                   */
53 /*                                                                        */
54 /*  CALLS                                                                 */
55 /*                                                                        */
56 /*    _ux_host_stack_interface_instance_create  Create interface instance */
57 /*    _ux_host_stack_interface_instance_delete  Delete interface instance */
58 /*    _ux_host_stack_interface_set              Set interface instance    */
59 /*    _ux_host_semaphore_get                    Get semaphore             */
60 /*    _ux_host_semaphore_put                    Put semaphore             */
61 /*                                                                        */
62 /*  CALLED BY                                                             */
63 /*                                                                        */
64 /*    Application                                                         */
65 /*    USBX Components                                                     */
66 /*                                                                        */
67 /*  RELEASE HISTORY                                                       */
68 /*                                                                        */
69 /*    DATE              NAME                      DESCRIPTION             */
70 /*                                                                        */
71 /*  05-19-2020     Chaoqiong Xiao           Initial Version 6.0           */
72 /*  09-30-2020     Chaoqiong Xiao           Modified comment(s),          */
73 /*                                            resulting in version 6.1    */
74 /*  07-29-2022     Chaoqiong Xiao           Modified comment(s),          */
75 /*                                            protected control request,  */
76 /*                                            resulting in version 6.1.12 */
77 /*                                                                        */
78 /**************************************************************************/
_ux_host_stack_interface_setting_select(UX_INTERFACE * interface_ptr)79 UINT  _ux_host_stack_interface_setting_select(UX_INTERFACE *interface_ptr)
80 {
81 
82 UX_CONFIGURATION    *configuration;
83 UX_INTERFACE        *current_interface;
84 UX_INTERFACE        *previous_interface;
85 UX_INTERFACE        *main_interface;
86 UINT                current_interface_number;
87 UINT                current_alternate_setting;
88 UINT                status;
89 
90     /* If trace is enabled, insert this event into the trace buffer.  */
91     UX_TRACE_IN_LINE_INSERT(UX_TRACE_HOST_STACK_INTERFACE_SETTING_SELECT, interface_ptr, 0, 0, 0, UX_TRACE_HOST_STACK_EVENTS, 0, 0)
92 
93     /* Check this alternate setting container. It must be valid before
94        we continue.  */
95     if (interface_ptr -> ux_interface_handle != (ULONG) (ALIGN_TYPE) interface_ptr)
96     {
97         /* If trace is enabled, insert this event into the trace buffer.  */
98         UX_TRACE_IN_LINE_INSERT(UX_TRACE_ERROR, UX_INTERFACE_HANDLE_UNKNOWN, interface_ptr, 0, 0, UX_TRACE_ERRORS, 0, 0)
99 
100         return(UX_INTERFACE_HANDLE_UNKNOWN);
101     }
102 
103     /* From the interface, get the configuration container and the first
104        interface hooked to this configuration.  */
105     configuration =             interface_ptr -> ux_interface_configuration;
106     current_interface_number =  interface_ptr -> ux_interface_descriptor.bInterfaceNumber;
107     current_interface =         configuration -> ux_configuration_first_interface;
108 
109     /* Remember the main interface to store the next alternate setting.  We set the main interface
110        to the first interface to keep the compiler happy. */
111     main_interface =  current_interface;
112 
113     /* In order to keep the compiler happy, we reset the alternate setting.  */
114     current_alternate_setting =  0;
115     previous_interface = UX_NULL;
116 
117     /* Search for the current alternate setting for this interface.
118        All its endpoints will need to be destroyed.
119        Since interfaces in the parent configuration should be linked together
120        in correct way, just find it in while loop.
121      */
122     while (1)
123     {
124 
125         /* Try to locate the first interface container in the interface chain which
126            has the same interface number as the one supplied by the caller.  */
127         if (current_interface -> ux_interface_descriptor.bInterfaceNumber == current_interface_number)
128         {
129 
130             /* The alternate setting 0 of this interface has the current selected
131                alternate setting.  */
132             if (current_interface -> ux_interface_descriptor.bAlternateSetting == 0)
133             {
134 
135                 /* Set the alternate setting.  */
136                 current_alternate_setting =  current_interface -> ux_interface_current_alternate_setting;
137 
138                 /* Remember the main interface to store the next alternate setting.  */
139                 main_interface =  current_interface;
140 
141             }
142 
143             /* See if the current alternate setting matches that of the interface alternate setting.  */
144             if (current_alternate_setting == current_interface -> ux_interface_descriptor.bAlternateSetting)
145             {
146 
147                 /* Yes, save the current alternate setting.  */
148                 previous_interface = current_interface;
149 
150                 /* Then delete the current interface.  */
151                 _ux_host_stack_interface_instance_delete(current_interface);
152 
153                 /* We are done in this loop.  */
154                 break;
155 
156             }
157         }
158 
159         /* Move to the next interface. */
160         current_interface =  current_interface -> ux_interface_next_interface;
161     }
162 
163     /* Remember the new alternate setting.  */
164     main_interface -> ux_interface_current_alternate_setting =  interface_ptr -> ux_interface_descriptor.bAlternateSetting;
165 
166     /* Now, the interface must be created with the new alternate setting.  */
167     status =  _ux_host_stack_interface_instance_create(interface_ptr);
168 
169     /* If we could not create it, we return to the default one.  */
170     if (status != UX_SUCCESS)
171     {
172 
173         /* Then delete the failed interface.  */
174         _ux_host_stack_interface_instance_delete(interface_ptr);
175 
176         /* Error, reset the main interface alternate setting to the default.  */
177         main_interface -> ux_interface_current_alternate_setting =  current_alternate_setting;
178 
179         /* Re-create the previous interface with the old alternate setting.  */
180         _ux_host_stack_interface_instance_create(previous_interface);
181 
182         /* Return error status.  */
183         return(status);
184     }
185 
186     /* Protect the control endpoint semaphore here.  It will be unprotected in the
187        transfer request function.  */
188     status =  _ux_host_semaphore_get(&configuration -> ux_configuration_device -> ux_device_protection_semaphore, UX_WAIT_FOREVER);
189     if (status != UX_SUCCESS)
190         return(status);
191 
192     /* Issue a SET_INTERFACE command to the target device.  */
193     status =  _ux_host_stack_interface_set(interface_ptr);
194 
195     /* Check completion status.  */
196     if (status != UX_SUCCESS)
197     {
198 
199         /* Error, reset the main interface alternate setting to the default.  */
200         main_interface -> ux_interface_current_alternate_setting =  current_alternate_setting;
201 
202         /* Delete the current interface.  */
203         _ux_host_stack_interface_instance_delete(interface_ptr);
204 
205         /* Re-create the previous interface with the old alternate setting.  */
206         _ux_host_stack_interface_instance_create(previous_interface);
207 
208         /* Return error status.  */
209         _ux_host_semaphore_put(&configuration -> ux_configuration_device -> ux_device_protection_semaphore);
210         return(status);
211     }
212 
213     /* Return to caller.  */
214     return(status);
215 }
216 
217 
218 /**************************************************************************/
219 /*                                                                        */
220 /*  FUNCTION                                               RELEASE        */
221 /*                                                                        */
222 /*    _uxe_host_stack_interface_setting_select            PORTABLE C      */
223 /*                                                           6.3.0        */
224 /*  AUTHOR                                                                */
225 /*                                                                        */
226 /*    Chaoqiong Xiao, Microsoft Corporation                               */
227 /*                                                                        */
228 /*  DESCRIPTION                                                           */
229 /*                                                                        */
230 /*    This function checks errors in host stack interface select function */
231 /*    call.                                                               */
232 /*                                                                        */
233 /*  INPUT                                                                 */
234 /*                                                                        */
235 /*    interface_ptr                         Pointer to interface          */
236 /*                                                                        */
237 /*  OUTPUT                                                                */
238 /*                                                                        */
239 /*    None                                                                */
240 /*                                                                        */
241 /*  CALLS                                                                 */
242 /*                                                                        */
243 /*    _ux_host_stack_interface_setting_select                             */
244 /*                                          Interface setting select      */
245 /*                                                                        */
246 /*  CALLED BY                                                             */
247 /*                                                                        */
248 /*    Application                                                         */
249 /*                                                                        */
250 /*  RELEASE HISTORY                                                       */
251 /*                                                                        */
252 /*    DATE              NAME                      DESCRIPTION             */
253 /*                                                                        */
254 /*  10-31-2023     Chaoqiong Xiao           Initial Version 6.3.0         */
255 /*                                                                        */
256 /**************************************************************************/
_uxe_host_stack_interface_setting_select(UX_INTERFACE * interface_ptr)257 UINT  _uxe_host_stack_interface_setting_select(UX_INTERFACE *interface_ptr)
258 {
259 
260     /* Sanity check.  */
261     if (interface_ptr == UX_NULL)
262         return(UX_INVALID_PARAMETER);
263 
264     /* Invoke interface setting select function.  */
265     return(_ux_host_stack_interface_setting_select(interface_ptr));
266 }
267