예제 #1
0
파일: Host.c 프로젝트: NetHome/CULFirmware
void USB_Host_PrepareForDeviceConnect(void)
{
	USB_Host_VBUS_Auto_Off();
	USB_OTGPAD_Off();

	USB_Host_VBUS_Manual_Enable();
	USB_Host_VBUS_Manual_On();
	
	USB_INT_Enable(USB_INT_SRPI);
	USB_INT_Enable(USB_INT_BCERRI);

	USB_HostState = HOST_STATE_Unattached;
}
예제 #2
0
static void USB_Init_Host(void)
{
	USB_HostState       = HOST_STATE_Unattached;
	USB_ControlPipeSize = PIPE_CONTROLPIPE_DEFAULT_SIZE;

	USB_Host_HostMode_On();
	
	USB_Host_VBUS_Auto_Off();
	USB_Host_VBUS_Manual_Enable();
	USB_Host_VBUS_Manual_On();

	USB_INT_Enable(USB_INT_SRPI);
	USB_INT_Enable(USB_INT_BCERRI);

	USB_Attach();
}
예제 #3
0
void USB_ResetInterface(void)
{
	USB_INT_DisableAllInterrupts();
	USB_INT_ClearAllInterrupts();
	
	#if defined(USB_CAN_BE_HOST)
	USB_HostState   = HOST_STATE_Unattached;
	#endif
	
	#if defined(USB_CAN_BE_DEVICE)
	USB_DeviceState = DEVICE_STATE_Unattached;
	USB_ConfigurationNumber  = 0;

	#if !defined(NO_DEVICE_REMOTE_WAKEUP)
		USB_RemoteWakeupEnabled  = false;
	#endif
	
	#if !defined(NO_DEVICE_SELF_POWER)
		USB_CurrentlySelfPowered = false;
	#endif
	#endif
	
	if (!(USB_Options & USB_OPT_MANUAL_PLL))
	{
		#if defined(USB_SERIES_4_AVR)
		PLLFRQ = ((1 << PLLUSB) | (1 << PDIV3) | (1 << PDIV1));
		#endif

		USB_PLL_On();
		while (!(USB_PLL_IsReady()));
	}
	
	USB_Controller_Reset();
	
	#if defined(USB_CAN_BE_BOTH)
	if (UHWCON & (1 << UIDE))
	{
		USB_INT_Clear(USB_INT_IDTI);
		USB_INT_Enable(USB_INT_IDTI);
		USB_CurrentMode = USB_GetUSBModeFromUID();
	}
	#endif
		
	if (!(USB_Options & USB_OPT_REG_DISABLED))
	  USB_REG_On();
	else
	  USB_REG_Off();
	
	USB_CLK_Unfreeze();

	#if (defined(USB_CAN_BE_DEVICE) && (defined(USB_SERIES_4_AVR) || defined(USB_SERIES_6_AVR) || defined(USB_SERIES_7_AVR)))
	if (USB_CurrentMode == USB_MODE_DEVICE)
	{
		if (USB_Options & USB_DEVICE_OPT_LOWSPEED)
		  USB_Device_SetLowSpeed();
		else
		  USB_Device_SetFullSpeed();
	}
	#endif
	
	#if (defined(USB_CAN_BE_DEVICE) && !defined(FIXED_CONTROL_ENDPOINT_SIZE))
	if (USB_CurrentMode == USB_MODE_DEVICE)
	{
		USB_Descriptor_Device_t* DeviceDescriptorPtr;

		if (CALLBACK_USB_GetDescriptor((DTYPE_Device << 8), 0, (void*)&DeviceDescriptorPtr) != NO_DESCRIPTOR)
		{		  
			#if defined(USE_RAM_DESCRIPTORS)
			USB_ControlEndpointSize = DeviceDescriptorPtr->Endpoint0Size;
			#elif defined(USE_EEPROM_DESCRIPTORS)
			USB_ControlEndpointSize = eeprom_read_byte(&DeviceDescriptorPtr->Endpoint0Size);
			#else
			USB_ControlEndpointSize = pgm_read_byte(&DeviceDescriptorPtr->Endpoint0Size);
			#endif
		}
	}
	#endif

	USB_Attach();
	
	#if defined(USB_DEVICE_ONLY)
	USB_INT_Clear(USB_INT_SUSPEND);
	USB_INT_Enable(USB_INT_SUSPEND);
	USB_INT_Clear(USB_INT_EORSTI);
	USB_INT_Enable(USB_INT_EORSTI);

		#if defined(USB_SERIES_4_AVR) || defined(USB_SERIES_6_AVR) || defined(USB_SERIES_7_AVR)
		USB_INT_Enable(USB_INT_VBUS);
		#endif
	#elif defined(USB_HOST_ONLY)
	USB_Host_HostMode_On();
	
	USB_Host_VBUS_Auto_Off();
	USB_OTGPAD_Off();

	USB_Host_VBUS_Manual_Enable();
	USB_Host_VBUS_Manual_On();
	
	USB_INT_Enable(USB_INT_SRPI);
	USB_INT_Enable(USB_INT_BCERRI);
	#else
	if (USB_CurrentMode == USB_MODE_DEVICE)
	{
		USB_INT_Clear(USB_INT_SUSPEND);
		USB_INT_Enable(USB_INT_SUSPEND);
		USB_INT_Clear(USB_INT_EORSTI);
		USB_INT_Enable(USB_INT_EORSTI);

		#if defined(USB_SERIES_4_AVR) || defined(USB_SERIES_6_AVR) || defined(USB_SERIES_7_AVR)
		USB_INT_Enable(USB_INT_VBUS);
		#endif

		#if defined(CONTROL_ONLY_DEVICE)
		UENUM = ENDPOINT_CONTROLEP;
		#endif
	}
	else if (USB_CurrentMode == USB_MODE_HOST)
	{
		USB_Host_HostMode_On();
		
		USB_Host_VBUS_Auto_Off();
		USB_OTGPAD_Off();

		USB_Host_VBUS_Manual_Enable();
		USB_Host_VBUS_Manual_On();
		
		USB_INT_Enable(USB_INT_SRPI);
		USB_INT_Enable(USB_INT_BCERRI);
	}
	#endif
}
예제 #4
0
void USB_Host_ProcessNextHostState(void)
{
	uint8_t ErrorCode    = HOST_ENUMERROR_NoError;
	uint8_t SubErrorCode = HOST_ENUMERROR_NoError;

	static uint16_t WaitMSRemaining;
	static uint8_t  PostWaitState;

	switch (USB_HostState)
	{
		case HOST_STATE_WaitForDevice:
			if (WaitMSRemaining)
			{
				if ((SubErrorCode = USB_Host_WaitMS(1)) != HOST_WAITERROR_Successful)
				{
					USB_HostState = PostWaitState;
					ErrorCode     = HOST_ENUMERROR_WaitStage;
					break;
				}

				if (!(--WaitMSRemaining))
				  USB_HostState = PostWaitState;
			}

			break;
		case HOST_STATE_Powered:
			WaitMSRemaining = HOST_DEVICE_SETTLE_DELAY_MS;

			USB_HostState = HOST_STATE_Powered_WaitForDeviceSettle;
			break;
		case HOST_STATE_Powered_WaitForDeviceSettle:
			if (WaitMSRemaining--)
			{
				Delay_MS(1);
				break;
			}
			else
			{
				USB_Host_VBUS_Manual_Off();

				USB_OTGPAD_On();
				USB_Host_VBUS_Auto_Enable();
				USB_Host_VBUS_Auto_On();

				#if defined(NO_AUTO_VBUS_MANAGEMENT)
				USB_Host_VBUS_Manual_Enable();
				USB_Host_VBUS_Manual_On();
				#endif

				USB_HostState = HOST_STATE_Powered_WaitForConnect;
			}

			break;
		case HOST_STATE_Powered_WaitForConnect:
			if (USB_INT_HasOccurred(USB_INT_DCONNI))
			{
				USB_INT_Clear(USB_INT_DCONNI);
				USB_INT_Clear(USB_INT_DDISCI);

				USB_INT_Clear(USB_INT_VBERRI);
				USB_INT_Enable(USB_INT_VBERRI);

				USB_Host_ResumeBus();
				Pipe_ClearPipes();

				HOST_TASK_NONBLOCK_WAIT(100, HOST_STATE_Powered_DoReset);
			}

			break;
		case HOST_STATE_Powered_DoReset:
			USB_Host_ResetDevice();

			HOST_TASK_NONBLOCK_WAIT(200, HOST_STATE_Powered_ConfigPipe);
			break;
		case HOST_STATE_Powered_ConfigPipe:
			if (!(Pipe_ConfigurePipe(PIPE_CONTROLPIPE, EP_TYPE_CONTROL, ENDPOINT_CONTROLEP, PIPE_CONTROLPIPE_DEFAULT_SIZE, 1)))
			{
				ErrorCode    = HOST_ENUMERROR_PipeConfigError;
				SubErrorCode = 0;
				break;
			}

			USB_HostState = HOST_STATE_Default;
			break;
		case HOST_STATE_Default:
			USB_ControlRequest = (USB_Request_Header_t)
				{
					.bmRequestType = (REQDIR_DEVICETOHOST | REQTYPE_STANDARD | REQREC_DEVICE),
					.bRequest      = REQ_GetDescriptor,
					.wValue        = (DTYPE_Device << 8),
					.wIndex        = 0,
					.wLength       = 8,
				};

			uint8_t DataBuffer[8];

			Pipe_SelectPipe(PIPE_CONTROLPIPE);
			if ((SubErrorCode = USB_Host_SendControlRequest(DataBuffer)) != HOST_SENDCONTROL_Successful)
			{
				ErrorCode = HOST_ENUMERROR_ControlError;
				break;
			}

			USB_Host_ControlPipeSize = DataBuffer[offsetof(USB_Descriptor_Device_t, Endpoint0Size)];

			USB_Host_ResetDevice();

			HOST_TASK_NONBLOCK_WAIT(200, HOST_STATE_Default_PostReset);
			break;
		case HOST_STATE_Default_PostReset:
			if (!(Pipe_ConfigurePipe(PIPE_CONTROLPIPE, EP_TYPE_CONTROL, ENDPOINT_CONTROLEP, USB_Host_ControlPipeSize, 1)))
			{
				ErrorCode    = HOST_ENUMERROR_PipeConfigError;
				SubErrorCode = 0;
				break;
			}

			USB_ControlRequest = (USB_Request_Header_t)
				{
					.bmRequestType = (REQDIR_HOSTTODEVICE | REQTYPE_STANDARD | REQREC_DEVICE),
					.bRequest      = REQ_SetAddress,
					.wValue        = USB_HOST_DEVICEADDRESS,
					.wIndex        = 0,
					.wLength       = 0,
				};

			if ((SubErrorCode = USB_Host_SendControlRequest(NULL)) != HOST_SENDCONTROL_Successful)
			{
				ErrorCode = HOST_ENUMERROR_ControlError;
				break;
			}

			HOST_TASK_NONBLOCK_WAIT(100, HOST_STATE_Default_PostAddressSet);
			break;
		case HOST_STATE_Default_PostAddressSet:
			USB_Host_SetDeviceAddress(USB_HOST_DEVICEADDRESS);

			USB_HostState = HOST_STATE_Addressed;

			EVENT_USB_Host_DeviceEnumerationComplete();
			break;

		default:
			break;
	}

	if ((ErrorCode != HOST_ENUMERROR_NoError) && (USB_HostState != HOST_STATE_Unattached))
	{
		EVENT_USB_Host_DeviceEnumerationFailed(ErrorCode, SubErrorCode);

		USB_Host_VBUS_Auto_Off();

		EVENT_USB_Host_DeviceUnattached();

		USB_ResetInterface();
	}
}

uint8_t USB_Host_WaitMS(uint8_t MS)
{
	bool    BusSuspended = USB_Host_IsBusSuspended();
	uint8_t ErrorCode    = HOST_WAITERROR_Successful;
	bool    HSOFIEnabled = USB_INT_IsEnabled(USB_INT_HSOFI);

	USB_INT_Disable(USB_INT_HSOFI);
	USB_INT_Clear(USB_INT_HSOFI);

	USB_Host_ResumeBus();

	while (MS)
	{
		if (USB_INT_HasOccurred(USB_INT_HSOFI))
		{
			USB_INT_Clear(USB_INT_HSOFI);
			MS--;
		}

		if ((USB_HostState == HOST_STATE_Unattached) || (USB_CurrentMode != USB_MODE_Host))
		{
			ErrorCode = HOST_WAITERROR_DeviceDisconnect;

			break;
		}

		if (Pipe_IsError())
		{
			Pipe_ClearError();
			ErrorCode = HOST_WAITERROR_PipeError;

			break;
		}

		if (Pipe_IsStalled())
		{
			Pipe_ClearStall();
			ErrorCode = HOST_WAITERROR_SetupStalled;

			break;
		}
	}

	if (BusSuspended)
	  USB_Host_SuspendBus();

	if (HSOFIEnabled)
	  USB_INT_Enable(USB_INT_HSOFI);

	return ErrorCode;
}

static void USB_Host_ResetDevice(void)
{
	bool BusSuspended = USB_Host_IsBusSuspended();

	USB_INT_Disable(USB_INT_DDISCI);

	USB_Host_ResetBus();
	while (!(USB_Host_IsBusResetComplete()));
	USB_Host_ResumeBus();

	USB_Host_ConfigurationNumber = 0;

	bool HSOFIEnabled = USB_INT_IsEnabled(USB_INT_HSOFI);

	USB_INT_Disable(USB_INT_HSOFI);
	USB_INT_Clear(USB_INT_HSOFI);

	for (uint8_t MSRem = 10; MSRem != 0; MSRem--)
	{
		/* Workaround for powerless-pull-up devices. After a USB bus reset,
		   all disconnection interrupts are suppressed while a USB frame is
		   looked for - if it is found within 10ms, the device is still
		   present.                                                        */

		if (USB_INT_HasOccurred(USB_INT_HSOFI))
		{
			USB_INT_Clear(USB_INT_HSOFI);
			USB_INT_Clear(USB_INT_DDISCI);
			break;
		}

		Delay_MS(1);
	}

	if (HSOFIEnabled)
	  USB_INT_Enable(USB_INT_HSOFI);

	if (BusSuspended)
	  USB_Host_SuspendBus();

	USB_INT_Enable(USB_INT_DDISCI);
}
예제 #5
0
void USB_Host_ProcessNextHostState(void)
{
    uint8_t ErrorCode    = HOST_ENUMERROR_NoError;
    uint8_t SubErrorCode = HOST_ENUMERROR_NoError;

    static uint16_t WaitMSRemaining;
    static uint8_t  PostWaitState;

    switch (USB_HostState)
    {
    case HOST_STATE_WaitForDevice:
        if (WaitMSRemaining)
        {
            if ((SubErrorCode = USB_Host_WaitMS(1)) != HOST_WAITERROR_Successful)
            {
                USB_HostState = PostWaitState;
                ErrorCode     = HOST_ENUMERROR_WaitStage;
                break;
            }

            if (!(--WaitMSRemaining))
                USB_HostState = PostWaitState;
        }

        break;
    case HOST_STATE_Powered:
        WaitMSRemaining = HOST_DEVICE_SETTLE_DELAY_MS;

        USB_HostState = HOST_STATE_Powered_WaitForDeviceSettle;
        break;
    case HOST_STATE_Powered_WaitForDeviceSettle:
        if (WaitMSRemaining--)
        {
            _delay_ms(1);
            break;
        }
        else
        {
            USB_Host_VBUS_Manual_Off();

            USB_OTGPAD_On();
            USB_Host_VBUS_Auto_Enable();
            USB_Host_VBUS_Auto_On();

#if (BOARD == BOARD_MICROPENDOUS)
            // Micropendous boards require manual control of pin PE7's UVcon function
            USB_Host_VBUS_Manual_Enable();
            // clear VBUSREQ; AT90USB128 2009-11 Datasheet Pg#260
            OTGCON &= ~(1 << VBUSREQ);
            USB_Host_VBUS_Manual_Off();
#endif
            /* The above can also be done directly instead of through LUFA functions
            UHWCON &= ~(1 << UVCONE);
            OTGCON &= ~(1 << VBUSREQ);
            OTGCON |=  (1 << VBUSHWC);
            DDRE |= (1 << PE7);   PORTE &= ~(1 << PE7); PORTE = 0;
            */

            USB_HostState = HOST_STATE_Powered_WaitForConnect;
        }

        break;
    case HOST_STATE_Powered_WaitForConnect:
        if (USB_INT_HasOccurred(USB_INT_DCONNI))
        {
            USB_INT_Clear(USB_INT_DCONNI);
            USB_INT_Clear(USB_INT_DDISCI);

            USB_INT_Clear(USB_INT_VBERRI);
            USB_INT_Enable(USB_INT_VBERRI);

            USB_Host_ResumeBus();
            Pipe_ClearPipes();

            HOST_TASK_NONBLOCK_WAIT(100, HOST_STATE_Powered_DoReset);
        }

        break;
    case HOST_STATE_Powered_DoReset:
        USB_Host_ResetDevice();

        HOST_TASK_NONBLOCK_WAIT(200, HOST_STATE_Powered_ConfigPipe);
        break;
    case HOST_STATE_Powered_ConfigPipe:
        Pipe_ConfigurePipe(PIPE_CONTROLPIPE, EP_TYPE_CONTROL,
                           PIPE_TOKEN_SETUP, ENDPOINT_CONTROLEP,
                           PIPE_CONTROLPIPE_DEFAULT_SIZE, PIPE_BANK_SINGLE);

        if (!(Pipe_IsConfigured()))
        {
            ErrorCode    = HOST_ENUMERROR_PipeConfigError;
            SubErrorCode = 0;
            break;
        }

        USB_HostState = HOST_STATE_Default;
        break;
    case HOST_STATE_Default:
        USB_ControlRequest = (USB_Request_Header_t)
        {
            .bmRequestType = (REQDIR_DEVICETOHOST | REQTYPE_STANDARD | REQREC_DEVICE),
             .bRequest      = REQ_GetDescriptor,
              .wValue        = (DTYPE_Device << 8),
               .wIndex        = 0,
                .wLength       = 8,
        };

        uint8_t DataBuffer[8];

        if ((SubErrorCode = USB_Host_SendControlRequest(DataBuffer)) != HOST_SENDCONTROL_Successful)
        {
            ErrorCode = HOST_ENUMERROR_ControlError;
            break;
        }

        USB_ControlPipeSize = DataBuffer[offsetof(USB_Descriptor_Device_t, Endpoint0Size)];

        USB_Host_ResetDevice();

        HOST_TASK_NONBLOCK_WAIT(200, HOST_STATE_Default_PostReset);
        break;
    case HOST_STATE_Default_PostReset:
        Pipe_ConfigurePipe(PIPE_CONTROLPIPE, EP_TYPE_CONTROL,
                           PIPE_TOKEN_SETUP, ENDPOINT_CONTROLEP,
                           USB_ControlPipeSize, PIPE_BANK_SINGLE);

        if (!(Pipe_IsConfigured()))
        {
            ErrorCode    = HOST_ENUMERROR_PipeConfigError;
            SubErrorCode = 0;
            break;
        }

        USB_ControlRequest = (USB_Request_Header_t)
        {
            .bmRequestType = (REQDIR_HOSTTODEVICE | REQTYPE_STANDARD | REQREC_DEVICE),
             .bRequest      = REQ_SetAddress,
              .wValue        = USB_HOST_DEVICEADDRESS,
               .wIndex        = 0,
                .wLength       = 0,
        };

        if ((SubErrorCode = USB_Host_SendControlRequest(NULL)) != HOST_SENDCONTROL_Successful)
        {
            ErrorCode = HOST_ENUMERROR_ControlError;
            break;
        }

        HOST_TASK_NONBLOCK_WAIT(100, HOST_STATE_Default_PostAddressSet);
        break;
    case HOST_STATE_Default_PostAddressSet:
        USB_Host_SetDeviceAddress(USB_HOST_DEVICEADDRESS);

        EVENT_USB_Host_DeviceEnumerationComplete();
        USB_HostState = HOST_STATE_Addressed;
        break;
    }

    if ((ErrorCode != HOST_ENUMERROR_NoError) && (USB_HostState != HOST_STATE_Unattached))
    {
        EVENT_USB_Host_DeviceEnumerationFailed(ErrorCode, SubErrorCode);

        //USB_Host_VBUS_Auto_Off();
        USB_Host_VBUS_Manual_Off();
        //PORTE = 0;

        EVENT_USB_Host_DeviceUnattached();

        USB_ResetInterface();
    }
}

uint8_t USB_Host_WaitMS(uint8_t MS)
{
    bool    BusSuspended = USB_Host_IsBusSuspended();
    uint8_t ErrorCode    = HOST_WAITERROR_Successful;
    bool    HSOFIEnabled = USB_INT_IsEnabled(USB_INT_HSOFI);

    USB_INT_Disable(USB_INT_HSOFI);
    USB_INT_Clear(USB_INT_HSOFI);

    USB_Host_ResumeBus();

    while (MS)
    {
        if (USB_INT_HasOccurred(USB_INT_HSOFI))
        {
            USB_INT_Clear(USB_INT_HSOFI);
            MS--;
        }

        if ((USB_HostState == HOST_STATE_Unattached) || (USB_CurrentMode != USB_MODE_Host))
        {
            ErrorCode = HOST_WAITERROR_DeviceDisconnect;

            break;
        }

        if (Pipe_IsError() == true)
        {
            Pipe_ClearError();
            ErrorCode = HOST_WAITERROR_PipeError;

            break;
        }

        if (Pipe_IsStalled() == true)
        {
            Pipe_ClearStall();
            ErrorCode = HOST_WAITERROR_SetupStalled;

            break;
        }
    }

    if (BusSuspended)
        USB_Host_SuspendBus();

    if (HSOFIEnabled)
        USB_INT_Enable(USB_INT_HSOFI);

    return ErrorCode;
}

static void USB_Host_ResetDevice(void)
{
    bool BusSuspended = USB_Host_IsBusSuspended();

    USB_INT_Disable(USB_INT_DDISCI);

    USB_Host_ResetBus();
    while (!(USB_Host_IsBusResetComplete()));
    USB_Host_ResumeBus();

    bool HSOFIEnabled = USB_INT_IsEnabled(USB_INT_HSOFI);

    USB_INT_Disable(USB_INT_HSOFI);
    USB_INT_Clear(USB_INT_HSOFI);

    for (uint8_t MSRem = 10; MSRem != 0; MSRem--)
    {
        /* Workaround for powerless-pull-up devices. After a USB bus reset,
           all disconnection interrupts are suppressed while a USB frame is
           looked for - if it is found within 10ms, the device is still
           present.                                                        */

        if (USB_INT_HasOccurred(USB_INT_HSOFI))
        {
            USB_INT_Clear(USB_INT_HSOFI);
            USB_INT_Clear(USB_INT_DDISCI);
            break;
        }

        _delay_ms(1);
    }

    if (HSOFIEnabled)
        USB_INT_Enable(USB_INT_HSOFI);

    if (BusSuspended)
        USB_Host_SuspendBus();

    USB_INT_Enable(USB_INT_DDISCI);
}