📄 usbhw.c
字号:
USB_CTRL = ((EPNum & 0x0F) << 2) | CTRL_WR_EN;
TX_PLENGTH = cnt;
for (n = 0; n < (cnt + 3) / 4; n++) {
TX_DATA = *((__packed DWORD *)pData);
pData += 4;
}
USB_CTRL = 0;
WrCmd(CMD_SEL_EP(EPAdr(EPNum)));
WrCmd(CMD_VALID_BUF);
return (cnt);
}
/***************************************************************************************************************
** 函数名称 : USB_ReadEndpoint() Name : USB_WriteEndpoint()
** 功能描述 : 向物理端点写入数据 Function : Write data to USB endpoint
** 入口参数 : INT8U endp: 物理端点号 Input : INT8U endp: the physical endpoint number
INT8U len : 要写的字节长度 INT8U len : the length that will be written
INT8U *buf: 发送缓冲区 INT8U *buf: sending buffer
** 输 出 : 实际写入的字节数 Output : the actual length that have being written
****************************************************************************************************************/
INT8U USB_WriteEndpoint(INT8U endp, INT8U len, INT8U *buf)
{
INT32U tmp;
INT32U i,j;
/* select logical endpoint and enable writing operation */
tmp = ((endp >> 1) << 2) | 0x02; /* 选择逻辑端点并使能写操作 */
USB_CTRL = (INT32U)tmp;
/* write the length that will be written into endpoint */
TX_PLENGTH = (INT32U)len; /* 写入要发送的数据的字节长度 */
/* write Data into Transmit Data Register */ /* 写入要发送的数据到发送数据寄存器 */
j = len >> 2;
for(i = 1; i <= j; i++)
{
tmp = (INT32U)*buf++;
tmp += (INT32U)(*buf++ << 8);
tmp += (INT32U)(*buf++ << 16);
tmp += (INT32U)(*buf++ << 24);
TX_DATA = tmp;
while((TX_PLENGTH & 0x3FF) != len - (i << 2));
}
j = len % 4;
if (j != 0)
{
tmp = 0;
for(i = 0; i < j; i++)
{
tmp |= ((INT32U)*buf) << (i << 3);
buf++;
}
TX_DATA = tmp;
while((TX_PLENGTH & 0x3FF) != 0); /* 等待 USBTxPLen 为 0 */
}
if (len != 0)
{
while((DEV_INT_STAT & 0x00000080) ==0); /*TxENDPKT 等待写入完成 */
DEV_INT_CLR = 0x00000080;
}
/* Validate buffer */
// USB_SelectEndpoint(endp); /* 使能发送缓冲区 */
// USB_ValidBuffer();
WrCmd(CMD_SEL_EP(EPAdr(endp)));
WrCmd(CMD_VALID_BUF);
USB_CTRL = 0;
return len; /* 返回实际写入长度 */
}
#if USB_DMA
/* DMA Descriptor Memory Layout */
const DWORD DDAdr[2] = { DD_NISO_ADR, DD_ISO_ADR };
const DWORD DDSz [2] = { 16, 20 };
/*
* Setup USB DMA Transfer for selected Endpoint
* Parameters: EPNum: Endpoint Number
* pDD: Pointer to DMA Descriptor
* Return Value: TRUE - Success, FALSE - Error
*/
BOOL USB_DMA_Setup(DWORD EPNum, USB_DMA_DESCRIPTOR *pDD) {
DWORD num, ptr, nxt, iso, n;
iso = pDD->Cfg.Type.IsoEP; /* Iso or Non-Iso Descriptor */
num = EPAdr(EPNum); /* Endpoint's Physical Address */
ptr = 0; /* Current Descriptor */
nxt = udca[num]; /* Initial Descriptor */
while (nxt) { /* Go through Descriptor List */
ptr = nxt; /* Current Descriptor */
if (!pDD->Cfg.Type.Link) { /* Check for Linked Descriptors */
n = (ptr - DDAdr[iso]) / DDSz[iso]; /* Descriptor Index */
DDMemMap[iso] &= ~(1 << n); /* Unmark Memory Usage */
}
nxt = *((DWORD *)ptr); /* Next Descriptor */
}
for (n = 0; n < 32; n++) { /* Search for available Memory */
if ((DDMemMap[iso] & (1 << n)) == 0) {
break; /* Memory found */
}
}
if (n == 32) return (FALSE); /* Memory not available */
DDMemMap[iso] |= 1 << n; /* Mark Memory Usage */
nxt = DDAdr[iso] + n * DDSz[iso]; /* Next Descriptor */
if (ptr && pDD->Cfg.Type.Link) {
*((DWORD *)(ptr + 0)) = nxt; /* Link in new Descriptor */
*((DWORD *)(ptr + 4)) |= 0x00000004; /* Next DD is Valid */
} else {
udca[num] = nxt; /* Save new Descriptor */
UDCA[num] = nxt; /* Update UDCA in USB */
}
/* Fill in DMA Descriptor */
*(((DWORD *)nxt)++) = 0; /* Next DD Pointer */
*(((DWORD *)nxt)++) = pDD->Cfg.Type.ATLE |
(pDD->Cfg.Type.IsoEP << 4) |
(pDD->MaxSize << 5) |
(pDD->BufLen << 16);
*(((DWORD *)nxt)++) = pDD->BufAdr;
*(((DWORD *)nxt)++) = pDD->Cfg.Type.LenPos << 8;
if (iso) {
*((DWORD *)nxt) = pDD->InfoAdr;
}
return (TRUE); /* Success */
}
/*
* Enable USB DMA Endpoint
* Parameters: EPNum: Endpoint Number
* EPNum.0..3: Address
* EPNum.7: Dir
* Return Value: None
*/
void USB_DMA_Enable (DWORD EPNum) {
EP_DMA_EN = 1 << EPAdr(EPNum);
}
/*
* Disable USB DMA Endpoint
* Parameters: EPNum: Endpoint Number
* EPNum.0..3: Address
* EPNum.7: Dir
* Return Value: None
*/
void USB_DMA_Disable (DWORD EPNum) {
EP_DMA_DIS = 1 << EPAdr(EPNum);
}
/*
* Get USB DMA Endpoint Status
* Parameters: EPNum: Endpoint Number
* EPNum.0..3: Address
* EPNum.7: Dir
* Return Value: DMA Status
*/
DWORD USB_DMA_Status (DWORD EPNum) {
DWORD ptr, val;
ptr = UDCA[EPAdr(EPNum)]; /* Current Descriptor */
if (ptr == 0) return (USB_DMA_INVALID);
val = *((DWORD *)(ptr + 3*4)); /* Status Information */
switch ((val >> 1) & 0x0F) {
case 0x00: /* Not serviced */
return (USB_DMA_IDLE);
case 0x01: /* Being serviced */
return (USB_DMA_BUSY);
case 0x02: /* Normal Completition */
return (USB_DMA_DONE);
case 0x03: /* Data Under Run */
return (USB_DMA_UNDER_RUN);
case 0x08: /* Data Over Run */
return (USB_DMA_OVER_RUN);
case 0x09: /* System Error */
return (USB_DMA_ERROR);
}
return (USB_DMA_UNKNOWN);
}
/*
* Get USB DMA Endpoint Current Buffer Address
* Parameters: EPNum: Endpoint Number
* EPNum.0..3: Address
* EPNum.7: Dir
* Return Value: DMA Address (or -1 when DMA is Invalid)
*/
DWORD USB_DMA_BufAdr (DWORD EPNum) {
DWORD ptr, val;
ptr = UDCA[EPAdr(EPNum)]; /* Current Descriptor */
if (ptr == 0) return ((DWORD)(-1)); /* DMA Invalid */
val = *((DWORD *)(ptr + 2*4)); /* Buffer Address */
return (val); /* Current Address */
}
/*
* Get USB DMA Endpoint Current Buffer Count
* Number of transfered Bytes or Iso Packets
* Parameters: EPNum: Endpoint Number
* EPNum.0..3: Address
* EPNum.7: Dir
* Return Value: DMA Count (or -1 when DMA is Invalid)
*/
DWORD USB_DMA_BufCnt (DWORD EPNum) {
DWORD ptr, val;
ptr = UDCA[EPAdr(EPNum)]; /* Current Descriptor */
if (ptr == 0) return ((DWORD)(-1)); /* DMA Invalid */
val = *((DWORD *)(ptr + 3*4)); /* Status Information */
return (val >> 16); /* Current Count */
}
#endif /* USB_DMA */
/*
* Get USB Last Frame Number
* Parameters: None
* Return Value: Frame Number
*/
DWORD USB_GetFrame (void) {
DWORD val;
WrCmd(CMD_RD_FRAME);
val = RdCmdDat(DAT_RD_FRAME);
val = val | (RdCmdDat(DAT_RD_FRAME) << 8);
return (val);
}
/*
* USB Interrupt Service Routine
*/
void USB_ISR (void) __irq {
DWORD disr, val, n, m;
disr = DEV_INT_STAT; /* Device Interrupt Status */
DEV_INT_CLR = disr; /* A known issue on LPC214x */
/* Device Status Interrupt (Reset, Connect change, Suspend/Resume) */
if (disr & DEV_STAT_INT) {
WrCmd(CMD_GET_DEV_STAT);
val = RdCmdDat(DAT_GET_DEV_STAT); /* Device Status */
if (val & DEV_RST) { /* Reset */
USB_Reset();
#if USB_RESET_EVENT
USB_Reset_Event();
#endif
goto isr_end;
}
if (val & DEV_CON_CH) { /* Connect change */
#if USB_POWER_EVENT
USB_Power_Event(val & DEV_CON);
#endif
goto isr_end;
}
if (val & DEV_SUS_CH) { /* Suspend/Resume */
if (val & DEV_SUS) { /* Suspend */
USB_Suspend();
#if USB_SUSPEND_EVENT
USB_Suspend_Event();
#endif
} else { /* Resume */
USB_Resume();
#if USB_RESUME_EVENT
USB_Resume_Event();
#endif
}
goto isr_end;
}
}
#if USB_SOF_EVENT
/* Start of Frame Interrupt */
if (disr & FRAME_INT) {
USB_SOF_Event();
}
#endif
#if USB_ERROR_EVENT
/* Error Interrupt */
if (disr & ERR_INT) {
WrCmd(CMD_RD_ERR_STAT);
val = RdCmdDat(DAT_RD_ERR_STAT);
USB_Error_Event(val);
}
#endif
/* Endpoint's Slow Interrupt */
if (disr & EP_SLOW_INT) {
while (EP_INT_STAT) { /* Endpoint Interrupt Status */
for (n = 0; n < USB_EP_NUM; n++) { /* Check All Endpoints */
if (EP_INT_STAT & (1 << n)) {
m = n >> 1;
EP_INT_CLR = 1 << n;
while ((DEV_INT_STAT & CDFULL_INT) == 0);
val = CMD_DATA;
if ((n & 1) == 0) { /* OUT Endpoint */
if (n == 0) { /* Control OUT Endpoint */
if (val & EP_SEL_STP) { /* Setup Packet */
if (USB_P_EP[0]) {
USB_P_EP[0](USB_EVT_SETUP);
continue;
}
}
}
if (USB_P_EP[m]) {
USB_P_EP[m](USB_EVT_OUT);
}
} else { /* IN Endpoint */
if (USB_P_EP[m]) {
USB_P_EP[m](USB_EVT_IN);
}
}
}
}
}
}
#if USB_DMA
if (DMA_INT_STAT & 0x00000001) { /* End of Transfer Interrupt */
val = EOT_INT_STAT;
for (n = 2; n < USB_EP_NUM; n++) { /* Check All Endpoints */
if (val & (1 << n)) {
m = n >> 1;
if ((n & 1) == 0) { /* OUT Endpoint */
if (USB_P_EP[m]) {
USB_P_EP[m](USB_EVT_OUT_DMA_EOT);
}
} else { /* IN Endpoint */
if (USB_P_EP[m]) {
USB_P_EP[m](USB_EVT_IN_DMA_EOT);
}
}
}
}
EOT_INT_CLR = val;
}
if (DMA_INT_STAT & 0x00000002) { /* New DD Request Interrupt */
val = NDD_REQ_INT_STAT;
for (n = 2; n < USB_EP_NUM; n++) { /* Check All Endpoints */
if (val & (1 << n)) {
m = n >> 1;
if ((n & 1) == 0) { /* OUT Endpoint */
if (USB_P_EP[m]) {
USB_P_EP[m](USB_EVT_OUT_DMA_NDR);
}
} else { /* IN Endpoint */
if (USB_P_EP[m]) {
USB_P_EP[m](USB_EVT_IN_DMA_NDR);
}
}
}
}
NDD_REQ_INT_CLR = val;
}
if (DMA_INT_STAT & 0x00000004) { /* System Error Interrupt */
val = SYS_ERR_INT_STAT;
for (n = 2; n < USB_EP_NUM; n++) { /* Check All Endpoints */
if (val & (1 << n)) {
m = n >> 1;
if ((n & 1) == 0) { /* OUT Endpoint */
if (USB_P_EP[m]) {
USB_P_EP[m](USB_EVT_OUT_DMA_ERR);
}
} else { /* IN Endpoint */
if (USB_P_EP[m]) {
USB_P_EP[m](USB_EVT_IN_DMA_ERR);
}
}
}
}
SYS_ERR_INT_CLR = val;
}
#endif /* USB_DMA */
isr_end:
// DEV_INT_CLR = disr;
VICVectAddr = 0; /* Acknowledge Interrupt */
}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -