726 lines
20 KiB
C
726 lines
20 KiB
C
|
/**************************************************************************//**
|
||
|
* @file hsusbd.c
|
||
|
* @version V1.00
|
||
|
* @brief M480 HSUSBD driver source file
|
||
|
*
|
||
|
* SPDX-License-Identifier: Apache-2.0
|
||
|
* @copyright (C) 2016-2020 Nuvoton Technology Corp. All rights reserved.
|
||
|
*****************************************************************************/
|
||
|
#include <stdio.h>
|
||
|
#include "NuMicro.h"
|
||
|
|
||
|
/** @addtogroup Standard_Driver Standard Driver
|
||
|
@{
|
||
|
*/
|
||
|
|
||
|
/** @addtogroup HSUSBD_Driver HSUSBD Driver
|
||
|
@{
|
||
|
*/
|
||
|
|
||
|
|
||
|
/** @addtogroup HSUSBD_EXPORTED_FUNCTIONS HSUSBD Exported Functions
|
||
|
@{
|
||
|
*/
|
||
|
/*--------------------------------------------------------------------------*/
|
||
|
/** @cond HIDDEN_SYMBOLS */
|
||
|
/* Global variables for Control Pipe */
|
||
|
S_HSUSBD_CMD_T gUsbCmd;
|
||
|
S_HSUSBD_INFO_T *g_hsusbd_sInfo;
|
||
|
|
||
|
HSUSBD_VENDOR_REQ g_hsusbd_pfnVendorRequest = NULL;
|
||
|
HSUSBD_CLASS_REQ g_hsusbd_pfnClassRequest = NULL;
|
||
|
HSUSBD_SET_INTERFACE_REQ g_hsusbd_pfnSetInterface = NULL;
|
||
|
uint32_t g_u32HsEpStallLock = 0ul; /* Bit map flag to lock specified EP when SET_FEATURE */
|
||
|
|
||
|
static uint8_t *g_hsusbd_CtrlInPointer = 0;
|
||
|
static uint32_t g_hsusbd_CtrlMaxPktSize = 64ul;
|
||
|
static uint8_t g_hsusbd_UsbConfig = 0ul;
|
||
|
static uint8_t g_hsusbd_UsbAltInterface = 0ul;
|
||
|
static uint8_t g_hsusbd_EnableTestMode = 0ul;
|
||
|
static uint8_t g_hsusbd_TestSelector = 0ul;
|
||
|
|
||
|
#ifdef __ICCARM__
|
||
|
#pragma data_alignment=4
|
||
|
static uint8_t g_hsusbd_buf[12];
|
||
|
#else
|
||
|
static uint8_t g_hsusbd_buf[12] __attribute__((aligned(4)));
|
||
|
#endif
|
||
|
|
||
|
uint8_t volatile g_hsusbd_Configured = 0ul;
|
||
|
uint8_t g_hsusbd_CtrlZero = 0ul;
|
||
|
uint8_t g_hsusbd_UsbAddr = 0ul;
|
||
|
uint8_t g_hsusbd_ShortPacket = 0ul;
|
||
|
uint32_t volatile g_hsusbd_DmaDone = 0ul;
|
||
|
uint32_t g_hsusbd_CtrlInSize = 0ul;
|
||
|
/** @endcond HIDDEN_SYMBOLS */
|
||
|
|
||
|
/**
|
||
|
* @brief HSUSBD Initial
|
||
|
*
|
||
|
* @param[in] param Descriptor
|
||
|
* @param[in] pfnClassReq Class Request Callback Function
|
||
|
* @param[in] pfnSetInterface SetInterface Request Callback Function
|
||
|
*
|
||
|
* @return None
|
||
|
*
|
||
|
* @details This function is used to initial HSUSBD.
|
||
|
*/
|
||
|
void HSUSBD_Open(S_HSUSBD_INFO_T *param, HSUSBD_CLASS_REQ pfnClassReq, HSUSBD_SET_INTERFACE_REQ pfnSetInterface)
|
||
|
{
|
||
|
g_hsusbd_sInfo = param;
|
||
|
g_hsusbd_pfnClassRequest = pfnClassReq;
|
||
|
g_hsusbd_pfnSetInterface = pfnSetInterface;
|
||
|
|
||
|
/* get EP0 maximum packet size */
|
||
|
g_hsusbd_CtrlMaxPktSize = g_hsusbd_sInfo->gu8DevDesc[7];
|
||
|
|
||
|
/* Initial USB engine */
|
||
|
//HSUSBD->PHYCTL |= (HSUSBD_PHYCTL_PHYEN_Msk | HSUSBD_PHYCTL_DPPUEN_Msk);
|
||
|
HSUSBD_ENABLE_PHY();
|
||
|
while((HSUSBD->BUSINTSTS & HSUSBD_BUSINTEN_PHYCLKVLDIEN_Msk) != HSUSBD_BUSINTEN_PHYCLKVLDIEN_Msk ){}
|
||
|
HSUSBD_CLR_SE0();
|
||
|
|
||
|
/* wait PHY clock ready */
|
||
|
while (1)
|
||
|
{
|
||
|
HSUSBD->EP[EPA].EPMPS = 0x20ul;
|
||
|
if (HSUSBD->EP[EPA].EPMPS == 0x20ul)
|
||
|
{
|
||
|
HSUSBD->EP[EPA].EPMPS = 0x0ul;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
/* Force SE0, and then clear it to connect*/
|
||
|
HSUSBD_SET_SE0();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief HSUSBD Start
|
||
|
*
|
||
|
* @param[in] None
|
||
|
*
|
||
|
* @return None
|
||
|
*
|
||
|
* @details This function is used to start transfer
|
||
|
*/
|
||
|
void HSUSBD_Start(void)
|
||
|
{
|
||
|
HSUSBD_CLR_SE0();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Process Setup Packet
|
||
|
*
|
||
|
* @param[in] None
|
||
|
*
|
||
|
* @return None
|
||
|
*
|
||
|
* @details This function is used to process Setup packet.
|
||
|
*/
|
||
|
void HSUSBD_ProcessSetupPacket(void)
|
||
|
{
|
||
|
/* Setup packet process */
|
||
|
gUsbCmd.bmRequestType = (uint8_t)(HSUSBD->SETUP1_0 & 0xfful);
|
||
|
gUsbCmd.bRequest = (uint8_t)((HSUSBD->SETUP1_0 >> 8) & 0xfful);
|
||
|
gUsbCmd.wValue = (uint16_t)HSUSBD->SETUP3_2;
|
||
|
gUsbCmd.wIndex = (uint16_t)HSUSBD->SETUP5_4;
|
||
|
gUsbCmd.wLength = (uint16_t)HSUSBD->SETUP7_6;
|
||
|
|
||
|
/* USB device request in setup packet: offset 0, D[6..5]: 0=Standard, 1=Class, 2=Vendor, 3=Reserved */
|
||
|
switch (gUsbCmd.bmRequestType & 0x60ul)
|
||
|
{
|
||
|
case REQ_STANDARD:
|
||
|
{
|
||
|
HSUSBD_StandardRequest();
|
||
|
break;
|
||
|
}
|
||
|
case REQ_CLASS:
|
||
|
{
|
||
|
if (g_hsusbd_pfnClassRequest != NULL)
|
||
|
{
|
||
|
g_hsusbd_pfnClassRequest();
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
case REQ_VENDOR:
|
||
|
{
|
||
|
if (g_hsusbd_pfnVendorRequest != NULL)
|
||
|
{
|
||
|
g_hsusbd_pfnVendorRequest();
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
default:
|
||
|
{
|
||
|
/* Setup error, stall the device */
|
||
|
HSUSBD_SET_CEP_STATE(HSUSBD_CEPCTL_STALLEN_Msk);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Get Descriptor request
|
||
|
*
|
||
|
* @param[in] None
|
||
|
*
|
||
|
* @return None
|
||
|
*
|
||
|
* @details This function is used to process GetDescriptor request.
|
||
|
*/
|
||
|
int HSUSBD_GetDescriptor(void)
|
||
|
{
|
||
|
uint32_t u32Len;
|
||
|
int val = 0;
|
||
|
|
||
|
u32Len = gUsbCmd.wLength;
|
||
|
g_hsusbd_CtrlZero = (uint8_t)0ul;
|
||
|
|
||
|
switch ((gUsbCmd.wValue & 0xff00ul) >> 8)
|
||
|
{
|
||
|
/* Get Device Descriptor */
|
||
|
case DESC_DEVICE:
|
||
|
{
|
||
|
u32Len = Minimum(u32Len, LEN_DEVICE);
|
||
|
HSUSBD_PrepareCtrlIn((uint8_t *)g_hsusbd_sInfo->gu8DevDesc, u32Len);
|
||
|
break;
|
||
|
}
|
||
|
/* Get Configuration Descriptor */
|
||
|
case DESC_CONFIG:
|
||
|
{
|
||
|
uint32_t u32TotalLen;
|
||
|
if ((HSUSBD->OPER & 0x04ul) == 0x04ul)
|
||
|
{
|
||
|
u32TotalLen = g_hsusbd_sInfo->gu8ConfigDesc[3];
|
||
|
u32TotalLen = g_hsusbd_sInfo->gu8ConfigDesc[2] + (u32TotalLen << 8);
|
||
|
|
||
|
if (u32Len > u32TotalLen)
|
||
|
{
|
||
|
u32Len = u32TotalLen;
|
||
|
if ((u32Len % g_hsusbd_CtrlMaxPktSize) == 0ul)
|
||
|
{
|
||
|
g_hsusbd_CtrlZero = (uint8_t)1ul;
|
||
|
}
|
||
|
}
|
||
|
HSUSBD_PrepareCtrlIn((uint8_t *)g_hsusbd_sInfo->gu8ConfigDesc, u32Len);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
u32TotalLen = g_hsusbd_sInfo->gu8FullConfigDesc[3];
|
||
|
u32TotalLen = g_hsusbd_sInfo->gu8FullConfigDesc[2] + (u32TotalLen << 8);
|
||
|
|
||
|
if (u32Len > u32TotalLen)
|
||
|
{
|
||
|
u32Len = u32TotalLen;
|
||
|
if ((u32Len % g_hsusbd_CtrlMaxPktSize) == 0ul)
|
||
|
{
|
||
|
g_hsusbd_CtrlZero = (uint8_t)1ul;
|
||
|
}
|
||
|
}
|
||
|
HSUSBD_PrepareCtrlIn((uint8_t *)g_hsusbd_sInfo->gu8FullConfigDesc, u32Len);
|
||
|
}
|
||
|
|
||
|
break;
|
||
|
}
|
||
|
/* Get Qualifier Descriptor */
|
||
|
case DESC_QUALIFIER:
|
||
|
{
|
||
|
u32Len = Minimum(u32Len, LEN_QUALIFIER);
|
||
|
HSUSBD_PrepareCtrlIn((uint8_t *)g_hsusbd_sInfo->gu8QualDesc, u32Len);
|
||
|
break;
|
||
|
}
|
||
|
/* Get Other Speed Descriptor - Full speed */
|
||
|
case DESC_OTHERSPEED:
|
||
|
{
|
||
|
uint32_t u32TotalLen;
|
||
|
if ((HSUSBD->OPER & 0x04ul) == 0x04ul)
|
||
|
{
|
||
|
u32TotalLen = g_hsusbd_sInfo->gu8HSOtherConfigDesc[3];
|
||
|
u32TotalLen = g_hsusbd_sInfo->gu8HSOtherConfigDesc[2] + (u32TotalLen << 8);
|
||
|
|
||
|
if (u32Len > u32TotalLen)
|
||
|
{
|
||
|
u32Len = u32TotalLen;
|
||
|
if ((u32Len % g_hsusbd_CtrlMaxPktSize) == 0ul)
|
||
|
{
|
||
|
g_hsusbd_CtrlZero = (uint8_t)1ul;
|
||
|
}
|
||
|
}
|
||
|
HSUSBD_PrepareCtrlIn((uint8_t *)g_hsusbd_sInfo->gu8HSOtherConfigDesc, u32Len);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
u32TotalLen = g_hsusbd_sInfo->gu8FSOtherConfigDesc[3];
|
||
|
u32TotalLen = g_hsusbd_sInfo->gu8FSOtherConfigDesc[2] + (u32TotalLen << 8);
|
||
|
|
||
|
if (u32Len > u32TotalLen)
|
||
|
{
|
||
|
u32Len = u32TotalLen;
|
||
|
if ((u32Len % g_hsusbd_CtrlMaxPktSize) == 0ul)
|
||
|
{
|
||
|
g_hsusbd_CtrlZero = (uint8_t)1ul;
|
||
|
}
|
||
|
}
|
||
|
HSUSBD_PrepareCtrlIn((uint8_t *)g_hsusbd_sInfo->gu8FSOtherConfigDesc, u32Len);
|
||
|
}
|
||
|
|
||
|
break;
|
||
|
}
|
||
|
/* Get HID Descriptor */
|
||
|
case DESC_HID:
|
||
|
{
|
||
|
uint32_t u32ConfigDescOffset; /* u32ConfigDescOffset is configuration descriptor offset (HID descriptor start index) */
|
||
|
u32Len = Minimum(u32Len, LEN_HID);
|
||
|
u32ConfigDescOffset = g_hsusbd_sInfo->gu32ConfigHidDescIdx[gUsbCmd.wIndex & 0xfful];
|
||
|
HSUSBD_PrepareCtrlIn((uint8_t *)&g_hsusbd_sInfo->gu8ConfigDesc[u32ConfigDescOffset], u32Len);
|
||
|
break;
|
||
|
}
|
||
|
/* Get Report Descriptor */
|
||
|
case DESC_HID_RPT:
|
||
|
{
|
||
|
if (u32Len > g_hsusbd_sInfo->gu32HidReportSize[gUsbCmd.wIndex & 0xfful])
|
||
|
{
|
||
|
u32Len = g_hsusbd_sInfo->gu32HidReportSize[gUsbCmd.wIndex & 0xfful];
|
||
|
if ((u32Len % g_hsusbd_CtrlMaxPktSize) == 0ul)
|
||
|
{
|
||
|
g_hsusbd_CtrlZero = (uint8_t)1ul;
|
||
|
}
|
||
|
}
|
||
|
HSUSBD_PrepareCtrlIn((uint8_t *)g_hsusbd_sInfo->gu8HidReportDesc[gUsbCmd.wIndex & 0xfful], u32Len);
|
||
|
break;
|
||
|
}
|
||
|
/* Get String Descriptor */
|
||
|
case DESC_STRING:
|
||
|
{
|
||
|
if((gUsbCmd.wValue & 0xfful) < 8ul)
|
||
|
{
|
||
|
if (u32Len > g_hsusbd_sInfo->gu8StringDesc[gUsbCmd.wValue & 0xfful][0])
|
||
|
{
|
||
|
u32Len = g_hsusbd_sInfo->gu8StringDesc[gUsbCmd.wValue & 0xfful][0];
|
||
|
if ((u32Len % g_hsusbd_CtrlMaxPktSize) == 0ul)
|
||
|
{
|
||
|
g_hsusbd_CtrlZero = (uint8_t)1ul;
|
||
|
}
|
||
|
}
|
||
|
HSUSBD_PrepareCtrlIn((uint8_t *)g_hsusbd_sInfo->gu8StringDesc[gUsbCmd.wValue & 0xfful], u32Len);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
HSUSBD_SET_CEP_STATE(HSUSBD_CEPCTL_STALLEN_Msk);
|
||
|
val = 1;
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
default:
|
||
|
/* Not support. Reply STALL. */
|
||
|
HSUSBD_SET_CEP_STATE(HSUSBD_CEPCTL_STALLEN_Msk);
|
||
|
val = 1;
|
||
|
break;
|
||
|
}
|
||
|
return val;
|
||
|
}
|
||
|
|
||
|
|
||
|
/**
|
||
|
* @brief Process USB standard request
|
||
|
*
|
||
|
* @param[in] None
|
||
|
*
|
||
|
* @return None
|
||
|
*
|
||
|
* @details This function is used to process USB Standard Request.
|
||
|
*/
|
||
|
void HSUSBD_StandardRequest(void)
|
||
|
{
|
||
|
/* clear global variables for new request */
|
||
|
g_hsusbd_CtrlInPointer = 0;
|
||
|
g_hsusbd_CtrlInSize = 0ul;
|
||
|
|
||
|
if ((gUsbCmd.bmRequestType & 0x80ul) == 0x80ul) /* request data transfer direction */
|
||
|
{
|
||
|
/* Device to host */
|
||
|
switch (gUsbCmd.bRequest)
|
||
|
{
|
||
|
case GET_CONFIGURATION:
|
||
|
{
|
||
|
/* Return current configuration setting */
|
||
|
HSUSBD_PrepareCtrlIn((uint8_t *)&g_hsusbd_UsbConfig, 1ul);
|
||
|
|
||
|
HSUSBD_CLR_CEP_INT_FLAG(HSUSBD_CEPINTSTS_INTKIF_Msk);
|
||
|
HSUSBD_ENABLE_CEP_INT(HSUSBD_CEPINTEN_INTKIEN_Msk);
|
||
|
break;
|
||
|
}
|
||
|
case GET_DESCRIPTOR:
|
||
|
{
|
||
|
if (!HSUSBD_GetDescriptor())
|
||
|
{
|
||
|
HSUSBD_CLR_CEP_INT_FLAG(HSUSBD_CEPINTSTS_INTKIF_Msk);
|
||
|
HSUSBD_ENABLE_CEP_INT(HSUSBD_CEPINTEN_INTKIEN_Msk);
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
case GET_INTERFACE:
|
||
|
{
|
||
|
/* Return current interface setting */
|
||
|
HSUSBD_PrepareCtrlIn((uint8_t *)&g_hsusbd_UsbAltInterface, 1ul);
|
||
|
|
||
|
HSUSBD_CLR_CEP_INT_FLAG(HSUSBD_CEPINTSTS_INTKIF_Msk);
|
||
|
HSUSBD_ENABLE_CEP_INT(HSUSBD_CEPINTEN_INTKIEN_Msk);
|
||
|
break;
|
||
|
}
|
||
|
case GET_STATUS:
|
||
|
{
|
||
|
/* Device */
|
||
|
if (gUsbCmd.bmRequestType == 0x80ul)
|
||
|
{
|
||
|
if ((g_hsusbd_sInfo->gu8ConfigDesc[7] & 0x40ul) == 0x40ul)
|
||
|
{
|
||
|
g_hsusbd_buf[0] = (uint8_t)1ul; /* Self-Powered */
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
g_hsusbd_buf[0] = (uint8_t)0ul; /* bus-Powered */
|
||
|
}
|
||
|
}
|
||
|
/* Interface */
|
||
|
else if (gUsbCmd.bmRequestType == 0x81ul)
|
||
|
{
|
||
|
g_hsusbd_buf[0] = (uint8_t)0ul;
|
||
|
}
|
||
|
/* Endpoint */
|
||
|
else if (gUsbCmd.bmRequestType == 0x82ul)
|
||
|
{
|
||
|
uint8_t ep = (uint8_t)(gUsbCmd.wIndex & 0xFul);
|
||
|
g_hsusbd_buf[0] = (uint8_t)HSUSBD_GetStall((uint32_t)ep)? (uint8_t)1 : (uint8_t)0;
|
||
|
}
|
||
|
g_hsusbd_buf[1] = (uint8_t)0ul;
|
||
|
HSUSBD_PrepareCtrlIn(g_hsusbd_buf, 2ul);
|
||
|
HSUSBD_CLR_CEP_INT_FLAG(HSUSBD_CEPINTSTS_INTKIF_Msk);
|
||
|
HSUSBD_ENABLE_CEP_INT(HSUSBD_CEPINTEN_INTKIEN_Msk);
|
||
|
break;
|
||
|
}
|
||
|
default:
|
||
|
{
|
||
|
/* Setup error, stall the device */
|
||
|
HSUSBD_SET_CEP_STATE(HSUSBD_CEPCTL_STALLEN_Msk);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
/* Host to device */
|
||
|
switch (gUsbCmd.bRequest)
|
||
|
{
|
||
|
case CLEAR_FEATURE:
|
||
|
{
|
||
|
if((gUsbCmd.wValue & 0xfful) == FEATURE_ENDPOINT_HALT)
|
||
|
{
|
||
|
|
||
|
uint32_t epNum, i;
|
||
|
|
||
|
/* EP number stall is not allow to be clear in MSC class "Error Recovery Test".
|
||
|
a flag: g_u32HsEpStallLock is added to support it */
|
||
|
epNum = (uint32_t)(gUsbCmd.wIndex & 0xFul);
|
||
|
for (i=0ul; i<HSUSBD_MAX_EP; i++)
|
||
|
{
|
||
|
if ((((HSUSBD->EP[i].EPCFG & 0xf0ul) >> 4) == epNum) && ((g_u32HsEpStallLock & (1ul << i)) == 0ul))
|
||
|
{
|
||
|
HSUSBD->EP[i].EPRSPCTL = (HSUSBD->EP[i].EPRSPCTL & 0xeful) | HSUSBD_EP_RSPCTL_TOGGLE;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
/* Status stage */
|
||
|
HSUSBD_CLR_CEP_INT_FLAG(HSUSBD_CEPINTSTS_STSDONEIF_Msk);
|
||
|
HSUSBD_SET_CEP_STATE(HSUSBD_CEPCTL_NAKCLR);
|
||
|
HSUSBD_ENABLE_CEP_INT(HSUSBD_CEPINTEN_STSDONEIEN_Msk);
|
||
|
break;
|
||
|
}
|
||
|
case SET_ADDRESS:
|
||
|
{
|
||
|
g_hsusbd_UsbAddr = (uint8_t)gUsbCmd.wValue;
|
||
|
/* Status Stage */
|
||
|
HSUSBD_CLR_CEP_INT_FLAG(HSUSBD_CEPINTSTS_STSDONEIF_Msk);
|
||
|
HSUSBD_SET_CEP_STATE(HSUSBD_CEPCTL_NAKCLR);
|
||
|
HSUSBD_ENABLE_CEP_INT(HSUSBD_CEPINTEN_STSDONEIEN_Msk);
|
||
|
break;
|
||
|
}
|
||
|
case SET_CONFIGURATION:
|
||
|
{
|
||
|
g_hsusbd_UsbConfig = (uint8_t)gUsbCmd.wValue;
|
||
|
g_hsusbd_Configured = (uint8_t)1ul;
|
||
|
/* Status stage */
|
||
|
HSUSBD_CLR_CEP_INT_FLAG(HSUSBD_CEPINTSTS_STSDONEIF_Msk);
|
||
|
HSUSBD_SET_CEP_STATE(HSUSBD_CEPCTL_NAKCLR);
|
||
|
HSUSBD_ENABLE_CEP_INT(HSUSBD_CEPINTEN_STSDONEIEN_Msk);
|
||
|
break;
|
||
|
}
|
||
|
case SET_FEATURE:
|
||
|
{
|
||
|
if ((gUsbCmd.wValue & 0x3ul) == 2ul) /* TEST_MODE */
|
||
|
{
|
||
|
g_hsusbd_EnableTestMode = (uint8_t)1ul;
|
||
|
g_hsusbd_TestSelector = (uint8_t)(gUsbCmd.wIndex >> 8);
|
||
|
}
|
||
|
if ((gUsbCmd.wValue & 0x3ul) == 3ul) /* HNP ebable */
|
||
|
{
|
||
|
HSOTG->CTL |= (HSOTG_CTL_HNPREQEN_Msk | HSOTG_CTL_BUSREQ_Msk);
|
||
|
}
|
||
|
|
||
|
/* Status stage */
|
||
|
HSUSBD_CLR_CEP_INT_FLAG(HSUSBD_CEPINTSTS_STSDONEIF_Msk);
|
||
|
HSUSBD_SET_CEP_STATE(HSUSBD_CEPCTL_NAKCLR);
|
||
|
HSUSBD_ENABLE_CEP_INT(HSUSBD_CEPINTEN_STSDONEIEN_Msk);
|
||
|
break;
|
||
|
}
|
||
|
case SET_INTERFACE:
|
||
|
{
|
||
|
g_hsusbd_UsbAltInterface = (uint8_t)gUsbCmd.wValue;
|
||
|
if (g_hsusbd_pfnSetInterface != NULL)
|
||
|
{
|
||
|
g_hsusbd_pfnSetInterface((uint32_t)g_hsusbd_UsbAltInterface);
|
||
|
}
|
||
|
/* Status stage */
|
||
|
HSUSBD_CLR_CEP_INT_FLAG(HSUSBD_CEPINTSTS_STSDONEIF_Msk);
|
||
|
HSUSBD_SET_CEP_STATE(HSUSBD_CEPCTL_NAKCLR);
|
||
|
HSUSBD_ENABLE_CEP_INT(HSUSBD_CEPINTEN_STSDONEIEN_Msk);
|
||
|
break;
|
||
|
}
|
||
|
default:
|
||
|
{
|
||
|
/* Setup error, stall the device */
|
||
|
HSUSBD_SET_CEP_STATE(HSUSBD_CEPCTL_STALLEN_Msk);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Update Device State
|
||
|
*
|
||
|
* @param[in] None
|
||
|
*
|
||
|
* @return None
|
||
|
*
|
||
|
* @details This function is used to update Device state when Setup packet complete
|
||
|
*/
|
||
|
/** @cond HIDDEN_SYMBOLS */
|
||
|
#define TEST_J 0x01ul
|
||
|
#define TEST_K 0x02ul
|
||
|
#define TEST_SE0_NAK 0x03ul
|
||
|
#define TEST_PACKET 0x04ul
|
||
|
#define TEST_FORCE_ENABLE 0x05ul
|
||
|
/** @endcond HIDDEN_SYMBOLS */
|
||
|
|
||
|
void HSUSBD_UpdateDeviceState(void)
|
||
|
{
|
||
|
switch (gUsbCmd.bRequest)
|
||
|
{
|
||
|
case SET_ADDRESS:
|
||
|
{
|
||
|
HSUSBD_SET_ADDR(g_hsusbd_UsbAddr);
|
||
|
break;
|
||
|
}
|
||
|
case SET_CONFIGURATION:
|
||
|
{
|
||
|
if (g_hsusbd_UsbConfig == 0ul)
|
||
|
{
|
||
|
uint32_t volatile i;
|
||
|
/* Reset PID DATA0 */
|
||
|
for (i=0ul; i<HSUSBD_MAX_EP; i++)
|
||
|
{
|
||
|
if ((HSUSBD->EP[i].EPCFG & 0x1ul) == 0x1ul)
|
||
|
{
|
||
|
HSUSBD->EP[i].EPRSPCTL = HSUSBD_EP_RSPCTL_TOGGLE;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
case SET_FEATURE:
|
||
|
{
|
||
|
if(gUsbCmd.wValue == FEATURE_ENDPOINT_HALT)
|
||
|
{
|
||
|
uint32_t idx;
|
||
|
idx = (uint32_t)(gUsbCmd.wIndex & 0xFul);
|
||
|
HSUSBD_SetStall(idx);
|
||
|
}
|
||
|
else if (g_hsusbd_EnableTestMode)
|
||
|
{
|
||
|
g_hsusbd_EnableTestMode = (uint8_t)0ul;
|
||
|
if (g_hsusbd_TestSelector == TEST_J)
|
||
|
{
|
||
|
HSUSBD->TEST = TEST_J;
|
||
|
}
|
||
|
else if (g_hsusbd_TestSelector == TEST_K)
|
||
|
{
|
||
|
HSUSBD->TEST = TEST_K;
|
||
|
}
|
||
|
else if (g_hsusbd_TestSelector == TEST_SE0_NAK)
|
||
|
{
|
||
|
HSUSBD->TEST = TEST_SE0_NAK;
|
||
|
}
|
||
|
else if (g_hsusbd_TestSelector == TEST_PACKET)
|
||
|
{
|
||
|
HSUSBD->TEST = TEST_PACKET;
|
||
|
}
|
||
|
else if (g_hsusbd_TestSelector == TEST_FORCE_ENABLE)
|
||
|
{
|
||
|
HSUSBD->TEST = TEST_FORCE_ENABLE;
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
case CLEAR_FEATURE:
|
||
|
{
|
||
|
if(gUsbCmd.wValue == FEATURE_ENDPOINT_HALT)
|
||
|
{
|
||
|
uint32_t idx;
|
||
|
idx = (uint32_t)(gUsbCmd.wIndex & 0xFul);
|
||
|
HSUSBD_ClearStall(idx);
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
default:
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
/**
|
||
|
* @brief Prepare Control IN transaction
|
||
|
*
|
||
|
* @param[in] pu8Buf Control IN data pointer
|
||
|
* @param[in] u32Size IN transfer size
|
||
|
*
|
||
|
* @return None
|
||
|
*
|
||
|
* @details This function is used to prepare Control IN transfer
|
||
|
*/
|
||
|
void HSUSBD_PrepareCtrlIn(uint8_t pu8Buf[], uint32_t u32Size)
|
||
|
{
|
||
|
g_hsusbd_CtrlInPointer = pu8Buf;
|
||
|
g_hsusbd_CtrlInSize = u32Size;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
/**
|
||
|
* @brief Start Control IN transfer
|
||
|
*
|
||
|
* @param[in] None
|
||
|
*
|
||
|
* @return None
|
||
|
*
|
||
|
* @details This function is used to start Control IN
|
||
|
*/
|
||
|
void HSUSBD_CtrlIn(void)
|
||
|
{
|
||
|
uint32_t volatile i, cnt;
|
||
|
uint8_t u8Value;
|
||
|
if(g_hsusbd_CtrlInSize >= g_hsusbd_CtrlMaxPktSize)
|
||
|
{
|
||
|
/* Data size > MXPLD */
|
||
|
cnt = g_hsusbd_CtrlMaxPktSize >> 2;
|
||
|
for (i=0ul; i<cnt; i++)
|
||
|
{
|
||
|
HSUSBD->CEPDAT = *(uint32_t *)g_hsusbd_CtrlInPointer;
|
||
|
g_hsusbd_CtrlInPointer = (uint8_t *)(g_hsusbd_CtrlInPointer + 4ul);
|
||
|
}
|
||
|
HSUSBD_START_CEP_IN(g_hsusbd_CtrlMaxPktSize);
|
||
|
g_hsusbd_CtrlInSize -= g_hsusbd_CtrlMaxPktSize;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
/* Data size <= MXPLD */
|
||
|
cnt = g_hsusbd_CtrlInSize >> 2;
|
||
|
for (i=0ul; i<cnt; i++)
|
||
|
{
|
||
|
HSUSBD->CEPDAT = *(uint32_t *)g_hsusbd_CtrlInPointer;
|
||
|
g_hsusbd_CtrlInPointer += 4ul;
|
||
|
}
|
||
|
|
||
|
for (i=0ul; i<(g_hsusbd_CtrlInSize % 4ul); i++)
|
||
|
{
|
||
|
u8Value = *(uint8_t *)(g_hsusbd_CtrlInPointer+i);
|
||
|
outpb(&HSUSBD->CEPDAT, u8Value);
|
||
|
}
|
||
|
|
||
|
HSUSBD_START_CEP_IN(g_hsusbd_CtrlInSize);
|
||
|
g_hsusbd_CtrlInPointer = 0;
|
||
|
g_hsusbd_CtrlInSize = 0ul;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Start Control OUT transaction
|
||
|
*
|
||
|
* @param[in] pu8Buf Control OUT data pointer
|
||
|
* @param[in] u32Size OUT transfer size
|
||
|
*
|
||
|
* @return None
|
||
|
*
|
||
|
* @details This function is used to start Control OUT transfer
|
||
|
*/
|
||
|
void HSUSBD_CtrlOut(uint8_t pu8Buf[], uint32_t u32Size)
|
||
|
{
|
||
|
uint32_t volatile i;
|
||
|
while(1)
|
||
|
{
|
||
|
if ((HSUSBD->CEPINTSTS & HSUSBD_CEPINTSTS_RXPKIF_Msk) == HSUSBD_CEPINTSTS_RXPKIF_Msk)
|
||
|
{
|
||
|
for (i=0ul; i<u32Size; i++)
|
||
|
{
|
||
|
pu8Buf[i] = inpb(&HSUSBD->CEPDAT);
|
||
|
}
|
||
|
HSUSBD->CEPINTSTS = HSUSBD_CEPINTSTS_RXPKIF_Msk;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief Clear all software flags
|
||
|
*
|
||
|
* @param[in] None
|
||
|
*
|
||
|
* @return None
|
||
|
*
|
||
|
* @details This function is used to clear all software control flag
|
||
|
*/
|
||
|
void HSUSBD_SwReset(void)
|
||
|
{
|
||
|
/* Reset all variables for protocol */
|
||
|
g_hsusbd_UsbAddr = (uint8_t)0ul;
|
||
|
g_hsusbd_DmaDone = 0ul;
|
||
|
g_hsusbd_ShortPacket = (uint8_t)0ul;
|
||
|
g_hsusbd_Configured = (uint8_t)0ul;
|
||
|
|
||
|
/* Reset USB device address */
|
||
|
HSUSBD_SET_ADDR(0ul);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @brief HSUSBD Set Vendor Request
|
||
|
*
|
||
|
* @param[in] pfnVendorReq Vendor Request Callback Function
|
||
|
*
|
||
|
* @return None
|
||
|
*
|
||
|
* @details This function is used to set HSUSBD vendor request callback function
|
||
|
*/
|
||
|
void HSUSBD_SetVendorRequest(HSUSBD_VENDOR_REQ pfnVendorReq)
|
||
|
{
|
||
|
g_hsusbd_pfnVendorRequest = pfnVendorReq;
|
||
|
}
|
||
|
|
||
|
|
||
|
/*@}*/ /* end of group HSUSBD_EXPORTED_FUNCTIONS */
|
||
|
|
||
|
/*@}*/ /* end of group HSUSBD_Driver */
|
||
|
|
||
|
/*@}*/ /* end of group Standard_Driver */
|
||
|
|
||
|
/*** (C) COPYRIGHT 2016 Nuvoton Technology Corp. ***/
|