1 /**************************************************************************/
2 /*                                                                        */
3 /*       Copyright (c) Microsoft Corporation. All rights reserved.        */
4 /*                                                                        */
5 /*       This software is licensed under the Microsoft Software License   */
6 /*       Terms for Microsoft Azure RTOS. Full text of the license can be  */
7 /*       found in the LICENSE file at https://aka.ms/AzureRTOS_EULA       */
8 /*       and in the root directory of this software.                      */
9 /*                                                                        */
10 /**************************************************************************/
11 
12 /**************************************************************************/
13 /**************************************************************************/
14 /**                                                                       */
15 /** USBX Component                                                        */
16 /**                                                                       */
17 /**   Device DPUMP Class                                                  */
18 /**                                                                       */
19 /**************************************************************************/
20 /**************************************************************************/
21 
22 #define UX_SOURCE_CODE
23 
24 
25 /* Include necessary system files.  */
26 
27 #include "ux_api.h"
28 #include "ux_device_class_dpump.h"
29 #include "ux_device_stack.h"
30 
31 
32 /**************************************************************************/
33 /*                                                                        */
34 /*  FUNCTION                                               RELEASE        */
35 /*                                                                        */
36 /*    _ux_device_class_dpump_read                         PORTABLE C      */
37 /*                                                           6.1          */
38 /*  AUTHOR                                                                */
39 /*                                                                        */
40 /*    Chaoqiong Xiao, Microsoft Corporation                               */
41 /*                                                                        */
42 /*  DESCRIPTION                                                           */
43 /*                                                                        */
44 /*    This function reads from the DPUMP class.                           */
45 /*                                                                        */
46 /*  INPUT                                                                 */
47 /*                                                                        */
48 /*    dpump                                   Address of dpump class      */
49 /*                                                instance                */
50 /*                                                                        */
51 /*  OUTPUT                                                                */
52 /*                                                                        */
53 /*    None                                                                */
54 /*                                                                        */
55 /*  CALLS                                                                 */
56 /*                                                                        */
57 /*    _ux_device_stack_transfer_request     Request transfer              */
58 /*    _ux_utility_memory_copy               Copy memory                   */
59 /*                                                                        */
60 /*  CALLED BY                                                             */
61 /*                                                                        */
62 /*    ThreadX                                                             */
63 /*                                                                        */
64 /*  RELEASE HISTORY                                                       */
65 /*                                                                        */
66 /*    DATE              NAME                      DESCRIPTION             */
67 /*                                                                        */
68 /*  05-19-2020     Chaoqiong Xiao           Initial Version 6.0           */
69 /*  09-30-2020     Chaoqiong Xiao           Modified comment(s),          */
70 /*                                            verified memset and memcpy  */
71 /*                                            cases,                      */
72 /*                                            resulting in version 6.1    */
73 /*                                                                        */
74 /**************************************************************************/
_ux_device_class_dpump_read(UX_SLAVE_CLASS_DPUMP * dpump,UCHAR * buffer,ULONG requested_length,ULONG * actual_length)75 UINT _ux_device_class_dpump_read(UX_SLAVE_CLASS_DPUMP *dpump, UCHAR *buffer,
76                                 ULONG requested_length, ULONG *actual_length)
77 {
78 
79 UX_SLAVE_ENDPOINT           *endpoint;
80 UX_SLAVE_DEVICE             *device;
81 UX_SLAVE_TRANSFER           *transfer_request;
82 UINT                        status;
83 ULONG                       local_requested_length;
84 
85     /* If trace is enabled, insert this event into the trace buffer.  */
86     UX_TRACE_IN_LINE_INSERT(UX_TRACE_DEVICE_CLASS_DPUMP_READ, dpump, buffer, requested_length, 0, UX_TRACE_DEVICE_CLASS_EVENTS, 0, 0)
87 
88     /* Get the pointer to the device.  */
89     device =  &_ux_system_slave -> ux_system_slave_device;
90 
91     /* As long as the device is in the CONFIGURED state.  */
92     if (device -> ux_slave_device_state != UX_DEVICE_CONFIGURED)
93     {
94 
95         /* Error trap. */
96         _ux_system_error_handler(UX_SYSTEM_LEVEL_THREAD, UX_SYSTEM_CONTEXT_CLASS, UX_CONFIGURATION_HANDLE_UNKNOWN);
97 
98         /* If trace is enabled, insert this event into the trace buffer.  */
99         UX_TRACE_IN_LINE_INSERT(UX_TRACE_ERROR, UX_CONFIGURATION_HANDLE_UNKNOWN, device, 0, 0, UX_TRACE_ERRORS, 0, 0)
100 
101         /* Cannot proceed with command, the interface is down.  */
102         return(UX_CONFIGURATION_HANDLE_UNKNOWN);
103     }
104 
105     /* Locate the OUT endpoint.  */
106     endpoint =  dpump -> ux_slave_class_dpump_bulkout_endpoint;
107 
108     /* Check endpoint. If NULL, we have not yet received the proper SET_INTERFACE command.  */
109     if (endpoint == UX_NULL)
110     {
111         /* Error trap. */
112         _ux_system_error_handler(UX_SYSTEM_LEVEL_THREAD, UX_SYSTEM_CONTEXT_CLASS, UX_ENDPOINT_HANDLE_UNKNOWN);
113 
114         return(UX_ENDPOINT_HANDLE_UNKNOWN);
115     }
116 
117     /* All DPUMP reading  are on the endpoint OUT, from the host.  */
118     transfer_request =  &endpoint -> ux_slave_endpoint_transfer_request;
119 
120     /* Reset the actual length.  */
121     *actual_length =  0;
122 
123     /* Set return status to SUCCESS to make certain compilers happy.  */
124     status =  UX_SUCCESS;
125 
126     /* Check if we need more transactions.  */
127     while (device -> ux_slave_device_state == UX_DEVICE_CONFIGURED && requested_length != 0)
128     {
129 
130         /* Check if we have enough in the local buffer.  */
131         if (requested_length > UX_SLAVE_REQUEST_DATA_MAX_LENGTH)
132 
133             /* We have too much to transfer.  */
134             local_requested_length = UX_SLAVE_REQUEST_DATA_MAX_LENGTH;
135 
136         else
137 
138             /* We can proceed with the demanded length.  */
139             local_requested_length = requested_length;
140 
141         /* Send the request to the device controller.  */
142         status =  _ux_device_stack_transfer_request(transfer_request, local_requested_length, local_requested_length);
143 
144         /* Check the status */
145         if (status == UX_SUCCESS)
146         {
147 
148             /* We need to copy the buffer locally.  */
149             _ux_utility_memory_copy(buffer, transfer_request -> ux_slave_transfer_request_data_pointer,
150                             local_requested_length); /* Use case of memcpy is verified. */
151 
152             /* Next buffer address.  */
153             buffer += transfer_request -> ux_slave_transfer_request_actual_length;
154 
155             /* Set the length actually received. */
156             *actual_length += transfer_request -> ux_slave_transfer_request_actual_length;
157 
158             /* Decrement what left has to be done.  */
159             requested_length -= transfer_request -> ux_slave_transfer_request_actual_length;
160 
161         }
162         else
163 
164             /* We got an error.  */
165             return(status);
166     }
167 
168     /* Check why we got here, either completion or device was extracted.  */
169     if (device -> ux_slave_device_state != UX_DEVICE_CONFIGURED)
170     {
171 
172         /* Error trap. */
173         _ux_system_error_handler(UX_SYSTEM_LEVEL_THREAD, UX_SYSTEM_CONTEXT_CLASS, UX_TRANSFER_NO_ANSWER);
174 
175         /* If trace is enabled, insert this event into the trace buffer.  */
176         UX_TRACE_IN_LINE_INSERT(UX_TRACE_ERROR, UX_TRANSFER_NO_ANSWER, transfer_request, 0, 0, UX_TRACE_ERRORS, 0, 0)
177 
178         /* Device must have been extracted.  */
179         return (UX_TRANSFER_NO_ANSWER);
180     }
181     else
182 
183         /* Simply return the last transaction result.  */
184         return(status);
185 }
186 
187