Adjust endpoint polling intervals for HID demos, to make them compliant with the...
[pub/lufa.git] / LUFA / Drivers / USB / LowLevel / Pipe.c
1 /*
2 LUFA Library
3 Copyright (C) Dean Camera, 2009.
4
5 dean [at] fourwalledcubicle [dot] com
6 www.fourwalledcubicle.com
7 */
8
9 /*
10 Copyright 2009 Dean Camera (dean [at] fourwalledcubicle [dot] com)
11
12 Permission to use, copy, modify, and distribute this software
13 and its documentation for any purpose and without fee is hereby
14 granted, provided that the above copyright notice appear in all
15 copies and that both that the copyright notice and this
16 permission notice and warranty disclaimer appear in supporting
17 documentation, and that the name of the author not be used in
18 advertising or publicity pertaining to distribution of the
19 software without specific, written prior permission.
20
21 The author disclaim all warranties with regard to this
22 software, including all implied warranties of merchantability
23 and fitness. In no event shall the author be liable for any
24 special, indirect or consequential damages or any damages
25 whatsoever resulting from loss of use, data or profits, whether
26 in an action of contract, negligence or other tortious action,
27 arising out of or in connection with the use or performance of
28 this software.
29 */
30
31 #include "../HighLevel/USBMode.h"
32
33 #if defined(USB_CAN_BE_HOST)
34
35 #define INCLUDE_FROM_PIPE_C
36 #include "Pipe.h"
37
38 uint8_t USB_ControlPipeSize = PIPE_CONTROLPIPE_DEFAULT_SIZE;
39
40 bool Pipe_ConfigurePipe(const uint8_t Number, const uint8_t Type, const uint8_t Token, const uint8_t EndpointNumber,
41 const uint16_t Size, const uint8_t Banks)
42 {
43 Pipe_SelectPipe(Number);
44 Pipe_EnablePipe();
45
46 UPCFG1X = 0;
47
48 UPCFG0X = ((Type << EPTYPE0) | Token | ((EndpointNumber & PIPE_EPNUM_MASK) << PEPNUM0));
49 UPCFG1X = ((1 << ALLOC) | Banks | Pipe_BytesToEPSizeMask(Size));
50
51 Pipe_SetInfiniteINRequests();
52
53 return Pipe_IsConfigured();
54 }
55
56 void Pipe_ClearPipes(void)
57 {
58 UPINT = 0;
59
60 for (uint8_t PNum = 0; PNum < PIPE_TOTAL_PIPES; PNum++)
61 {
62 Pipe_ResetPipe(PNum);
63 Pipe_SelectPipe(PNum);
64 UPIENX = 0;
65 UPINTX = 0;
66 Pipe_ClearError();
67 Pipe_ClearErrorFlags();
68 Pipe_DeallocateMemory();
69 Pipe_DisablePipe();
70 }
71 }
72
73 bool Pipe_IsEndpointBound(uint8_t EndpointAddress)
74 {
75 uint8_t PrevPipeNumber = Pipe_GetCurrentPipe();
76
77 for (uint8_t PNum = 0; PNum < PIPE_TOTAL_PIPES; PNum++)
78 {
79 Pipe_SelectPipe(PNum);
80
81 if (Pipe_IsConfigured() && (Pipe_BoundEndpointNumber() == EndpointAddress))
82 return true;
83 }
84
85 Pipe_SelectPipe(PrevPipeNumber);
86 return false;
87 }
88
89 uint8_t Pipe_WaitUntilReady(void)
90 {
91 #if (USB_STREAM_TIMEOUT_MS < 0xFF)
92 uint8_t TimeoutMSRem = USB_STREAM_TIMEOUT_MS;
93 #else
94 uint16_t TimeoutMSRem = USB_STREAM_TIMEOUT_MS;
95 #endif
96
97 for (;;)
98 {
99 if (Pipe_GetPipeToken() == PIPE_TOKEN_IN)
100 {
101 if (Pipe_IsINReceived())
102 return PIPE_READYWAIT_NoError;
103 }
104 else
105 {
106 if (Pipe_IsOUTReady())
107 return PIPE_READYWAIT_NoError;
108 }
109
110 if (Pipe_IsStalled())
111 return PIPE_READYWAIT_PipeStalled;
112 else if (USB_HostState == HOST_STATE_Unattached)
113 return PIPE_READYWAIT_DeviceDisconnected;
114
115 if (USB_INT_HasOccurred(USB_INT_HSOFI))
116 {
117 USB_INT_Clear(USB_INT_HSOFI);
118
119 if (!(TimeoutMSRem--))
120 return PIPE_READYWAIT_Timeout;
121 }
122 }
123 }
124
125 uint8_t Pipe_Discard_Stream(uint16_t Length
126 #if !defined(NO_STREAM_CALLBACKS)
127 , StreamCallbackPtr_t Callback
128 #endif
129 )
130 {
131 uint8_t ErrorCode;
132
133 Pipe_SetToken(PIPE_TOKEN_IN);
134
135 if ((ErrorCode = Pipe_WaitUntilReady()))
136 return ErrorCode;
137
138 #if defined(FAST_STREAM_TRANSFERS)
139 uint8_t BytesRemToAlignment = (Pipe_BytesInPipe() & 0x07);
140
141 if (Length >= 8)
142 {
143 Length -= BytesRemToAlignment;
144
145 switch (BytesRemToAlignment)
146 {
147 default:
148 do
149 {
150 if (!(Pipe_IsReadWriteAllowed()))
151 {
152 Pipe_ClearIN();
153
154 #if !defined(NO_STREAM_CALLBACKS)
155 if ((Callback != NULL) && (Callback() == STREAMCALLBACK_Abort))
156 return PIPE_RWSTREAM_CallbackAborted;
157 #endif
158
159 if ((ErrorCode = Pipe_WaitUntilReady()))
160 return ErrorCode;
161 }
162
163 Length -= 8;
164
165 Pipe_Discard_Byte();
166 case 7: Pipe_Discard_Byte();
167 case 6: Pipe_Discard_Byte();
168 case 5: Pipe_Discard_Byte();
169 case 4: Pipe_Discard_Byte();
170 case 3: Pipe_Discard_Byte();
171 case 2: Pipe_Discard_Byte();
172 case 1: Pipe_Discard_Byte();
173 } while (Length >= 8);
174 }
175 }
176 #endif
177
178 while (Length)
179 {
180 if (!(Pipe_IsReadWriteAllowed()))
181 {
182 Pipe_ClearIN();
183
184 #if !defined(NO_STREAM_CALLBACKS)
185 if ((Callback != NULL) && (Callback() == STREAMCALLBACK_Abort))
186 return PIPE_RWSTREAM_CallbackAborted;
187 #endif
188
189 if ((ErrorCode = Pipe_WaitUntilReady()))
190 return ErrorCode;
191 }
192 else
193 {
194 Pipe_Discard_Byte();
195 Length--;
196 }
197 }
198
199 return PIPE_RWSTREAM_NoError;
200 }
201
202 #define TEMPLATE_FUNC_NAME Pipe_Write_Stream_LE
203 #define TEMPLATE_TOKEN PIPE_TOKEN_OUT
204 #define TEMPLATE_CLEAR_PIPE() Pipe_ClearOUT()
205 #define TEMPLATE_BUFFER_OFFSET(Length) 0
206 #define TEMPLATE_TRANSFER_BYTE(BufferPtr) Pipe_Write_Byte(*(BufferPtr++))
207 #include "Template/Template_Pipe_RW.c"
208
209 #define TEMPLATE_FUNC_NAME Pipe_Write_PStream_LE
210 #define TEMPLATE_TOKEN PIPE_TOKEN_OUT
211 #define TEMPLATE_CLEAR_PIPE() Pipe_ClearOUT()
212 #define TEMPLATE_BUFFER_OFFSET(Length) 0
213 #define TEMPLATE_TRANSFER_BYTE(BufferPtr) Pipe_Write_Byte(pgm_read_byte(BufferPtr++))
214 #include "Template/Template_Pipe_RW.c"
215
216 #define TEMPLATE_FUNC_NAME Pipe_Write_EStream_LE
217 #define TEMPLATE_TOKEN PIPE_TOKEN_OUT
218 #define TEMPLATE_CLEAR_PIPE() Pipe_ClearOUT()
219 #define TEMPLATE_BUFFER_OFFSET(Length) 0
220 #define TEMPLATE_TRANSFER_BYTE(BufferPtr) Pipe_Write_Byte(eeprom_read_byte(BufferPtr++))
221 #include "Template/Template_Pipe_RW.c"
222
223 #define TEMPLATE_FUNC_NAME Pipe_Write_Stream_BE
224 #define TEMPLATE_TOKEN PIPE_TOKEN_OUT
225 #define TEMPLATE_CLEAR_PIPE() Pipe_ClearOUT()
226 #define TEMPLATE_BUFFER_OFFSET(Length) Length - 1
227 #define TEMPLATE_TRANSFER_BYTE(BufferPtr) Pipe_Write_Byte(*(BufferPtr--))
228 #include "Template/Template_Pipe_RW.c"
229
230 #define TEMPLATE_FUNC_NAME Pipe_Write_PStream_BE
231 #define TEMPLATE_TOKEN PIPE_TOKEN_OUT
232 #define TEMPLATE_CLEAR_PIPE() Pipe_ClearOUT()
233 #define TEMPLATE_BUFFER_OFFSET(Length) Length - 1
234 #define TEMPLATE_TRANSFER_BYTE(BufferPtr) Pipe_Write_Byte(pgm_read_byte(BufferPtr--))
235 #include "Template/Template_Pipe_RW.c"
236
237 #define TEMPLATE_FUNC_NAME Pipe_Write_EStream_BE
238 #define TEMPLATE_TOKEN PIPE_TOKEN_OUT
239 #define TEMPLATE_CLEAR_PIPE() Pipe_ClearOUT()
240 #define TEMPLATE_BUFFER_OFFSET(Length) Length - 1
241 #define TEMPLATE_TRANSFER_BYTE(BufferPtr) Pipe_Write_Byte(eeprom_read_byte(BufferPtr--))
242 #include "Template/Template_Pipe_RW.c"
243
244 #define TEMPLATE_FUNC_NAME Pipe_Read_Stream_LE
245 #define TEMPLATE_TOKEN PIPE_TOKEN_IN
246 #define TEMPLATE_CLEAR_PIPE() Pipe_ClearIN()
247 #define TEMPLATE_BUFFER_OFFSET(Length) 0
248 #define TEMPLATE_TRANSFER_BYTE(BufferPtr) *(BufferPtr++) = Pipe_Read_Byte()
249 #include "Template/Template_Pipe_RW.c"
250
251 #define TEMPLATE_FUNC_NAME Pipe_Read_EStream_LE
252 #define TEMPLATE_TOKEN PIPE_TOKEN_IN
253 #define TEMPLATE_CLEAR_PIPE() Pipe_ClearIN()
254 #define TEMPLATE_BUFFER_OFFSET(Length) 0
255 #define TEMPLATE_TRANSFER_BYTE(BufferPtr) eeprom_write_byte(BufferPtr++, Pipe_Read_Byte())
256 #include "Template/Template_Pipe_RW.c"
257
258 #define TEMPLATE_FUNC_NAME Pipe_Read_Stream_BE
259 #define TEMPLATE_TOKEN PIPE_TOKEN_IN
260 #define TEMPLATE_CLEAR_PIPE() Pipe_ClearIN()
261 #define TEMPLATE_BUFFER_OFFSET(Length) Length - 1
262 #define TEMPLATE_TRANSFER_BYTE(BufferPtr) *(BufferPtr--) = Pipe_Read_Byte()
263 #include "Template/Template_Pipe_RW.c"
264
265 #define TEMPLATE_FUNC_NAME Pipe_Read_EStream_BE
266 #define TEMPLATE_TOKEN PIPE_TOKEN_IN
267 #define TEMPLATE_CLEAR_PIPE() Pipe_ClearIN()
268 #define TEMPLATE_BUFFER_OFFSET(Length) Length - 1
269 #define TEMPLATE_TRANSFER_BYTE(BufferPtr) eeprom_write_byte(BufferPtr--, Pipe_Read_Byte())
270 #include "Template/Template_Pipe_RW.c"
271
272 #endif