3      Copyright (C) Dean Camera, 2010. 
   5   dean [at] fourwalledcubicle [dot] com 
   6       www.fourwalledcubicle.com 
  10   Copyright 2010  Dean Camera (dean [at] fourwalledcubicle [dot] com) 
  12   Permission to use, copy, modify, distribute, and sell this  
  13   software and its documentation for any purpose is hereby granted 
  14   without fee, provided that the above copyright notice appear in  
  15   all 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 #define  __INCLUDE_FROM_USB_DRIVER 
  32 #include "../../HighLevel/USBMode.h" 
  33 #if defined(USB_CAN_BE_HOST) 
  35 #define  __INCLUDE_FROM_CDC_CLASS_HOST_C 
  36 #define  __INCLUDE_FROM_CDC_DRIVER 
  39 uint8_t CDC_Host_ConfigurePipes(USB_ClassInfo_CDC_Host_t
* const CDCInterfaceInfo
, 
  40                                 uint16_t ConfigDescriptorSize
, 
  41                                 void* ConfigDescriptorData
) 
  43         uint8_t FoundEndpoints 
= 0; 
  45         memset(&CDCInterfaceInfo
->State
, 0x00, sizeof(CDCInterfaceInfo
->State
)); 
  47         if (DESCRIPTOR_TYPE(ConfigDescriptorData
) != DTYPE_Configuration
) 
  48           return CDC_ENUMERROR_InvalidConfigDescriptor
; 
  50         if (USB_GetNextDescriptorComp(&ConfigDescriptorSize
, &ConfigDescriptorData
, 
  51                                       DCOMP_CDC_Host_NextCDCControlInterface
) != DESCRIPTOR_SEARCH_COMP_Found
) 
  53                 return CDC_ENUMERROR_NoCDCInterfaceFound
; 
  56         CDCInterfaceInfo
->State
.ControlInterfaceNumber 
= DESCRIPTOR_CAST(ConfigDescriptorData
, USB_Descriptor_Interface_t
).InterfaceNumber
; 
  58         while (FoundEndpoints 
!= (CDC_FOUND_NOTIFICATION_IN 
| CDC_FOUND_DATAPIPE_IN 
| CDC_FOUND_DATAPIPE_OUT
)) 
  60                 if (USB_GetNextDescriptorComp(&ConfigDescriptorSize
, &ConfigDescriptorData
, 
  61                                               DCOMP_CDC_Host_NextCDCInterfaceEndpoint
) != DESCRIPTOR_SEARCH_COMP_Found
) 
  63                         if (FoundEndpoints 
& CDC_FOUND_NOTIFICATION_IN
) 
  65                                 if (USB_GetNextDescriptorComp(&ConfigDescriptorSize
, &ConfigDescriptorData
,  
  66                                                               DCOMP_CDC_Host_NextCDCDataInterface
) != DESCRIPTOR_SEARCH_COMP_Found
) 
  68                                         return CDC_ENUMERROR_NoCDCInterfaceFound
; 
  75                                 Pipe_SelectPipe(CDCInterfaceInfo
->Config
.DataINPipeNumber
); 
  77                                 Pipe_SelectPipe(CDCInterfaceInfo
->Config
.DataOUTPipeNumber
); 
  79                                 Pipe_SelectPipe(CDCInterfaceInfo
->Config
.NotificationPipeNumber
); 
  82                                 if (USB_GetNextDescriptorComp(&ConfigDescriptorSize
, &ConfigDescriptorData
, 
  83                                                               DCOMP_CDC_Host_NextCDCControlInterface
) != DESCRIPTOR_SEARCH_COMP_Found
) 
  85                                         return CDC_ENUMERROR_NoCDCInterfaceFound
; 
  89                         if (USB_GetNextDescriptorComp(&ConfigDescriptorSize
, &ConfigDescriptorData
, 
  90                                                       DCOMP_CDC_Host_NextCDCInterfaceEndpoint
) != DESCRIPTOR_SEARCH_COMP_Found
) 
  92                                 return CDC_ENUMERROR_EndpointsNotFound
; 
  96                 USB_Descriptor_Endpoint_t
* EndpointData 
= DESCRIPTOR_PCAST(ConfigDescriptorData
, USB_Descriptor_Endpoint_t
); 
  98                 if ((EndpointData
->Attributes 
& EP_TYPE_MASK
) == EP_TYPE_INTERRUPT
) 
 100                         if (EndpointData
->EndpointAddress 
& ENDPOINT_DESCRIPTOR_DIR_IN
) 
 102                                 Pipe_ConfigurePipe(CDCInterfaceInfo
->Config
.NotificationPipeNumber
, EP_TYPE_INTERRUPT
, PIPE_TOKEN_IN
, 
 103                                                                    EndpointData
->EndpointAddress
, EndpointData
->EndpointSize
, 
 104                                                                    CDCInterfaceInfo
->Config
.NotificationPipeDoubleBank ? PIPE_BANK_DOUBLE 
: PIPE_BANK_SINGLE
); 
 105                                 CDCInterfaceInfo
->State
.NotificationPipeSize 
= EndpointData
->EndpointSize
; 
 107                                 Pipe_SetInterruptPeriod(EndpointData
->PollingIntervalMS
); 
 109                                 FoundEndpoints 
|= CDC_FOUND_NOTIFICATION_IN
; 
 114                         if (EndpointData
->EndpointAddress 
& ENDPOINT_DESCRIPTOR_DIR_IN
) 
 116                                 Pipe_ConfigurePipe(CDCInterfaceInfo
->Config
.DataINPipeNumber
, EP_TYPE_BULK
, PIPE_TOKEN_IN
, 
 117                                                    EndpointData
->EndpointAddress
, EndpointData
->EndpointSize
,  
 118                                                    CDCInterfaceInfo
->Config
.DataINPipeDoubleBank ? PIPE_BANK_DOUBLE 
: PIPE_BANK_SINGLE
); 
 120                                 CDCInterfaceInfo
->State
.DataINPipeSize 
= EndpointData
->EndpointSize
; 
 122                                 FoundEndpoints 
|= CDC_FOUND_DATAPIPE_IN
; 
 126                                 Pipe_ConfigurePipe(CDCInterfaceInfo
->Config
.DataOUTPipeNumber
, EP_TYPE_BULK
, PIPE_TOKEN_OUT
, 
 127                                                                    EndpointData
->EndpointAddress
, EndpointData
->EndpointSize
,  
 128                                                                    CDCInterfaceInfo
->Config
.DataOUTPipeDoubleBank ? PIPE_BANK_DOUBLE 
: PIPE_BANK_SINGLE
); 
 130                                 CDCInterfaceInfo
->State
.DataOUTPipeSize 
= EndpointData
->EndpointSize
; 
 132                                 FoundEndpoints 
|= CDC_FOUND_DATAPIPE_OUT
; 
 137         CDCInterfaceInfo
->State
.ControlLineStates
.HostToDevice 
= (CDC_CONTROL_LINE_OUT_RTS 
| CDC_CONTROL_LINE_OUT_DTR
); 
 138         CDCInterfaceInfo
->State
.ControlLineStates
.DeviceToHost 
= (CDC_CONTROL_LINE_IN_DCD  
| CDC_CONTROL_LINE_IN_DSR
); 
 139         CDCInterfaceInfo
->State
.IsActive 
= true; 
 140         return CDC_ENUMERROR_NoError
; 
 143 static uint8_t DCOMP_CDC_Host_NextCDCControlInterface(void* const CurrentDescriptor
) 
 145         if (DESCRIPTOR_TYPE(CurrentDescriptor
) == DTYPE_Interface
) 
 147                 USB_Descriptor_Interface_t
* CurrentInterface 
= DESCRIPTOR_PCAST(CurrentDescriptor
, 
 148                                                                                 USB_Descriptor_Interface_t
); 
 150                 if ((CurrentInterface
->Class    
== CDC_CONTROL_CLASS
)    && 
 151                     (CurrentInterface
->SubClass 
== CDC_CONTROL_SUBCLASS
) && 
 152                         (CurrentInterface
->Protocol 
== CDC_CONTROL_PROTOCOL
)) 
 154                         return DESCRIPTOR_SEARCH_Found
; 
 158         return DESCRIPTOR_SEARCH_NotFound
; 
 161 static uint8_t DCOMP_CDC_Host_NextCDCDataInterface(void* const CurrentDescriptor
) 
 163         if (DESCRIPTOR_TYPE(CurrentDescriptor
) == DTYPE_Interface
) 
 165                 USB_Descriptor_Interface_t
* CurrentInterface 
= DESCRIPTOR_PCAST(CurrentDescriptor
, 
 166                                                                                 USB_Descriptor_Interface_t
); 
 168                 if ((CurrentInterface
->Class    
== CDC_DATA_CLASS
)    && 
 169                     (CurrentInterface
->SubClass 
== CDC_DATA_SUBCLASS
) && 
 170                         (CurrentInterface
->Protocol 
== CDC_DATA_PROTOCOL
)) 
 172                         return DESCRIPTOR_SEARCH_Found
; 
 176         return DESCRIPTOR_SEARCH_NotFound
; 
 179 static uint8_t DCOMP_CDC_Host_NextCDCInterfaceEndpoint(void* const CurrentDescriptor
) 
 181         if (DESCRIPTOR_TYPE(CurrentDescriptor
) == DTYPE_Endpoint
) 
 183                 USB_Descriptor_Endpoint_t
* CurrentEndpoint 
= DESCRIPTOR_PCAST(CurrentDescriptor
, 
 184                                                                               USB_Descriptor_Endpoint_t
); 
 186                 uint8_t EndpointType 
= (CurrentEndpoint
->Attributes 
& EP_TYPE_MASK
); 
 188                 if (((EndpointType 
== EP_TYPE_BULK
) || (EndpointType 
== EP_TYPE_INTERRUPT
)) && 
 189                     !(Pipe_IsEndpointBound(CurrentEndpoint
->EndpointAddress
))) 
 191                         return DESCRIPTOR_SEARCH_Found
; 
 194         else if (DESCRIPTOR_TYPE(CurrentDescriptor
) == DTYPE_Interface
) 
 196                 return DESCRIPTOR_SEARCH_Fail
; 
 199         return DESCRIPTOR_SEARCH_NotFound
; 
 202 void CDC_Host_USBTask(USB_ClassInfo_CDC_Host_t
* const CDCInterfaceInfo
) 
 204         if ((USB_HostState 
!= HOST_STATE_Configured
) || !(CDCInterfaceInfo
->State
.IsActive
)) 
 207         Pipe_SelectPipe(CDCInterfaceInfo
->Config
.NotificationPipeNumber
); 
 208         Pipe_SetPipeToken(PIPE_TOKEN_IN
); 
 211         if (Pipe_IsINReceived()) 
 213                 USB_Request_Header_t Notification
; 
 214                 Pipe_Read_Stream_LE(&Notification
, sizeof(USB_Request_Header_t
), NO_STREAM_CALLBACK
); 
 216                 if ((Notification
.bRequest      
== NOTIF_SerialState
) && 
 217                     (Notification
.bmRequestType 
== (REQDIR_DEVICETOHOST 
| REQTYPE_CLASS 
| REQREC_INTERFACE
))) 
 219                         Pipe_Read_Stream_LE(&CDCInterfaceInfo
->State
.ControlLineStates
.DeviceToHost
, 
 220                                             sizeof(CDCInterfaceInfo
->State
.ControlLineStates
.DeviceToHost
), 
 225                         EVENT_CDC_Host_ControLineStateChanged(CDCInterfaceInfo
); 
 235         CDC_Host_Flush(CDCInterfaceInfo
); 
 238 uint8_t CDC_Host_SetLineEncoding(USB_ClassInfo_CDC_Host_t
* const CDCInterfaceInfo
) 
 240         USB_ControlRequest 
= (USB_Request_Header_t
) 
 242                 .bmRequestType 
= (REQDIR_HOSTTODEVICE 
| REQTYPE_CLASS 
| REQREC_INTERFACE
), 
 243                 .bRequest      
= REQ_SetLineEncoding
, 
 245                 .wIndex        
= CDCInterfaceInfo
->State
.ControlInterfaceNumber
, 
 246                 .wLength       
= sizeof(CDCInterfaceInfo
->State
.LineEncoding
), 
 249         Pipe_SelectPipe(PIPE_CONTROLPIPE
); 
 251         return USB_Host_SendControlRequest(&CDCInterfaceInfo
->State
.LineEncoding
); 
 254 uint8_t CDC_Host_SendControlLineStateChange(USB_ClassInfo_CDC_Host_t
* const CDCInterfaceInfo
) 
 256         USB_ControlRequest 
= (USB_Request_Header_t
) 
 258                 .bmRequestType 
= (REQDIR_HOSTTODEVICE 
| REQTYPE_CLASS 
| REQREC_INTERFACE
), 
 259                 .bRequest      
= REQ_SetControlLineState
, 
 260                 .wValue        
= CDCInterfaceInfo
->State
.ControlLineStates
.HostToDevice
, 
 261                 .wIndex        
= CDCInterfaceInfo
->State
.ControlInterfaceNumber
, 
 265         Pipe_SelectPipe(PIPE_CONTROLPIPE
); 
 267         return USB_Host_SendControlRequest(NULL
); 
 270 uint8_t CDC_Host_SendBreak(USB_ClassInfo_CDC_Host_t
* const CDCInterfaceInfo
, 
 271                            const uint8_t Duration
) 
 273         USB_ControlRequest 
= (USB_Request_Header_t
) 
 275                 .bmRequestType 
= (REQDIR_HOSTTODEVICE 
| REQTYPE_CLASS 
| REQREC_INTERFACE
), 
 276                 .bRequest      
= REQ_SendBreak
, 
 278                 .wIndex        
= CDCInterfaceInfo
->State
.ControlInterfaceNumber
, 
 282         Pipe_SelectPipe(PIPE_CONTROLPIPE
); 
 284         return USB_Host_SendControlRequest(NULL
); 
 287 uint8_t CDC_Host_SendString(USB_ClassInfo_CDC_Host_t
* const CDCInterfaceInfo
, 
 289                             const uint16_t Length
) 
 291         if ((USB_HostState 
!= HOST_STATE_Configured
) || !(CDCInterfaceInfo
->State
.IsActive
)) 
 292           return PIPE_READYWAIT_DeviceDisconnected
; 
 296         Pipe_SelectPipe(CDCInterfaceInfo
->Config
.DataOUTPipeNumber
);     
 299         ErrorCode 
= Pipe_Write_Stream_LE(Data
, Length
, NO_STREAM_CALLBACK
); 
 305 uint8_t CDC_Host_SendByte(USB_ClassInfo_CDC_Host_t
* const CDCInterfaceInfo
, 
 308         if ((USB_HostState 
!= HOST_STATE_Configured
) || !(CDCInterfaceInfo
->State
.IsActive
)) 
 309           return PIPE_READYWAIT_DeviceDisconnected
; 
 313         Pipe_SelectPipe(CDCInterfaceInfo
->Config
.DataOUTPipeNumber
);     
 316         if (!(Pipe_IsReadWriteAllowed())) 
 320                 if ((ErrorCode 
= Pipe_WaitUntilReady()) != PIPE_READYWAIT_NoError
) 
 324         Pipe_Write_Byte(Data
);   
 327         return PIPE_READYWAIT_NoError
; 
 330 uint16_t CDC_Host_BytesReceived(USB_ClassInfo_CDC_Host_t
* const CDCInterfaceInfo
) 
 332         if ((USB_HostState 
!= HOST_STATE_Configured
) || !(CDCInterfaceInfo
->State
.IsActive
)) 
 335         Pipe_SelectPipe(CDCInterfaceInfo
->Config
.DataINPipeNumber
); 
 336         Pipe_SetPipeToken(PIPE_TOKEN_IN
); 
 339         if (Pipe_IsINReceived()) 
 341                 if (!(Pipe_BytesInPipe())) 
 350                         return Pipe_BytesInPipe(); 
 361 int16_t CDC_Host_ReceiveByte(USB_ClassInfo_CDC_Host_t
* const CDCInterfaceInfo
) 
 363         uint8_t ReceivedByte 
= -1; 
 365         if ((USB_HostState 
!= HOST_STATE_Configured
) || !(CDCInterfaceInfo
->State
.IsActive
)) 
 368         Pipe_SelectPipe(CDCInterfaceInfo
->Config
.DataINPipeNumber
); 
 369         Pipe_SetPipeToken(PIPE_TOKEN_IN
); 
 372         if (!(Pipe_IsINReceived())) 
 374         else if (Pipe_BytesInPipe()) 
 375           ReceivedByte 
= Pipe_Read_Byte(); 
 377         if (!(Pipe_BytesInPipe())) 
 385 uint8_t CDC_Host_Flush(USB_ClassInfo_CDC_Host_t
* const CDCInterfaceInfo
) 
 387         if ((USB_HostState 
!= HOST_STATE_Configured
) || !(CDCInterfaceInfo
->State
.IsActive
)) 
 388           return PIPE_READYWAIT_DeviceDisconnected
; 
 392         Pipe_SelectPipe(CDCInterfaceInfo
->Config
.DataOUTPipeNumber
);     
 395         if (!(Pipe_BytesInPipe())) 
 396           return PIPE_READYWAIT_NoError
; 
 398         bool BankFull 
= !(Pipe_IsReadWriteAllowed()); 
 404                 if ((ErrorCode 
= Pipe_WaitUntilReady()) != PIPE_READYWAIT_NoError
) 
 412         return PIPE_READYWAIT_NoError
; 
 415 void CDC_Host_CreateStream(USB_ClassInfo_CDC_Host_t
* const CDCInterfaceInfo
, 
 418         *Stream 
= (FILE)FDEV_SETUP_STREAM(CDC_Host_putchar
, CDC_Host_getchar
, _FDEV_SETUP_RW
); 
 419         fdev_set_udata(Stream
, CDCInterfaceInfo
); 
 422 void CDC_Host_CreateBlockingStream(USB_ClassInfo_CDC_Host_t
* const CDCInterfaceInfo
, 
 425         *Stream 
= (FILE)FDEV_SETUP_STREAM(CDC_Host_putchar
, CDC_Host_getchar_Blocking
, _FDEV_SETUP_RW
); 
 426         fdev_set_udata(Stream
, CDCInterfaceInfo
); 
 429 static int CDC_Host_putchar(char c
, 
 432         return CDC_Host_SendByte((USB_ClassInfo_CDC_Host_t
*)fdev_get_udata(Stream
), c
) ? _FDEV_ERR 
: 0; 
 435 static int CDC_Host_getchar(FILE* Stream
) 
 437         int16_t ReceivedByte 
= CDC_Host_ReceiveByte((USB_ClassInfo_CDC_Host_t
*)fdev_get_udata(Stream
)); 
 439         if (ReceivedByte 
< 0) 
 445 static int CDC_Host_getchar_Blocking(FILE* Stream
) 
 447         int16_t ReceivedByte
; 
 449         while ((ReceivedByte 
= CDC_Host_ReceiveByte((USB_ClassInfo_CDC_Host_t
*)fdev_get_udata(Stream
))) < 0) 
 451                 if (USB_HostState 
== HOST_STATE_Unattached
) 
 454                 CDC_Host_USBTask((USB_ClassInfo_CDC_Host_t
*)fdev_get_udata(Stream
)); 
 461 void CDC_Host_Event_Stub(void)