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