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" 
  32 #if defined(USB_CAN_BE_HOST) 
  34 #define  INCLUDE_FROM_CDC_CLASS_HOST_C 
  37 uint8_t CDC_Host_ConfigurePipes(USB_ClassInfo_CDC_Host_t
* const CDCInterfaceInfo
, uint16_t ConfigDescriptorSize
, 
  38                                 uint8_t* ConfigDescriptorData
) 
  40         uint8_t FoundEndpoints 
= 0; 
  42         memset(&CDCInterfaceInfo
->State
, 0x00, sizeof(CDCInterfaceInfo
->State
)); 
  44         if (DESCRIPTOR_TYPE(ConfigDescriptorData
) != DTYPE_Configuration
) 
  45           return CDC_ENUMERROR_InvalidConfigDescriptor
; 
  47         if (USB_GetNextDescriptorComp(&ConfigDescriptorSize
, &ConfigDescriptorData
, 
  48                                       DComp_CDC_Host_NextCDCControlInterface
) != DESCRIPTOR_SEARCH_COMP_Found
) 
  50                 return CDC_ENUMERROR_NoCDCInterfaceFound
; 
  53         CDCInterfaceInfo
->State
.ControlInterfaceNumber 
= DESCRIPTOR_CAST(ConfigDescriptorData
, USB_Descriptor_Interface_t
).InterfaceNumber
; 
  55         while (FoundEndpoints 
!= (CDC_FOUND_NOTIFICATION_IN 
| CDC_FOUND_DATAPIPE_IN 
| CDC_FOUND_DATAPIPE_OUT
)) 
  57                 if (USB_GetNextDescriptorComp(&ConfigDescriptorSize
, &ConfigDescriptorData
, 
  58                                               DComp_CDC_Host_NextCDCInterfaceEndpoint
) != DESCRIPTOR_SEARCH_COMP_Found
) 
  60                         if (FoundEndpoints 
& CDC_FOUND_NOTIFICATION_IN
) 
  62                                 if (USB_GetNextDescriptorComp(&ConfigDescriptorSize
, &ConfigDescriptorData
,  
  63                                                               DComp_CDC_Host_NextCDCDataInterface
) != DESCRIPTOR_SEARCH_COMP_Found
) 
  65                                         return CDC_ENUMERROR_NoCDCInterfaceFound
; 
  72                                 Pipe_SelectPipe(CDCInterfaceInfo
->Config
.DataINPipeNumber
); 
  74                                 Pipe_SelectPipe(CDCInterfaceInfo
->Config
.DataOUTPipeNumber
); 
  76                                 Pipe_SelectPipe(CDCInterfaceInfo
->Config
.NotificationPipeNumber
); 
  79                                 if (USB_GetNextDescriptorComp(&ConfigDescriptorSize
, &ConfigDescriptorData
, 
  80                                                               DComp_CDC_Host_NextCDCControlInterface
) != DESCRIPTOR_SEARCH_COMP_Found
) 
  82                                         return CDC_ENUMERROR_NoCDCInterfaceFound
; 
  86                         if (USB_GetNextDescriptorComp(&ConfigDescriptorSize
, &ConfigDescriptorData
, 
  87                                                       DComp_CDC_Host_NextCDCInterfaceEndpoint
) != DESCRIPTOR_SEARCH_COMP_Found
) 
  89                                 return CDC_ENUMERROR_EndpointsNotFound
; 
  93                 USB_Descriptor_Endpoint_t
* EndpointData 
= DESCRIPTOR_PCAST(ConfigDescriptorData
, USB_Descriptor_Endpoint_t
); 
  95                 if ((EndpointData
->Attributes 
& EP_TYPE_MASK
) == EP_TYPE_INTERRUPT
) 
  97                         if (EndpointData
->EndpointAddress 
& ENDPOINT_DESCRIPTOR_DIR_IN
) 
  99                                 Pipe_ConfigurePipe(CDCInterfaceInfo
->Config
.NotificationPipeNumber
, EP_TYPE_INTERRUPT
, PIPE_TOKEN_IN
, 
 100                                                                    EndpointData
->EndpointAddress
, EndpointData
->EndpointSize
, PIPE_BANK_SINGLE
); 
 101                                 CDCInterfaceInfo
->State
.NotificationPipeSize 
= EndpointData
->EndpointSize
; 
 103                                 Pipe_SetInterruptPeriod(EndpointData
->PollingIntervalMS
); 
 105                                 FoundEndpoints 
|= CDC_FOUND_NOTIFICATION_IN
; 
 110                         if (EndpointData
->EndpointAddress 
& ENDPOINT_DESCRIPTOR_DIR_IN
) 
 112                                 Pipe_ConfigurePipe(CDCInterfaceInfo
->Config
.DataINPipeNumber
, EP_TYPE_BULK
, PIPE_TOKEN_IN
, 
 113                                                                    EndpointData
->EndpointAddress
, EndpointData
->EndpointSize
, PIPE_BANK_SINGLE
); 
 114                                 CDCInterfaceInfo
->State
.DataINPipeSize 
= EndpointData
->EndpointSize
; 
 116                                 FoundEndpoints 
|= CDC_FOUND_DATAPIPE_IN
; 
 120                                 Pipe_ConfigurePipe(CDCInterfaceInfo
->Config
.DataOUTPipeNumber
, EP_TYPE_BULK
, PIPE_TOKEN_OUT
, 
 121                                                                    EndpointData
->EndpointAddress
, EndpointData
->EndpointSize
, PIPE_BANK_SINGLE
); 
 122                                 CDCInterfaceInfo
->State
.DataOUTPipeSize 
= EndpointData
->EndpointSize
; 
 124                                 FoundEndpoints 
|= CDC_FOUND_DATAPIPE_OUT
; 
 129         CDCInterfaceInfo
->State
.ControlLineStates
.HostToDevice 
= (CDC_CONTROL_LINE_OUT_RTS 
| CDC_CONTROL_LINE_OUT_DTR
); 
 130         CDCInterfaceInfo
->State
.ControlLineStates
.DeviceToHost 
= (CDC_CONTROL_LINE_IN_DCD  
| CDC_CONTROL_LINE_IN_DSR
); 
 131         CDCInterfaceInfo
->State
.IsActive 
= true; 
 132         return CDC_ENUMERROR_NoError
; 
 135 static uint8_t DComp_CDC_Host_NextCDCControlInterface(void* const CurrentDescriptor
) 
 137         if (DESCRIPTOR_TYPE(CurrentDescriptor
) == DTYPE_Interface
) 
 139                 USB_Descriptor_Interface_t
* CurrentInterface 
= DESCRIPTOR_PCAST(CurrentDescriptor
, 
 140                                                                                 USB_Descriptor_Interface_t
); 
 142                 if ((CurrentInterface
->Class    
== CDC_CONTROL_CLASS
)    && 
 143                     (CurrentInterface
->SubClass 
== CDC_CONTROL_SUBCLASS
) && 
 144                         (CurrentInterface
->Protocol 
== CDC_CONTROL_PROTOCOL
)) 
 146                         return DESCRIPTOR_SEARCH_Found
; 
 150         return DESCRIPTOR_SEARCH_NotFound
; 
 153 static uint8_t DComp_CDC_Host_NextCDCDataInterface(void* const CurrentDescriptor
) 
 155         if (DESCRIPTOR_TYPE(CurrentDescriptor
) == DTYPE_Interface
) 
 157                 USB_Descriptor_Interface_t
* CurrentInterface 
= DESCRIPTOR_PCAST(CurrentDescriptor
, 
 158                                                                                 USB_Descriptor_Interface_t
); 
 160                 if ((CurrentInterface
->Class    
== CDC_DATA_CLASS
)    && 
 161                     (CurrentInterface
->SubClass 
== CDC_DATA_SUBCLASS
) && 
 162                         (CurrentInterface
->Protocol 
== CDC_DATA_PROTOCOL
)) 
 164                         return DESCRIPTOR_SEARCH_Found
; 
 168         return DESCRIPTOR_SEARCH_NotFound
; 
 171 static uint8_t DComp_CDC_Host_NextCDCInterfaceEndpoint(void* const CurrentDescriptor
) 
 173         if (DESCRIPTOR_TYPE(CurrentDescriptor
) == DTYPE_Endpoint
) 
 175                 USB_Descriptor_Endpoint_t
* CurrentEndpoint 
= DESCRIPTOR_PCAST(CurrentDescriptor
, 
 176                                                                               USB_Descriptor_Endpoint_t
); 
 178                 uint8_t EndpointType 
= (CurrentEndpoint
->Attributes 
& EP_TYPE_MASK
); 
 180                 if (((EndpointType 
== EP_TYPE_BULK
) || (EndpointType 
== EP_TYPE_INTERRUPT
)) && 
 181                     !(Pipe_IsEndpointBound(CurrentEndpoint
->EndpointAddress
))) 
 183                         return DESCRIPTOR_SEARCH_Found
; 
 186         else if (DESCRIPTOR_TYPE(CurrentDescriptor
) == DTYPE_Interface
) 
 188                 return DESCRIPTOR_SEARCH_Fail
; 
 191         return DESCRIPTOR_SEARCH_NotFound
; 
 194 void CDC_Host_USBTask(USB_ClassInfo_CDC_Host_t
* const CDCInterfaceInfo
) 
 196         if ((USB_HostState 
!= HOST_STATE_Configured
) || !(CDCInterfaceInfo
->State
.IsActive
)) 
 199         Pipe_SelectPipe(CDCInterfaceInfo
->Config
.NotificationPipeNumber
);        
 202         if (Pipe_IsINReceived()) 
 204                 USB_Request_Header_t Notification
; 
 205                 Pipe_Read_Stream_LE(&Notification
, sizeof(USB_Request_Header_t
), NO_STREAM_CALLBACK
); 
 207                 if ((Notification
.bRequest      
== NOTIF_SerialState
) && 
 208                     (Notification
.bmRequestType 
== (REQDIR_DEVICETOHOST 
| REQTYPE_CLASS 
| REQREC_INTERFACE
))) 
 210                         Pipe_Read_Stream_LE(&CDCInterfaceInfo
->State
.ControlLineStates
.DeviceToHost
, 
 211                                             sizeof(CDCInterfaceInfo
->State
.ControlLineStates
.DeviceToHost
), 
 218                 EVENT_CDC_Host_ControLineStateChanged(CDCInterfaceInfo
); 
 224 uint8_t CDC_Host_SetLineEncoding(USB_ClassInfo_CDC_Host_t
* const CDCInterfaceInfo
) 
 226         USB_ControlRequest 
= (USB_Request_Header_t
) 
 228                 .bmRequestType 
= (REQDIR_HOSTTODEVICE 
| REQTYPE_CLASS 
| REQREC_INTERFACE
), 
 229                 .bRequest      
= REQ_SetLineEncoding
, 
 231                 .wIndex        
= CDCInterfaceInfo
->State
.ControlInterfaceNumber
, 
 232                 .wLength       
= sizeof(CDCInterfaceInfo
->State
.LineEncoding
), 
 235         Pipe_SelectPipe(PIPE_CONTROLPIPE
); 
 237         return USB_Host_SendControlRequest(&CDCInterfaceInfo
->State
.LineEncoding
); 
 240 uint8_t CDC_Host_SendControlLineStateChange(USB_ClassInfo_CDC_Host_t
* const CDCInterfaceInfo
) 
 242         USB_ControlRequest 
= (USB_Request_Header_t
) 
 244                 .bmRequestType 
= (REQDIR_HOSTTODEVICE 
| REQTYPE_CLASS 
| REQREC_INTERFACE
), 
 245                 .bRequest      
= REQ_SetControlLineState
, 
 246                 .wValue        
= CDCInterfaceInfo
->State
.ControlLineStates
.HostToDevice
, 
 247                 .wIndex        
= CDCInterfaceInfo
->State
.ControlInterfaceNumber
, 
 251         Pipe_SelectPipe(PIPE_CONTROLPIPE
); 
 253         return USB_Host_SendControlRequest(NULL
); 
 256 uint8_t CDC_Host_SendString(USB_ClassInfo_CDC_Host_t
* const CDCInterfaceInfo
, char* Data
, const uint16_t Length
) 
 258         if ((USB_HostState 
!= HOST_STATE_Configured
) || !(CDCInterfaceInfo
->State
.IsActive
)) 
 259           return PIPE_READYWAIT_NoError
; 
 263         Pipe_SelectPipe(CDCInterfaceInfo
->Config
.DataOUTPipeNumber
);     
 265         ErrorCode 
= Pipe_Write_Stream_LE(Data
, Length
, NO_STREAM_CALLBACK
); 
 271 uint8_t CDC_Host_SendByte(USB_ClassInfo_CDC_Host_t
* const CDCInterfaceInfo
, const uint8_t Data
) 
 273         if ((USB_HostState 
!= HOST_STATE_Configured
) || !(CDCInterfaceInfo
->State
.IsActive
)) 
 274           return PIPE_READYWAIT_NoError
;; 
 278         Pipe_SelectPipe(CDCInterfaceInfo
->Config
.DataOUTPipeNumber
);     
 281         if (!(Pipe_IsReadWriteAllowed())) 
 285                 if ((ErrorCode 
= Pipe_WaitUntilReady()) != PIPE_READYWAIT_NoError
) 
 289         Pipe_Write_Byte(Data
);   
 292         return PIPE_READYWAIT_NoError
; 
 295 uint16_t CDC_Host_BytesReceived(USB_ClassInfo_CDC_Host_t
* const CDCInterfaceInfo
) 
 297         uint16_t BytesInPipe 
= 0; 
 299         if ((USB_HostState 
!= HOST_STATE_Configured
) || !(CDCInterfaceInfo
->State
.IsActive
)) 
 302         Pipe_SelectPipe(CDCInterfaceInfo
->Config
.DataINPipeNumber
);      
 305         if (Pipe_IsINReceived() && !(Pipe_BytesInPipe())) 
 308         BytesInPipe 
= Pipe_BytesInPipe(); 
 314 uint8_t CDC_Host_ReceiveByte(USB_ClassInfo_CDC_Host_t
* const CDCInterfaceInfo
) 
 316         uint8_t ReceivedByte 
= 0; 
 318         if ((USB_HostState 
!= HOST_STATE_Configured
) || !(CDCInterfaceInfo
->State
.IsActive
)) 
 321         Pipe_SelectPipe(CDCInterfaceInfo
->Config
.DataINPipeNumber
);      
 324         ReceivedByte 
= Pipe_Read_Byte(); 
 326         if (!(Pipe_BytesInPipe())) 
 334 void CDC_Host_Event_Stub(void)