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 uint8_t bmRequestType
= USB_ControlRequest
.bmRequestType
;
52 switch (USB_ControlRequest
.bRequest
)
55 if ((bmRequestType
== (REQDIR_DEVICETOHOST
| REQTYPE_STANDARD
| REQREC_DEVICE
)) ||
56 (bmRequestType
== (REQDIR_DEVICETOHOST
| REQTYPE_STANDARD
| REQREC_ENDPOINT
)))
58 USB_Device_GetStatus();
59 RequestHandled
= true;
63 case REQ_ClearFeature
:
65 if ((bmRequestType
== (REQDIR_HOSTTODEVICE
| REQTYPE_STANDARD
| REQREC_DEVICE
)) ||
66 (bmRequestType
== (REQDIR_HOSTTODEVICE
| REQTYPE_STANDARD
| REQREC_ENDPOINT
)))
68 USB_Device_ClearSetFeature();
69 RequestHandled
= true;
74 if (bmRequestType
== (REQDIR_HOSTTODEVICE
| REQTYPE_STANDARD
| REQREC_DEVICE
))
76 USB_Device_SetAddress();
77 RequestHandled
= true;
81 case REQ_GetDescriptor
:
82 if ((bmRequestType
== (REQDIR_DEVICETOHOST
| REQTYPE_STANDARD
| REQREC_DEVICE
)) ||
83 (bmRequestType
== (REQDIR_DEVICETOHOST
| REQTYPE_STANDARD
| REQREC_INTERFACE
)))
85 USB_Device_GetDescriptor();
86 RequestHandled
= true;
90 case REQ_GetConfiguration
:
91 if (bmRequestType
== (REQDIR_DEVICETOHOST
| REQTYPE_STANDARD
| REQREC_DEVICE
))
93 USB_Device_GetConfiguration();
94 RequestHandled
= true;
98 case REQ_SetConfiguration
:
99 if (bmRequestType
== (REQDIR_HOSTTODEVICE
| REQTYPE_STANDARD
| REQREC_DEVICE
))
101 USB_Device_SetConfiguration();
102 RequestHandled
= true;
108 if (!(RequestHandled
))
109 EVENT_USB_UnhandledControlPacket();
111 if (Endpoint_IsSETUPReceived())
113 Endpoint_StallTransaction();
114 Endpoint_ClearSETUP();
118 static void USB_Device_SetAddress(void)
120 uint8_t DeviceAddress
= (USB_ControlRequest
.wValue
& 0x7F);
122 Endpoint_ClearSETUP();
126 while (!(Endpoint_IsINReady()))
128 if (USB_DeviceState
== DEVICE_STATE_Unattached
)
132 UDADDR
= ((1 << ADDEN
) | DeviceAddress
);
135 USB_DeviceState
= DEVICE_STATE_Addressed
;
140 static void USB_Device_SetConfiguration(void)
142 bool AlreadyConfigured
= (USB_ConfigurationNumber
!= 0);
144 #if defined(FIXED_NUM_CONFIGURATIONS)
145 if ((uint8_t)USB_ControlRequest
.wValue
> FIXED_NUM_CONFIGURATIONS
)
148 #if !defined(USE_FLASH_DESCRIPTORS) && !defined(USE_EEPROM_DESCRIPTORS) && !defined(USE_RAM_DESCRIPTORS)
149 uint8_t MemoryAddressSpace
;
152 USB_Descriptor_Device_t
* DevDescriptorPtr
;
154 if (CALLBACK_USB_GetDescriptor((DTYPE_Device
<< 8), 0, (void*)&DevDescriptorPtr
155 #if !defined(USE_FLASH_DESCRIPTORS) && !defined(USE_EEPROM_DESCRIPTORS) && !defined(USE_RAM_DESCRIPTORS)
156 , &MemoryAddressSpace
163 #if defined(USE_RAM_DESCRIPTORS)
164 if ((uint8_t)USB_ControlRequest
.wValue
> DevDescriptorPtr
->NumberOfConfigurations
)
166 #elif defined (USE_EEPROM_DESCRIPTORS)
167 if ((uint8_t)USB_ControlRequest
.wValue
> eeprom_read_byte(&DevDescriptorPtr
->NumberOfConfigurations
))
169 #elif defined (USE_FLASH_DESCRIPTORS)
170 if ((uint8_t)USB_ControlRequest
.wValue
> pgm_read_byte(&DevDescriptorPtr
->NumberOfConfigurations
))
173 if (MemoryAddressSpace
== MEMSPACE_FLASH
)
175 if (((uint8_t)USB_ControlRequest
.wValue
> pgm_read_byte(&DevDescriptorPtr
->NumberOfConfigurations
)))
178 else if (MemoryAddressSpace
== MEMSPACE_EEPROM
)
180 if (((uint8_t)USB_ControlRequest
.wValue
> eeprom_read_byte(&DevDescriptorPtr
->NumberOfConfigurations
)))
185 if ((uint8_t)USB_ControlRequest
.wValue
> DevDescriptorPtr
->NumberOfConfigurations
)
191 Endpoint_ClearSETUP();
193 USB_ConfigurationNumber
= (uint8_t)USB_ControlRequest
.wValue
;
197 if (USB_ConfigurationNumber
)
199 USB_DeviceState
= DEVICE_STATE_Configured
;
201 if (!(AlreadyConfigured
))
202 EVENT_USB_DeviceEnumerationComplete();
206 USB_DeviceState
= DEVICE_STATE_Addressed
;
209 EVENT_USB_ConfigurationChanged();
212 void USB_Device_GetConfiguration(void)
214 Endpoint_ClearSETUP();
216 Endpoint_Write_Byte(USB_ConfigurationNumber
);
220 while (!(Endpoint_IsOUTReceived()))
222 if (USB_DeviceState
== DEVICE_STATE_Unattached
)
229 #if !defined(NO_INTERNAL_SERIAL) && (defined(USB_SERIES_6_AVR) || defined(USB_SERIES_7_AVR))
230 static char USB_Device_NibbleToASCII(uint8_t Nibble
)
232 Nibble
= ((Nibble
& 0x0F) + '0');
233 return (Nibble
> '9') ?
(Nibble
+ ('A' - '9' - 1)) : Nibble
;
236 static void USB_Device_GetInternalSerialDescriptor(void)
240 USB_Descriptor_Header_t Header
;
241 int16_t UnicodeString
[20];
242 } SignatureDescriptor
;
244 #if defined(USE_NONSTANDARD_DESCRIPTOR_NAMES)
245 SignatureDescriptor
.Header
.Size
= sizeof(SignatureDescriptor
);
246 SignatureDescriptor
.Header
.Type
= DTYPE_String
;
248 SignatureDescriptor
.Header
.bLength
= sizeof(SignatureDescriptor
);
249 SignatureDescriptor
.Header
.bDescriptorType
= DTYPE_String
;
252 uint8_t SigReadAddress
= 0x0E;
254 for (uint8_t SerialCharNum
= 0; SerialCharNum
< 20; SerialCharNum
++)
256 uint8_t SerialByte
= boot_signature_byte_get(SigReadAddress
);
258 if (SerialCharNum
& 0x01)
264 SignatureDescriptor
.UnicodeString
[SerialCharNum
] = USB_Device_NibbleToASCII(SerialByte
);
267 Endpoint_ClearSETUP();
268 Endpoint_Write_Control_Stream_LE(&SignatureDescriptor
, sizeof(SignatureDescriptor
));
273 static void USB_Device_GetDescriptor(void)
275 void* DescriptorPointer
;
276 uint16_t DescriptorSize
;
278 #if !defined(USE_FLASH_DESCRIPTORS) && !defined(USE_EEPROM_DESCRIPTORS) && !defined(USE_RAM_DESCRIPTORS)
279 uint8_t DescriptorAddressSpace
;
282 #if !defined(NO_INTERNAL_SERIAL) && (defined(USB_SERIES_6_AVR) || defined(USB_SERIES_7_AVR))
283 if (USB_ControlRequest
.wValue
== ((DTYPE_String
<< 8) | USE_INTERNAL_SERIAL
))
285 USB_Device_GetInternalSerialDescriptor();
290 if ((DescriptorSize
= CALLBACK_USB_GetDescriptor(USB_ControlRequest
.wValue
, USB_ControlRequest
.wIndex
,
292 #if !defined(USE_FLASH_DESCRIPTORS) && !defined(USE_EEPROM_DESCRIPTORS) && !defined(USE_RAM_DESCRIPTORS)
293 , &DescriptorAddressSpace
300 Endpoint_ClearSETUP();
302 #if defined(USE_RAM_DESCRIPTORS)
303 Endpoint_Write_Control_Stream_LE(DescriptorPointer
, DescriptorSize
);
304 #elif defined(USE_EEPROM_DESCRIPTORS)
305 Endpoint_Write_Control_EStream_LE(DescriptorPointer
, DescriptorSize
);
306 #elif defined(USE_FLASH_DESCRIPTORS)
307 Endpoint_Write_Control_PStream_LE(DescriptorPointer
, DescriptorSize
);
309 if (DescriptorAddressSpace
== MEMSPACE_FLASH
)
310 Endpoint_Write_Control_PStream_LE(DescriptorPointer
, DescriptorSize
);
311 else if (DescriptorAddressSpace
== MEMSPACE_EEPROM
)
312 Endpoint_Write_Control_EStream_LE(DescriptorPointer
, DescriptorSize
);
314 Endpoint_Write_Control_Stream_LE(DescriptorPointer
, DescriptorSize
);
320 static void USB_Device_GetStatus(void)
322 uint8_t CurrentStatus
= 0;
324 switch (USB_ControlRequest
.bmRequestType
)
326 case (REQDIR_DEVICETOHOST
| REQTYPE_STANDARD
| REQREC_DEVICE
):
327 if (USB_CurrentlySelfPowered
)
328 CurrentStatus
|= FEATURE_SELFPOWERED_ENABLED
;
330 if (USB_RemoteWakeupEnabled
)
331 CurrentStatus
|= FEATURE_REMOTE_WAKEUP_ENABLED
;
334 #if !defined(CONTROL_ONLY_DEVICE)
335 case (REQDIR_DEVICETOHOST
| REQTYPE_STANDARD
| REQREC_ENDPOINT
):
336 Endpoint_SelectEndpoint((uint8_t)USB_ControlRequest
.wIndex
);
338 CurrentStatus
= Endpoint_IsStalled();
340 Endpoint_SelectEndpoint(ENDPOINT_CONTROLEP
);
346 Endpoint_ClearSETUP();
348 Endpoint_Write_Word_LE(CurrentStatus
);
352 while (!(Endpoint_IsOUTReceived()))
354 if (USB_DeviceState
== DEVICE_STATE_Unattached
)
361 static void USB_Device_ClearSetFeature(void)
363 switch (USB_ControlRequest
.bmRequestType
& CONTROL_REQTYPE_RECIPIENT
)
366 if ((uint8_t)USB_ControlRequest
.wValue
== FEATURE_REMOTE_WAKEUP
)
367 USB_RemoteWakeupEnabled
= (USB_ControlRequest
.bRequest
== REQ_SetFeature
);
372 #if !defined(CONTROL_ONLY_DEVICE)
373 case REQREC_ENDPOINT
:
374 if ((uint8_t)USB_ControlRequest
.wValue
== FEATURE_ENDPOINT_HALT
)
376 uint8_t EndpointIndex
= ((uint8_t)USB_ControlRequest
.wIndex
& ENDPOINT_EPNUM_MASK
);
378 if (EndpointIndex
== ENDPOINT_CONTROLEP
)
381 Endpoint_SelectEndpoint(EndpointIndex
);
383 if (Endpoint_IsEnabled())
385 if (USB_ControlRequest
.bRequest
== REQ_ClearFeature
)
387 Endpoint_ClearStall();
388 Endpoint_ResetFIFO(EndpointIndex
);
389 Endpoint_ResetDataToggle();
393 Endpoint_StallTransaction();
402 Endpoint_SelectEndpoint(ENDPOINT_CONTROLEP
);
404 Endpoint_ClearSETUP();