rt-thread-official/bsp/gd32107c-eval/Libraries/GD32F10x_usbd_driver/Source/usbd_int.c

459 lines
14 KiB
C

/*!
\file usbd_int.c
\brief USB device power interrupt routines
\version 2014-12-26, V1.0.0, firmware for GD32F10x
\version 2017-06-20, V2.0.0, firmware for GD32F10x
\version 2018-07-31, V2.1.0, firmware for GD32F10x
*/
/*
Copyright (c) 2018, GigaDevice Semiconductor Inc.
All rights reserved.
Redistribution and use in source and binary forms, with or without modification,
are permitted provided that the following conditions are met:
1. Redistributions of source code must retain the above copyright notice, this
list of conditions and the following disclaimer.
2. Redistributions in binary form must reproduce the above copyright notice,
this list of conditions and the following disclaimer in the documentation
and/or other materials provided with the distribution.
3. Neither the name of the copyright holder nor the names of its contributors
may be used to endorse or promote products derived from this software without
specific prior written permission.
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY
OF SUCH DAMAGE.
*/
#include "usbd_int.h"
extern uint32_t g_interrupt_mask;
extern __IO uint8_t g_suspend_enabled;
extern __IO uint8_t g_remote_wakeup_on;
extern __IO uint8_t g_ESOF_count;
#ifdef LPM_ENABLED
__IO uint32_t L1_remote_wakeup = 0U;
__IO uint32_t L1_resume = 0U;
__IO uint32_t besl = 0U;
#endif /* LPM_ENABLED */
static uint8_t usbd_intf_lpst (usbd_core_handle_struct *pudev);
static uint8_t usbd_intf_sof (usbd_core_handle_struct *pudev);
static uint8_t usbd_intf_esof (usbd_core_handle_struct *pudev);
static uint8_t usbd_intf_reset (usbd_core_handle_struct *pudev);
static uint8_t usbd_intf_suspend (usbd_core_handle_struct *pudev);
static uint8_t usbd_intf_wakeup (usbd_core_handle_struct *pudev);
/*!
\brief USB interrupt events service routine
\param[in] none
\param[out] none
\retval none
*/
void usbd_isr (void)
{
__IO uint16_t interrupt_flag = 0U;
__IO uint16_t ctlr = 0U;
interrupt_flag = USBD_REG_GET(USBD_INTF);
if (g_interrupt_mask & INTF_STIF & interrupt_flag) {
/* the endpoint successful transfer interrupt service */
usbd_intf_lpst(&usb_device_dev);
}
if (g_interrupt_mask & INTF_WKUPIF & interrupt_flag) {
/* clear wakeup interrupt flag in INTF */
USBD_REG_SET(USBD_INTF, (uint16_t)CLR_WKUPIF);
/* USB wakeup interrupt handle */
usbd_intf_wakeup(&usb_device_dev);
#ifdef LPM_ENABLED
/* clear L1 remote wakeup flag */
L1_remote_wakeup = 0;
#endif /* LPM_ENABLED */
}
if (g_interrupt_mask & INTF_SPSIF & interrupt_flag) {
if(!(USBD_REG_GET(USBD_CTL) & CTL_RSREQ)) {
/* process library core layer suspend routine*/
usbd_intf_suspend(&usb_device_dev);
/* clear of suspend interrupt flag bit must be done after setting of CTLR_SETSPS */
USBD_REG_SET(USBD_INTF, (uint16_t)CLR_SPSIF);
}
}
if (g_interrupt_mask & INTF_SOFIF & interrupt_flag) {
/* clear SOF interrupt flag in INTF */
USBD_REG_SET(USBD_INTF, (uint16_t)CLR_SOFIF);
/* USB SOF interrupt handle */
usbd_intf_sof(&usb_device_dev);
}
if (g_interrupt_mask & INTF_ESOFIF & interrupt_flag) {
/* clear ESOF interrupt flag in INTF */
USBD_REG_SET(USBD_INTF, (uint16_t)CLR_ESOFIF);
/* USB ESOF interrupt handle */
usbd_intf_esof(&usb_device_dev);
}
if (g_interrupt_mask & INTF_RSTIF & interrupt_flag) {
/* clear reset interrupt flag in INTF */
USBD_REG_SET(USBD_INTF, (uint16_t)CLR_RSTIF);
/* USB reset interrupt handle */
usbd_intf_reset(&usb_device_dev);
}
#ifdef LPM_ENABLED
if (g_interrupt_mask & INTF_L1REQ & interrupt_flag) {
/* clear L1 ST bit in LPM INTF */
USBD_REG_SET(USBD_INTF, CLR_L1REQ);
/* read BESL field from subendpoint0 register which coressponds to HIRD parameter in LPM spec */
besl = (USBD_REG_GET(USBD_LPMCS) & LPMCS_BLSTAT) >> 4;
/* read BREMOTEWAKE bit from subendpoint0 register which corresponding to bRemoteWake bit in LPM request */
L1_remote_wakeup = (USBD_REG_GET(USBD_LPMCS) & LPMCS_REMWK) >> 8;
/* process USB device core layer suspend routine */
/* enter USB model in suspend and system in low power mode (DEEP_SLEEP mode) */
usbd_intf_suspend(&usb_device_dev);
}
#endif /* LPM_ENABLED */
}
/*!
\brief handle USB high priority successful transfer event
\param[in] pudev: pointer to USB device instance
\param[out] none
\retval USB device operation status
*/
uint8_t usbd_intf_hpst (usbd_core_handle_struct *pudev)
{
uint8_t ep_num = 0U;
__IO uint16_t int_status = 0U;
__IO uint16_t ep_value = 0U;
usb_ep_struct *ep = NULL;
/* wait till interrupts are not pending */
while (0U != ((int_status = USBD_INTF) & (uint16_t)INTF_STIF)) {
/* get endpoint number and the value of control and state register */
ep_num = (uint8_t)(int_status & INTF_EPNUM);
ep_value = USBD_EPxCS(ep_num);
if (0U == (int_status & INTF_DIR)) {
/* handle the in direction transaction */
ep = &(pudev->in_ep[ep_num]);
if (0U != (ep_value & EPxCS_TX_ST)) {
/* clear successful transmit interrupt flag */
USBD_ENDP_TX_STAT_CLEAR(ep_num);
if (ep_value & EPxCS_TX_DTG) {
/* just handle single buffer situation */
ep->trs_count = (pbuf_reg + ep_num)->tx_count & EPTCNT_CNT;
}
/* maybe mutiple packets */
ep->trs_buf += ep->trs_count;
usbd_in_transaction(pudev, ep_num);
}
} else {
/* handle the out direction transaction */
uint16_t count = 0U;
ep = &(pudev->out_ep[ep_num]);
if (0U != (ep_value & EPxCS_RX_ST)) {
/* clear successful receive interrupt flag */
USBD_ENDP_RX_STAT_CLEAR(ep_num);
if (ep_value & EPxCS_TX_DTG) {
count = (pbuf_reg + ep_num)->tx_count & (uint16_t)EPRCNT_CNT;
if (0U != count) {
usbd_ep_data_read(ep->trs_buf, (pbuf_reg + ep_num)->tx_addr, count);
}
} else {
count = (pbuf_reg + ep_num)->rx_count & (uint16_t)EPRCNT_CNT;
if (0U != count) {
usbd_ep_data_read(ep->trs_buf, (pbuf_reg + ep_num)->rx_addr, count);
}
}
user_buffer_free(ep_num, DBUF_EP_OUT);
/* maybe mutiple packets */
ep->trs_count += count;
ep->trs_buf += count;
ep->trs_len -= count;
if ((0U == ep->trs_len) || (count < ep->maxpacket)) {
USBD_ENDP_TX_STATUS_SET(ep_num, EPRX_NAK);
/* enter data OUT status */
usbd_out_transaction(pudev, ep_num);
ep->trs_count = 0U;
}
}
}
}
return USBD_OK;
}
/*!
\brief handle USB low priority successful transfer event
\param[in] pudev: pointer to USB device instance
\param[out] none
\retval USB device operation status
*/
static uint8_t usbd_intf_lpst (usbd_core_handle_struct *pudev)
{
uint8_t ep_num = 0U;
__IO uint16_t int_status = 0U;
__IO uint16_t ep_value = 0U;
usb_ep_struct *ep = NULL;
/* wait till interrupts are not pending */
while (0U != ((int_status = USBD_REG_GET(USBD_INTF)) & (uint16_t)INTF_STIF)) {
/* get endpoint number and the value of control and state register */
ep_num = (uint8_t)(int_status & INTF_EPNUM);
ep_value = USBD_REG_GET(USBD_EPxCS(ep_num));
if (0U == (int_status & INTF_DIR)) {
/* handle the in direction transaction */
ep = &(pudev->in_ep[ep_num]);
if (0U != (ep_value & EPxCS_TX_ST)) {
/* clear successful transmit interrupt flag */
USBD_ENDP_TX_STAT_CLEAR(ep_num);
/* just handle single buffer situation */
ep->trs_count = (pbuf_reg + ep_num)->tx_count & EPTCNT_CNT;
/* maybe mutiple packets */
ep->trs_buf += ep->trs_count;
usbd_in_transaction(pudev, ep_num);
}
} else {
/* handle the out direction transaction */
uint16_t count = 0U;
ep = &(pudev->out_ep[ep_num]);
if (0U != (ep_value & EPxCS_RX_ST)) {
/* clear successful receive interrupt flag */
USBD_ENDP_RX_STAT_CLEAR(ep_num);
count = (pbuf_reg + ep_num)->rx_count & (uint16_t)EPRCNT_CNT;
if (0U != count) {
if (0U != (ep_value & EPxCS_SETUP)) {
/* handle setup packet */
usbd_ep_data_read(&(pudev->setup_packet[0]), pbuf_reg->rx_addr, count);
/* enter setup status */
usbd_setup_transaction(pudev);
return USBD_OK;
} else {
usbd_ep_data_read(ep->trs_buf, (pbuf_reg + ep_num)->rx_addr, count);
}
}
/* maybe mutiple packets */
ep->trs_count += count;
ep->trs_buf += count;
ep->trs_len -= count;
if ((0U == ep->trs_len) || (count < ep->maxpacket)) {
/* enter data OUT status */
usbd_out_transaction(pudev, ep_num);
ep->trs_count = 0U;
} else {
usbd_ep_rx(pudev, ep_num, ep->trs_buf, (uint16_t)ep->trs_len);
}
}
}
}
return USBD_OK;
}
/*!
\brief handle USB SOF event
\param[in] pudev: pointer to USB device instance
\param[out] none
\retval USB device operation status
*/
static uint8_t usbd_intf_sof (usbd_core_handle_struct *pudev)
{
/* if necessary, user can add code here */
if (NULL != usbd_int_fops) {
usbd_int_fops->SOF(pudev);
}
return USBD_OK;
}
/*!
\brief handle USB expect SOF event
\param[in] pudev: pointer to USB device instance
\param[out] none
\retval USB device operation status
*/
static uint8_t usbd_intf_esof (usbd_core_handle_struct *pudev)
{
/* control resume time by ESOFs */
if (g_ESOF_count > 0U) {
g_ESOF_count--;
if (0U == g_ESOF_count) {
if (1U == g_remote_wakeup_on) {
USBD_REG_SET(USBD_CTL, USBD_REG_GET(USBD_CTL) & ~CTL_RSREQ);
g_remote_wakeup_on = 0U;
} else {
USBD_REG_SET(USBD_CTL, USBD_REG_GET(USBD_CTL) | CTL_RSREQ);
g_ESOF_count = 3U;
g_remote_wakeup_on = 1U;
}
}
}
return USBD_OK;
}
/*!
\brief handle USB reset event
\param[in] pudev: pointer to USB device instance
\param[out] none
\retval USB device operation status
*/
static uint8_t usbd_intf_reset (usbd_core_handle_struct *pudev)
{
uint8_t i;
g_free_buf_addr = ENDP_BUF_ADDR;
/* configure endpoint 0 buffer */
pbuf_reg->tx_addr = (uint16_t)g_free_buf_addr;
g_free_buf_addr += USBD_EP0_MAX_SIZE;
pbuf_reg->rx_addr = (uint16_t)g_free_buf_addr;
g_free_buf_addr += USBD_EP0_MAX_SIZE;
/* configure endpoint 0 Rx count */
if (USBD_EP0_MAX_SIZE > 62U) {
pbuf_reg->rx_count = ((USBD_EP0_MAX_SIZE << 5U) - 1U) | 0x8000U;
} else {
pbuf_reg->rx_count = ((USBD_EP0_MAX_SIZE + 1U) & ~1U) << 9U;
}
pudev->in_ep[EP0].maxpacket = USBD_EP0_MAX_SIZE;
pudev->out_ep[EP0].maxpacket = USBD_EP0_MAX_SIZE;
/* set endpoints address */
for (i = 0U; i < EP_COUNT; i++)
{
_EP_ADDR_SET(i, i);
}
/* set device address as default address 0 */
USBD_REG_SET(USBD_DADDR, DADDR_USBEN);
/* clear endpoint 0 register */
USBD_REG_SET(USBD_EPxCS(EP0), USBD_EPxCS(EP0));
USBD_REG_SET(USBD_EPxCS(EP0), EP_CONTROL | EPRX_VALID | EPTX_NAK);
pudev->status = USBD_DEFAULT;
return USBD_OK;
}
/*!
\brief handle USB suspend event
\param[in] pudev: pointer to USB device instance
\param[out] none
\retval USB device operation status
*/
static uint8_t usbd_intf_suspend (usbd_core_handle_struct *pudev)
{
/* store the device current status */
pudev->prev_status = pudev->status;
/* set device in suspended state */
pudev->status = USBD_SUSPENDED;
/* enter USB in suspend and mcu system in low power mode */
if (g_suspend_enabled) {
usbd_suspend();
} else {
/* if not possible then resume after xx ms */
g_ESOF_count = 3U;
}
return USBD_OK;
}
/*!
\brief handle USB wakeup event
\param[in] pudev: pointer to USB device instance
\param[out] none
\retval USB device operation status
*/
static uint8_t usbd_intf_wakeup (usbd_core_handle_struct *pudev)
{
/* restore the old status */
pudev->status = pudev->prev_status;
#ifdef LPM_ENABLED
if ((0U == g_remote_wakeup_on) && (0U == L1_resume)) {
resume_mcu();
} else if (1U == g_remote_wakeup_on) {
/* no operation */
} else {
L1_resume = 0U;
}
#else
if (0U == g_remote_wakeup_on) {
resume_mcu();
}
#endif /* LPM_ENABLED */
return USBD_OK;
}