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, 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 #include "../HighLevel/USBMode.h"
33 #if defined(USB_CAN_BE_HOST)
35 #define INCLUDE_FROM_HOST_C
38 void USB_Host_ProcessNextHostState(void)
40 uint8_t ErrorCode
= HOST_ENUMERROR_NoError
;
41 uint8_t SubErrorCode
= HOST_ENUMERROR_NoError
;
43 static uint16_t WaitMSRemaining
;
44 static uint8_t PostWaitState
;
46 switch (USB_HostState
)
48 case HOST_STATE_WaitForDevice
:
51 if ((SubErrorCode
= USB_Host_WaitMS(1)) != HOST_WAITERROR_Successful
)
53 USB_HostState
= PostWaitState
;
54 ErrorCode
= HOST_ENUMERROR_WaitStage
;
58 if (!(--WaitMSRemaining
))
59 USB_HostState
= PostWaitState
;
63 case HOST_STATE_Powered
:
64 WaitMSRemaining
= HOST_DEVICE_SETTLE_DELAY_MS
;
66 USB_HostState
= HOST_STATE_Powered_WaitForDeviceSettle
;
68 case HOST_STATE_Powered_WaitForDeviceSettle
:
69 if (WaitMSRemaining
--)
76 USB_Host_VBUS_Manual_Off();
79 USB_Host_VBUS_Auto_Enable();
80 USB_Host_VBUS_Auto_On();
82 USB_HostState
= HOST_STATE_Powered_WaitForConnect
;
86 case HOST_STATE_Powered_WaitForConnect
:
87 if (USB_INT_HasOccurred(USB_INT_DCONNI
))
89 USB_INT_Clear(USB_INT_DCONNI
);
90 USB_INT_Clear(USB_INT_DDISCI
);
92 USB_INT_Clear(USB_INT_VBERRI
);
93 USB_INT_Enable(USB_INT_VBERRI
);
98 HOST_TASK_NONBLOCK_WAIT(100, HOST_STATE_Powered_DoReset
);
102 case HOST_STATE_Powered_DoReset
:
103 USB_Host_ResetDevice();
105 HOST_TASK_NONBLOCK_WAIT(200, HOST_STATE_Powered_ConfigPipe
);
107 case HOST_STATE_Powered_ConfigPipe
:
108 Pipe_ConfigurePipe(PIPE_CONTROLPIPE
, EP_TYPE_CONTROL
,
109 PIPE_TOKEN_SETUP
, ENDPOINT_CONTROLEP
,
110 PIPE_CONTROLPIPE_DEFAULT_SIZE
, PIPE_BANK_SINGLE
);
112 if (!(Pipe_IsConfigured()))
114 ErrorCode
= HOST_ENUMERROR_PipeConfigError
;
119 USB_HostState
= HOST_STATE_Default
;
121 case HOST_STATE_Default
:
122 USB_ControlRequest
= (USB_Request_Header_t
)
124 .bmRequestType
= (REQDIR_DEVICETOHOST
| REQTYPE_STANDARD
| REQREC_DEVICE
),
125 .bRequest
= REQ_GetDescriptor
,
126 .wValue
= (DTYPE_Device
<< 8),
131 uint8_t DataBuffer
[8];
133 if ((SubErrorCode
= USB_Host_SendControlRequest(DataBuffer
)) != HOST_SENDCONTROL_Successful
)
135 ErrorCode
= HOST_ENUMERROR_ControlError
;
139 USB_ControlPipeSize
= DataBuffer
[offsetof(USB_Descriptor_Device_t
, Endpoint0Size
)];
141 USB_Host_ResetDevice();
143 HOST_TASK_NONBLOCK_WAIT(200, HOST_STATE_Default_PostReset
);
145 case HOST_STATE_Default_PostReset
:
147 Pipe_DeallocateMemory();
148 Pipe_ResetPipe(PIPE_CONTROLPIPE
);
150 Pipe_ConfigurePipe(PIPE_CONTROLPIPE
, EP_TYPE_CONTROL
,
151 PIPE_TOKEN_SETUP
, ENDPOINT_CONTROLEP
,
152 USB_ControlPipeSize
, PIPE_BANK_SINGLE
);
154 if (!(Pipe_IsConfigured()))
156 ErrorCode
= HOST_ENUMERROR_PipeConfigError
;
161 USB_ControlRequest
= (USB_Request_Header_t
)
163 .bmRequestType
= (REQDIR_HOSTTODEVICE
| REQTYPE_STANDARD
| REQREC_DEVICE
),
164 .bRequest
= REQ_SetAddress
,
165 .wValue
= USB_HOST_DEVICEADDRESS
,
170 if ((SubErrorCode
= USB_Host_SendControlRequest(NULL
)) != HOST_SENDCONTROL_Successful
)
172 ErrorCode
= HOST_ENUMERROR_ControlError
;
176 HOST_TASK_NONBLOCK_WAIT(100, HOST_STATE_Default_PostAddressSet
);
178 case HOST_STATE_Default_PostAddressSet
:
179 USB_Host_SetDeviceAddress(USB_HOST_DEVICEADDRESS
);
181 EVENT_USB_Host_DeviceEnumerationComplete();
182 USB_HostState
= HOST_STATE_Addressed
;
186 if ((ErrorCode
!= HOST_ENUMERROR_NoError
) && (USB_HostState
!= HOST_STATE_Unattached
))
188 EVENT_USB_Host_DeviceEnumerationFailed(ErrorCode
, SubErrorCode
);
190 USB_Host_VBUS_Auto_Off();
192 EVENT_USB_Host_DeviceUnattached();
194 USB_ResetInterface();
198 uint8_t USB_Host_WaitMS(uint8_t MS
)
200 bool BusSuspended
= USB_Host_IsBusSuspended();
201 uint8_t ErrorCode
= HOST_WAITERROR_Successful
;
203 USB_Host_ResumeBus();
207 if (USB_INT_HasOccurred(USB_INT_HSOFI
))
209 USB_INT_Clear(USB_INT_HSOFI
);
213 if ((USB_HostState
== HOST_STATE_Unattached
) || (USB_CurrentMode
== USB_MODE_DEVICE
))
215 ErrorCode
= HOST_WAITERROR_DeviceDisconnect
;
220 if (Pipe_IsError() == true)
223 ErrorCode
= HOST_WAITERROR_PipeError
;
228 if (Pipe_IsStalled() == true)
231 ErrorCode
= HOST_WAITERROR_SetupStalled
;
238 USB_Host_SuspendBus();
243 static void USB_Host_ResetDevice(void)
245 bool BusSuspended
= USB_Host_IsBusSuspended();
247 USB_INT_Disable(USB_INT_DDISCI
);
250 while (!(USB_Host_IsBusResetComplete()));
252 USB_Host_ResumeBus();
254 USB_INT_Clear(USB_INT_HSOFI
);
256 for (uint8_t MSRem
= 10; MSRem
!= 0; MSRem
--)
258 /* Workaround for powerless-pull-up devices. After a USB bus reset,
259 all disconnection interrupts are suppressed while a USB frame is
260 looked for - if it is found within 10ms, the device is still
263 if (USB_INT_HasOccurred(USB_INT_HSOFI
))
265 USB_INT_Clear(USB_INT_HSOFI
);
266 USB_INT_Clear(USB_INT_DDISCI
);
274 USB_Host_SuspendBus();
276 USB_INT_Enable(USB_INT_DDISCI
);
279 uint8_t USB_Host_SetDeviceConfiguration(const uint8_t ConfigNumber
)
281 USB_ControlRequest
= (USB_Request_Header_t
)
283 .bmRequestType
= (REQDIR_HOSTTODEVICE
| REQTYPE_STANDARD
| REQREC_DEVICE
),
284 .bRequest
= REQ_SetConfiguration
,
285 .wValue
= ConfigNumber
,
290 Pipe_SelectPipe(PIPE_CONTROLPIPE
);
292 return USB_Host_SendControlRequest(NULL
);
295 uint8_t USB_Host_GetDeviceDescriptor(void* const DeviceDescriptorPtr
)
297 USB_ControlRequest
= (USB_Request_Header_t
)
299 .bmRequestType
= (REQDIR_DEVICETOHOST
| REQTYPE_STANDARD
| REQREC_DEVICE
),
300 .bRequest
= REQ_GetDescriptor
,
301 .wValue
= (DTYPE_Device
<< 8),
303 .wLength
= sizeof(USB_Descriptor_Device_t
),
306 Pipe_SelectPipe(PIPE_CONTROLPIPE
);
308 return USB_Host_SendControlRequest(DeviceDescriptorPtr
);
311 uint8_t USB_Host_GetDeviceStringDescriptor(uint8_t Index
, void* const Buffer
, uint8_t BufferLength
)
313 USB_ControlRequest
= (USB_Request_Header_t
)
315 .bmRequestType
= (REQDIR_DEVICETOHOST
| REQTYPE_STANDARD
| REQREC_DEVICE
),
316 .bRequest
= REQ_GetDescriptor
,
317 .wValue
= (DTYPE_String
<< 8) | Index
,
319 .wLength
= BufferLength
,
322 Pipe_SelectPipe(PIPE_CONTROLPIPE
);
324 return USB_Host_SendControlRequest(Buffer
);
327 uint8_t USB_Host_ClearPipeStall(uint8_t EndpointNum
)
329 if (Pipe_GetPipeToken() == PIPE_TOKEN_IN
)
330 EndpointNum
|= ENDPOINT_DESCRIPTOR_DIR_IN
;
332 USB_ControlRequest
= (USB_Request_Header_t
)
334 .bmRequestType
= (REQDIR_HOSTTODEVICE
| REQTYPE_STANDARD
| REQREC_ENDPOINT
),
335 .bRequest
= REQ_ClearFeature
,
336 .wValue
= FEATURE_ENDPOINT_HALT
,
337 .wIndex
= EndpointNum
,
341 Pipe_SelectPipe(PIPE_CONTROLPIPE
);
343 return USB_Host_SendControlRequest(NULL
);