3 Copyright (C) Dean Camera, 2009.
5 dean [at] fourwalledcubicle [dot] com
6 www.fourwalledcubicle.com
10 Copyright 2009 Dean Camera (dean [at] fourwalledcubicle [dot] com)
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.
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
31 #include "../HighLevel/USBMode.h"
33 #if defined(USB_CAN_BE_DEVICE)
35 #define INCLUDE_FROM_DEVCHAPTER9_C
36 #include "DevChapter9.h"
38 uint8_t USB_ConfigurationNumber
;
39 bool USB_RemoteWakeupEnabled
;
40 bool USB_CurrentlySelfPowered
;
42 void USB_Device_ProcessControlPacket(void)
44 bool RequestHandled
= false;
45 uint8_t* RequestHeader
= (uint8_t*)&USB_ControlRequest
;
47 for (uint8_t RequestHeaderByte
= 0; RequestHeaderByte
< sizeof(USB_Request_Header_t
); RequestHeaderByte
++)
48 *(RequestHeader
++) = Endpoint_Read_Byte();
50 switch (USB_ControlRequest
.bRequest
)
53 if ((USB_ControlRequest
.bmRequestType
== (REQDIR_DEVICETOHOST
| REQTYPE_STANDARD
| REQREC_DEVICE
)) ||
54 (USB_ControlRequest
.bmRequestType
== (REQDIR_DEVICETOHOST
| REQTYPE_STANDARD
| REQREC_ENDPOINT
)))
56 USB_Device_GetStatus();
57 RequestHandled
= true;
61 #if !defined(FEATURELESS_CONTROL_ONLY_DEVICE)
62 case REQ_ClearFeature
:
64 if (USB_ControlRequest
.bmRequestType
== (REQDIR_HOSTTODEVICE
| REQTYPE_STANDARD
| REQREC_ENDPOINT
))
66 USB_Device_ClearSetFeature();
67 RequestHandled
= true;
73 if (USB_ControlRequest
.bmRequestType
== (REQDIR_HOSTTODEVICE
| REQTYPE_STANDARD
| REQREC_DEVICE
))
75 USB_Device_SetAddress();
76 RequestHandled
= true;
80 case REQ_GetDescriptor
:
81 if ((USB_ControlRequest
.bmRequestType
== (REQDIR_DEVICETOHOST
| REQTYPE_STANDARD
| REQREC_DEVICE
)) ||
82 (USB_ControlRequest
.bmRequestType
== (REQDIR_DEVICETOHOST
| REQTYPE_STANDARD
| REQREC_INTERFACE
)))
84 USB_Device_GetDescriptor();
85 RequestHandled
= true;
89 case REQ_GetConfiguration
:
90 if (USB_ControlRequest
.bmRequestType
== (REQDIR_DEVICETOHOST
| REQTYPE_STANDARD
| REQREC_DEVICE
))
92 USB_Device_GetConfiguration();
93 RequestHandled
= true;
97 case REQ_SetConfiguration
:
98 if (USB_ControlRequest
.bmRequestType
== (REQDIR_HOSTTODEVICE
| REQTYPE_STANDARD
| REQREC_DEVICE
))
100 USB_Device_SetConfiguration();
101 RequestHandled
= true;
107 if (!(RequestHandled
))
108 RAISE_EVENT(USB_UnhandledControlPacket
);
110 if (Endpoint_IsSETUPReceived())
112 Endpoint_StallTransaction();
113 Endpoint_ClearSETUP();
117 static void USB_Device_SetAddress(void)
119 Endpoint_ClearSETUP();
121 while (!(Endpoint_IsINReady()));
125 while (!(Endpoint_IsINReady()));
127 UDADDR
= ((1 << ADDEN
) | (USB_ControlRequest
.wValue
& 0x7F));
132 static void USB_Device_SetConfiguration(void)
134 bool AlreadyConfigured
= (USB_ConfigurationNumber
!= 0);
136 #if defined(USE_SINGLE_DEVICE_CONFIGURATION)
137 if (USB_ControlRequest
.wValue
> 1)
139 USB_Descriptor_Device_t
* DevDescriptorPtr
;
141 if ((USB_GetDescriptor((DTYPE_Device
<< 8), 0, (void*)&DevDescriptorPtr
) == NO_DESCRIPTOR
) ||
142 #if defined(USE_RAM_DESCRIPTORS)
143 (USB_ControlRequest
.wValue
> DevDescriptorPtr
->NumberOfConfigurations
))
144 #elif defined (USE_EEPROM_DESCRIPTORS)
145 (USB_ControlRequest
.wValue
> eeprom_read_byte(&DevDescriptorPtr
->NumberOfConfigurations
)))
147 (USB_ControlRequest
.wValue
> pgm_read_byte(&DevDescriptorPtr
->NumberOfConfigurations
)))
154 Endpoint_ClearSETUP();
156 USB_ConfigurationNumber
= USB_ControlRequest
.wValue
;
160 if (!(AlreadyConfigured
) && USB_ConfigurationNumber
)
161 RAISE_EVENT(USB_DeviceEnumerationComplete
);
163 RAISE_EVENT(USB_ConfigurationChanged
);
166 void USB_Device_GetConfiguration(void)
168 Endpoint_ClearSETUP();
170 Endpoint_Write_Byte(USB_ConfigurationNumber
);
174 while (!(Endpoint_IsOUTReceived()));
178 static void USB_Device_GetDescriptor(void)
180 void* DescriptorPointer
;
181 uint16_t DescriptorSize
;
183 if ((DescriptorSize
= USB_GetDescriptor(USB_ControlRequest
.wValue
, USB_ControlRequest
.wIndex
, &DescriptorPointer
)) == NO_DESCRIPTOR
)
186 Endpoint_ClearSETUP();
188 #if defined(USE_RAM_DESCRIPTORS)
189 Endpoint_Write_Control_Stream_LE(DescriptorPointer
, DescriptorSize
);
193 if (USB_ControlRequest
.wLength
> DescriptorSize
)
194 USB_ControlRequest
.wLength
= DescriptorSize
;
196 while (USB_ControlRequest
.wLength
)
198 while (!(Endpoint_IsINReady()))
200 if (Endpoint_IsOUTReceived())
207 while (USB_ControlRequest
.wLength
&& (Endpoint_BytesInEndpoint() < USB_ControlEndpointSize
))
209 #if defined (USE_EEPROM_DESCRIPTORS)
210 Endpoint_Write_Byte(eeprom_read_byte(DescriptorPointer
++));
212 Endpoint_Write_Byte(pgm_read_byte(DescriptorPointer
++));
215 USB_ControlRequest
.wLength
--;
218 SendZLP
= (Endpoint_BytesInEndpoint() == USB_ControlEndpointSize
);
224 while (!(Endpoint_IsINReady()));
228 while (!(Endpoint_IsOUTReceived()));
234 static void USB_Device_GetStatus(void)
236 uint8_t CurrentStatus
= 0;
238 switch (USB_ControlRequest
.bmRequestType
)
240 case (REQDIR_DEVICETOHOST
| REQTYPE_STANDARD
| REQREC_DEVICE
):
241 if (USB_CurrentlySelfPowered
)
242 CurrentStatus
|= FEATURE_SELFPOWERED_ENABLED
;
244 if (USB_RemoteWakeupEnabled
)
245 CurrentStatus
|= FEATURE_REMOTE_WAKEUP_ENABLED
;
248 #if !defined(FEATURELESS_CONTROL_ONLY_DEVICE)
249 case (REQDIR_DEVICETOHOST
| REQTYPE_STANDARD
| REQREC_ENDPOINT
):
250 Endpoint_SelectEndpoint(USB_ControlRequest
.wIndex
);
252 CurrentStatus
= Endpoint_IsStalled();
254 Endpoint_SelectEndpoint(ENDPOINT_CONTROLEP
);
260 Endpoint_ClearSETUP();
262 Endpoint_Write_Word_LE(CurrentStatus
);
266 while (!(Endpoint_IsOUTReceived()));
270 #if !defined(FEATURELESS_CONTROL_ONLY_DEVICE)
271 static void USB_Device_ClearSetFeature(void)
273 switch (USB_ControlRequest
.bmRequestType
& CONTROL_REQTYPE_RECIPIENT
)
275 case REQREC_ENDPOINT
:
276 if (USB_ControlRequest
.wValue
== FEATURE_ENDPOINT_HALT
)
278 uint8_t EndpointIndex
= (USB_ControlRequest
.wIndex
& ENDPOINT_EPNUM_MASK
);
280 if (EndpointIndex
!= ENDPOINT_CONTROLEP
)
282 Endpoint_SelectEndpoint(EndpointIndex
);
284 if (Endpoint_IsEnabled())
286 if (USB_ControlRequest
.bRequest
== REQ_ClearFeature
)
288 Endpoint_ClearStall();
289 Endpoint_ResetFIFO(EndpointIndex
);
290 Endpoint_ResetDataToggle();
294 Endpoint_StallTransaction();
298 Endpoint_SelectEndpoint(ENDPOINT_CONTROLEP
);
299 Endpoint_ClearSETUP();