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         Endpoint_ClearSETUP(); 
 124         while (!(Endpoint_IsINReady())); 
 126         UDADDR 
= ((1 << ADDEN
) | ((uint8_t)USB_ControlRequest
.wValue 
& 0x7F)); 
 131 static void USB_Device_SetConfiguration(void) 
 133         bool AlreadyConfigured 
= (USB_ConfigurationNumber 
!= 0); 
 135 #if defined(TOTAL_NUM_CONFIGURATIONS) 
 136         if ((uint8_t)USB_ControlRequest
.wValue 
> TOTAL_NUM_CONFIGURATIONS
) 
 139         #if !defined(USE_FLASH_DESCRIPTORS) && !defined(USE_EEPROM_DESCRIPTORS) && !defined(USE_RAM_DESCRIPTORS) 
 140         uint8_t MemoryAddressSpace
; 
 143         USB_Descriptor_Device_t
* DevDescriptorPtr
; 
 145         if (CALLBACK_USB_GetDescriptor((DTYPE_Device 
<< 8), 0, (void*)&DevDescriptorPtr
 
 146         #if !defined(USE_FLASH_DESCRIPTORS) && !defined(USE_EEPROM_DESCRIPTORS) && !defined(USE_RAM_DESCRIPTORS) 
 147                                        , &MemoryAddressSpace
) == NO_DESCRIPTOR
) 
 153         #if defined(USE_RAM_DESCRIPTORS) 
 154         if ((uint8_t)USB_ControlRequest
.wValue 
> DevDescriptorPtr
->NumberOfConfigurations
) 
 156         #elif defined (USE_EEPROM_DESCRIPTORS) 
 157         if ((uint8_t)USB_ControlRequest
.wValue 
> eeprom_read_byte(&DevDescriptorPtr
->NumberOfConfigurations
)) 
 159         #elif defined (USE_FLASH_DESCRIPTORS) 
 160         if ((uint8_t)USB_ControlRequest
.wValue 
> pgm_read_byte(&DevDescriptorPtr
->NumberOfConfigurations
)) 
 163         if (MemoryAddressSpace 
== MEMSPACE_FLASH
) 
 165                 if (((uint8_t)USB_ControlRequest
.wValue 
> pgm_read_byte(&DevDescriptorPtr
->NumberOfConfigurations
))) 
 168         else if (MemoryAddressSpace 
== MEMSPACE_EEPROM
) 
 170                 if (((uint8_t)USB_ControlRequest
.wValue 
> eeprom_read_byte(&DevDescriptorPtr
->NumberOfConfigurations
))) 
 175                 if ((uint8_t)USB_ControlRequest
.wValue 
> DevDescriptorPtr
->NumberOfConfigurations
) 
 181         Endpoint_ClearSETUP(); 
 183         USB_ConfigurationNumber 
= (uint8_t)USB_ControlRequest
.wValue
; 
 187         if (!(AlreadyConfigured
) && USB_ConfigurationNumber
) 
 188           EVENT_USB_DeviceEnumerationComplete(); 
 190         EVENT_USB_ConfigurationChanged(); 
 193 void USB_Device_GetConfiguration(void) 
 195         Endpoint_ClearSETUP(); 
 197         Endpoint_Write_Byte(USB_ConfigurationNumber
); 
 201         while (!(Endpoint_IsOUTReceived())); 
 205 #if !defined(NO_INTERNAL_SERIAL) && (defined(USB_SERIES_6_AVR) || defined(USB_SERIES_7_AVR)) 
 206 static void USB_Device_GetInternalSerialDescriptor(void) 
 210                 USB_Descriptor_Header_t Header
; 
 211                 int16_t                 UnicodeString
[12]; 
 212         } SignatureDescriptor
; 
 214         uint8_t SigReadAddress  
= 0x0E;          
 215         bool    OddNibbleRead   
= false; 
 217         #if defined(USE_NONSTANDARD_DESCRIPTOR_NAMES) 
 218                 SignatureDescriptor
.Header
.Size            
= sizeof(SignatureDescriptor
); 
 219                 SignatureDescriptor
.Header
.Type            
= DTYPE_String
; 
 221                 SignatureDescriptor
.Header
.bLength         
= sizeof(SignatureDescriptor
); 
 222                 SignatureDescriptor
.Header
.bDescriptorType 
= DTYPE_String
; 
 225         for (uint8_t SerialCharNum 
= 0; SerialCharNum 
< 12; SerialCharNum
++) 
 227                 uint8_t SerialByte 
= boot_signature_byte_get(SigReadAddress
); 
 239                 OddNibbleRead 
= !(OddNibbleRead
); 
 241                 if (SerialByte 
< 0x0A) 
 244                   SerialByte 
+= ('A' - 0x0A); 
 246                 SignatureDescriptor
.UnicodeString
[SerialCharNum
] = SerialByte
; 
 249         Endpoint_ClearSETUP(); 
 250         Endpoint_Write_Control_Stream_LE(&SignatureDescriptor
, sizeof(SignatureDescriptor
)); 
 255 static void USB_Device_GetDescriptor(void) 
 257         void*    DescriptorPointer
; 
 258         uint16_t DescriptorSize
; 
 260         #if !defined(USE_FLASH_DESCRIPTORS) && !defined(USE_EEPROM_DESCRIPTORS) && !defined(USE_RAM_DESCRIPTORS) 
 261         uint8_t  DescriptorAddressSpace
; 
 264         #if !defined(NO_INTERNAL_SERIAL) && (defined(USB_SERIES_6_AVR) || defined(USB_SERIES_7_AVR)) 
 265         if (USB_ControlRequest
.wValue 
== ((DTYPE_String 
<< 8) | USE_INTERNAL_SERIAL
)) 
 267                 USB_Device_GetInternalSerialDescriptor(); 
 272         if ((DescriptorSize 
= CALLBACK_USB_GetDescriptor(USB_ControlRequest
.wValue
, USB_ControlRequest
.wIndex
, 
 274         #if !defined(USE_FLASH_DESCRIPTORS) && !defined(USE_EEPROM_DESCRIPTORS) && !defined(USE_RAM_DESCRIPTORS) 
 275                                                          , &DescriptorAddressSpace
 
 282         Endpoint_ClearSETUP(); 
 284         #if defined(USE_RAM_DESCRIPTORS) 
 285         Endpoint_Write_Control_Stream_LE(DescriptorPointer
, DescriptorSize
); 
 286         #elif defined(USE_EEPROM_DESCRIPTORS) 
 287         Endpoint_Write_Control_EStream_LE(DescriptorPointer
, DescriptorSize
); 
 288         #elif defined(USE_FLASH_DESCRIPTORS) 
 289         Endpoint_Write_Control_PStream_LE(DescriptorPointer
, DescriptorSize
);    
 291         if (DescriptorAddressSpace 
== MEMSPACE_FLASH
) 
 292           Endpoint_Write_Control_PStream_LE(DescriptorPointer
, DescriptorSize
);  
 293         else if (DescriptorAddressSpace 
== MEMSPACE_EEPROM
) 
 294           Endpoint_Write_Control_EStream_LE(DescriptorPointer
, DescriptorSize
); 
 296           Endpoint_Write_Control_Stream_LE(DescriptorPointer
, DescriptorSize
);   
 302 static void USB_Device_GetStatus(void) 
 304         uint8_t CurrentStatus 
= 0; 
 306         switch (USB_ControlRequest
.bmRequestType
) 
 308                 case (REQDIR_DEVICETOHOST 
| REQTYPE_STANDARD 
| REQREC_DEVICE
): 
 309                         if (USB_CurrentlySelfPowered
) 
 310                           CurrentStatus 
|= FEATURE_SELFPOWERED_ENABLED
; 
 312                         if (USB_RemoteWakeupEnabled
) 
 313                           CurrentStatus 
|= FEATURE_REMOTE_WAKEUP_ENABLED
; 
 316 #if !defined(CONTROL_ONLY_DEVICE) 
 317                 case (REQDIR_DEVICETOHOST 
| REQTYPE_STANDARD 
| REQREC_ENDPOINT
): 
 318                         Endpoint_SelectEndpoint((uint8_t)USB_ControlRequest
.wIndex
); 
 320                         CurrentStatus 
= Endpoint_IsStalled(); 
 322                         Endpoint_SelectEndpoint(ENDPOINT_CONTROLEP
);                       
 328         Endpoint_ClearSETUP(); 
 330         Endpoint_Write_Word_LE(CurrentStatus
); 
 334         while (!(Endpoint_IsOUTReceived())); 
 338 static void USB_Device_ClearSetFeature(void) 
 340         switch (USB_ControlRequest
.bmRequestType 
& CONTROL_REQTYPE_RECIPIENT
) 
 343                         if ((uint8_t)USB_ControlRequest
.wValue 
== FEATURE_REMOTE_WAKEUP
) 
 344                           USB_RemoteWakeupEnabled 
= (USB_ControlRequest
.bRequest 
== REQ_SetFeature
); 
 349 #if !defined(CONTROL_ONLY_DEVICE) 
 350                 case REQREC_ENDPOINT
: 
 351                         if ((uint8_t)USB_ControlRequest
.wValue 
== FEATURE_ENDPOINT_HALT
) 
 353                                 uint8_t EndpointIndex 
= ((uint8_t)USB_ControlRequest
.wIndex 
& ENDPOINT_EPNUM_MASK
); 
 355                                 if (EndpointIndex 
== ENDPOINT_CONTROLEP
) 
 358                                 Endpoint_SelectEndpoint(EndpointIndex
); 
 360                                 if (Endpoint_IsEnabled()) 
 362                                         if (USB_ControlRequest
.bRequest 
== REQ_ClearFeature
) 
 364                                                 Endpoint_ClearStall(); 
 365                                                 Endpoint_ResetFIFO(EndpointIndex
); 
 366                                                 Endpoint_ResetDataToggle(); 
 370                                                 Endpoint_StallTransaction(); 
 379         Endpoint_SelectEndpoint(ENDPOINT_CONTROLEP
); 
 381         Endpoint_ClearSETUP();