📄 usbhost_lpc17xx.c
字号:
/*
**************************************************************************************************************
* NXP USB Host Stack
*
* (c) Copyright 2008, NXP SemiConductors
* (c) Copyright 2008, OnChip Technologies LLC
* All Rights Reserved
*
* www.nxp.com
* www.onchiptech.com
*
* File : usbhost_lpc17xx.c
* Programmer(s) : Ravikanth.P
* Version :
*
**************************************************************************************************************
*/
/*
**************************************************************************************************************
* INCLUDE HEADER FILES
**************************************************************************************************************
*/
#include "usbhost_lpc17xx.h"
/*
**************************************************************************************************************
* GLOBAL VARIABLES
**************************************************************************************************************
*/
int gUSBConnected;
volatile USB_INT32U HOST_RhscIntr = 0; /* Root Hub Status Change interrupt */
volatile USB_INT32U HOST_WdhIntr = 0; /* Semaphore to wait until the TD is submitted */
volatile USB_INT08U HOST_TDControlStatus = 0;
volatile HCED *EDCtrl; /* Control endpoint descriptor structure */
volatile HCED *EDBulkIn; /* BulkIn endpoint descriptor structure */
volatile HCED *EDBulkOut; /* BulkOut endpoint descriptor structure */
volatile HCTD *TDHead; /* Head transfer descriptor structure */
volatile HCTD *TDTail; /* Tail transfer descriptor structure */
volatile HCCA *Hcca; /* Host Controller Communications Area structure */
USB_INT16U *TDBufNonVol; /* Identical to TDBuffer just to reduce compiler warnings */
volatile USB_INT08U *TDBuffer; /* Current Buffer Pointer of transfer descriptor */
volatile USB_INT08U *FATBuffer; /* Buffer used by FAT file system */
volatile USB_INT08U *UserBuffer; /* Buffer used by application */
/*
**************************************************************************************************************
* DELAY IN MILLI SECONDS
*
* Description: This function provides a delay in milli seconds
*
* Arguments : delay The delay required
*
* Returns : None
*
**************************************************************************************************************
*/
void Host_DelayMS (USB_INT32U delay)
{
volatile USB_INT32U i;
for (i = 0; i < delay; i++) {
Host_DelayUS(1000);
}
}
/*
**************************************************************************************************************
* DELAY IN MICRO SECONDS
*
* Description: This function provides a delay in micro seconds
*
* Arguments : delay The delay required
*
* Returns : None
*
**************************************************************************************************************
*/
void Host_DelayUS (USB_INT32U delay)
{
volatile USB_INT32U i;
for (i = 0; i < (4 * delay); i++) { /* This logic was tested. It gives app. 1 micro sec delay */
;
}
}
/*
**************************************************************************************************************
* INITIALIZE THE HOST CONTROLLER
*
* Description: This function initializes lpc17xx host controller
*
* Arguments : None
*
* Returns :
*
**************************************************************************************************************
*/
void Host_Init (void)
{
USB_INT32U HostBaseAddr;
SC->PCONP |= (1UL<<31);
USB->OTGClkCtrl = 0x0000001F;
while ((USB->OTGClkSt & 0x0000001F) == 0) { /* Host clock is available */
;
}
USB->OTGStCtrl = 0x3;
/* P1[18] = USB_UP_LED, 01 */
/* P1[19] = /USB_PPWR, 10 */
/* P1[22] = USB_PWRD, 10 */
/* P1[27] = /USB_OVRCR, 10 */
PINCON->PINSEL3 &= ~((3<<4) | (3<<6) | (3<<12) | (3<<22));
PINCON->PINSEL3 |= ((1<<4)|(2<<6) | (2<<12) | (2<<22)); // 0x00802080
/* P0[29] = USB_D+, 01 */
/* P0[30] = USB_D-, 01 */
PINCON->PINSEL1 &= ~((3<<26) | (3<<28));
PINCON->PINSEL1 |= ((1<<26)|(1<<28)); // 0x14000000
PRINT_Log("Initializing Host Stack\n");
// For chip revision V01 and later
HostBaseAddr = 0x20080000;
Hcca = (volatile HCCA *)(HostBaseAddr+0x000);
TDHead = (volatile HCTD *)(HostBaseAddr+0x100);
TDTail = (volatile HCTD *)(HostBaseAddr+0x110);
EDCtrl = (volatile HCED *)(HostBaseAddr+0x120);
EDBulkIn = (volatile HCED *)(HostBaseAddr+0x130);
EDBulkOut = (volatile HCED *)(HostBaseAddr+0x140);
TDBuffer = (volatile USB_INT08U *)(HostBaseAddr+0x150);
FATBuffer = (volatile USB_INT08U *)(HostBaseAddr+0x1D0);
UserBuffer = (volatile USB_INT08U *)(HostBaseAddr+0x1000);
/* Initialize all the TDs, EDs and HCCA to 0 */
Host_EDInit(EDCtrl);
Host_EDInit(EDBulkIn);
Host_EDInit(EDBulkOut);
Host_TDInit(TDHead);
Host_TDInit(TDTail);
Host_HCCAInit(Hcca);
Host_DelayMS(50); /* Wait 50 ms before apply reset */
USB->HcControl = 0; /* HARDWARE RESET */
USB->HcControlHeadED = 0; /* Initialize Control list head to Zero */
USB->HcBulkHeadED = 0; /* Initialize Bulk list head to Zero */
/* SOFTWARE RESET */
USB->HcCommandStatus = OR_CMD_STATUS_HCR;
USB->HcFmInterval = DEFAULT_FMINTERVAL; /* Write Fm Interval and Largest Data Packet Counter */
/* Put HC in operational state */
USB->HcControl = (USB->HcControl & (~OR_CONTROL_HCFS)) | OR_CONTROL_HC_OPER;
USB->HcRhStatus = OR_RH_STATUS_LPSC; /* Set Global Power */
USB->HcHCCA = (USB_INT32U)Hcca;
USB->HcInterruptStatus |= USB->HcInterruptStatus; /* Clear Interrrupt Status */
USB->HcInterruptEnable = OR_INTR_ENABLE_MIE |
OR_INTR_ENABLE_WDH |
OR_INTR_ENABLE_RHSC;
/* Enable the USB Interrupt */
NVIC_EnableIRQ(USB_IRQn); /* enable USB interrupt */
NVIC_SetPriority (USB_IRQn, 0); /* highest priority */
PRINT_Log("Host Initialized\n");
}
/*
**************************************************************************************************************
* INTERRUPT SERVICE ROUTINE
*
* Description: This function services the interrupt caused by host controller
*
* Arguments : None
*
* Returns : None
*
**************************************************************************************************************
*/
void USB_IRQHandler (void)
{
USB_INT32U int_status;
USB_INT32U ie_status;
int_status = USB->HcInterruptStatus; /* Read Interrupt Status */
ie_status = USB->HcInterruptEnable; /* Read Interrupt enable status */
if (!(int_status & ie_status)) {
return;
} else {
int_status = int_status & ie_status;
if (int_status & OR_INTR_STATUS_RHSC) { /* Root hub status change interrupt */
if (USB->HcRhPortStatus1 & OR_RH_PORT_CSC) {
if (USB->HcRhStatus & OR_RH_STATUS_DRWE) {
/*
* When DRWE is on, Connect Status Change
* means a remote wakeup event.
*/
HOST_RhscIntr = 1;// JUST SOMETHING FOR A BREAKPOINT
}
else {
/*
* When DRWE is off, Connect Status Change
* is NOT a remote wakeup event
*/
if (USB->HcRhPortStatus1 & OR_RH_PORT_CCS) {
if (!gUSBConnected) {
HOST_TDControlStatus = 0;
HOST_WdhIntr = 0;
HOST_RhscIntr = 1;
gUSBConnected = 1;
}
else
PRINT_Log("Spurious status change (connected)?\n");
} else {
if (gUSBConnected) {
USB->HcInterruptEnable = 0; // why do we get multiple disc. rupts???
HOST_RhscIntr = 0;
gUSBConnected = 0;
}
else
PRINT_Log("Spurious status change (disconnected)?\n");
}
}
USB->HcRhPortStatus1 = OR_RH_PORT_CSC;
}
if (USB->HcRhPortStatus1 & OR_RH_PORT_PRSC) {
USB->HcRhPortStatus1 = OR_RH_PORT_PRSC;
}
}
if (int_status & OR_INTR_STATUS_WDH) { /* Writeback Done Head interrupt */
HOST_WdhIntr = 1;
HOST_TDControlStatus = (TDHead->Control >> 28) & 0xf;
}
USB->HcInterruptStatus = int_status; /* Clear interrupt status register */
}
return;
}
/*
**************************************************************************************************************
* PROCESS TRANSFER DESCRIPTOR
*
* Description: This function processes the transfer descriptor
*
* Arguments : ed Endpoint descriptor that contains this transfer descriptor
* token SETUP, IN, OUT
* buffer Current Buffer Pointer of the transfer descriptor
* buffer_len Length of the buffer
*
* Returns : OK if TD submission is successful
* ERROR if TD submission fails
*
**************************************************************************************************************
*/
USB_INT32S Host_ProcessTD (volatile HCED *ed,
volatile USB_INT32U token,
volatile USB_INT08U *buffer,
USB_INT32U buffer_len)
{
volatile USB_INT32U td_toggle;
if (ed == EDCtrl) {
if (token == TD_SETUP) {
td_toggle = TD_TOGGLE_0;
} else {
td_toggle = TD_TOGGLE_1;
}
} else {
td_toggle = 0;
}
TDHead->Control = (TD_ROUNDING |
token |
TD_DELAY_INT(0) |
td_toggle |
TD_CC);
TDTail->Control = 0;
TDHead->CurrBufPtr = (USB_INT32U) buffer;
TDTail->CurrBufPtr = 0;
TDHead->Next = (USB_INT32U) TDTail;
TDTail->Next = 0;
TDHead->BufEnd = (USB_INT32U)(buffer + (buffer_len - 1));
TDTail->BufEnd = 0;
ed->HeadTd = (USB_INT32U)TDHead | ((ed->HeadTd) & 0x00000002);
ed->TailTd = (USB_INT32U)TDTail;
ed->Next = 0;
if (ed == EDCtrl) {
USB->HcControlHeadED = (USB_INT32U)ed;
USB->HcCommandStatus = USB->HcCommandStatus | OR_CMD_STATUS_CLF;
USB->HcControl = USB->HcControl | OR_CONTROL_CLE;
} else {
USB->HcBulkHeadED = (USB_INT32U)ed;
USB->HcCommandStatus = USB->HcCommandStatus | OR_CMD_STATUS_BLF;
USB->HcControl = USB->HcControl | OR_CONTROL_BLE;
}
Host_WDHWait();
// if (!(TDHead->Control & 0xF0000000)) {
if (!HOST_TDControlStatus) {
return (OK);
} else {
return (ERR_TD_FAIL);
}
}
/*
**************************************************************************************************************
* ENUMERATE THE DEVICE
*
* Description: This function is used to enumerate the device connected
*
* Arguments : None
*
* Returns : None
*
**************************************************************************************************************
*/
USB_INT32S Host_EnumDev (void)
{
USB_INT32S rc;
PRINT_Log("Connect a Mass Storage device\n");
while (!HOST_RhscIntr);
Host_DelayMS(100); /* USB 2.0 spec says atleast 50ms delay beore port reset */
USB->HcRhPortStatus1 = OR_RH_PORT_PRS; // Initiate port reset
while (USB->HcRhPortStatus1 & OR_RH_PORT_PRS)
; // Wait for port reset to complete...
USB->HcRhPortStatus1 = OR_RH_PORT_PRSC; // ...and clear port reset signal
Host_DelayMS(200); /* Wait for 100 MS after port reset */
EDCtrl->Control = 8 << 16; /* Put max pkt size = 8 */
/* Read first 8 bytes of device desc */
rc = HOST_GET_DESCRIPTOR(USB_DESCRIPTOR_TYPE_DEVICE, 0, TDBuffer, 8);
if (rc != OK) {
PRINT_Err(rc);
return (rc);
}
EDCtrl->Control = TDBuffer[7] << 16; /* Get max pkt size of endpoint 0 */
rc = HOST_SET_ADDRESS(1); /* Set the device address to 1 */
if (rc != OK) {
PRINT_Err(rc);
return (rc);
}
Host_DelayMS(2);
EDCtrl->Control = (EDCtrl->Control) | 1; /* Modify control pipe with address 1 */
/* Get the configuration descriptor */
rc = HOST_GET_DESCRIPTOR(USB_DESCRIPTOR_TYPE_CONFIGURATION, 0, TDBuffer, 9);
if (rc != OK) {
PRINT_Err(rc);
return (rc);
}
/* Get the first configuration data */
rc = HOST_GET_DESCRIPTOR(USB_DESCRIPTOR_TYPE_CONFIGURATION, 0, TDBuffer, ReadLE16U(&TDBuffer[2]));
if (rc != OK) {
PRINT_Err(rc);
return (rc);
}
rc = MS_ParseConfiguration(); /* Parse the configuration */
if (rc != OK) {
PRINT_Err(rc);
return (rc);
}
rc = USBH_SET_CONFIGURATION(1); /* Select device configuration 1 */
if (rc != OK) {
PRINT_Err(rc);
}
Host_DelayMS(100); /* Some devices may require this delay */
return (rc);
}
/*
**************************************************************************************************************
* RECEIVE THE CONTROL INFORMATION
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -