1666 lines
36 KiB
C
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/**
*********************************************************************************
*
* @file ald_usb.c
* @brief USB module driver.
*
* @version V1.0
* @date 25 Feb 2022
* @author AE Team
* @note
* Change Logs:
* Date Author Notes
* 25 Feb 2022 AE Team The first version
*
* Copyright (C) Shanghai Eastsoft Microelectronics Co. Ltd. All rights reserved.
*
* SPDX-License-Identifier: Apache-2.0
*
* Licensed under the Apache License, Version 2.0 (the License); you may
* not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an AS IS BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
**********************************************************************************
*/
#include "ald_conf.h"
#include "usb_lowlayer_api.h"
/** @addtogroup ES32VF2264_ALD
* @{
*/
/** @defgroup USB USB
* @brief USB module driver
* @{
*/
#ifdef ALD_USB
/**
* @defgroup USB_Public_Functions USB Public Function
* @{
*/
/** @defgroup USB_Public_Functions_Group1 Base functions
* @brief Base functions
* @{
*/
/**
* @brief Gets the number of current frame.
* @retval Number of the frame.
*/
uint32_t ald_usb_frame_number_get(void)
{
uint8_t framel = (uint8_t)USB->FRAME1;
uint8_t frameh = (uint8_t)USB->FRAME2;
return ((uint32_t)(framel & 0xFF) + ((uint32_t)(frameh & 0xFF) << 8));
}
/**
* @brief Request the session.
* @param start: true/false.
* @retval None
*/
void ald_usb_otg_session_request(bool start)
{
if (start)
USB->DEVCON |= (uint8_t)ALD_USB_DEVCON_SESSION;
else
USB->DEVCON &= (uint8_t)(~(ALD_USB_DEVCON_SESSION));
return;
}
/**
* @brief Gets the mode.
* @retval Mode
*/
uint32_t ald_usb_mode_get(void)
{
return (USB->DEVCON & ((uint8_t)(ALD_USB_DEVCON_HOST | ALD_USB_DEVCON_SESSION)));
}
/**
* @brief Start host require.
* @retval Mode
*/
uint32_t ald_usb_mode_host_req(void)
{
USB->DEVCON |= (uint8_t)ALD_USB_DEVCON_HOSTREQ;
return 0;
}
/**
* @brief Clear host require.
* @retval Mode
*/
uint32_t ald_usb_mode_host_req_clear(void)
{
USB->DEVCON &= (uint8_t)(~ALD_USB_DEVCON_HOSTREQ);
return 0;
}
/**
* @brief Enable/Disable the high mode.
* @param enable: ENABLE/DISABLE.
* @retval None
*/
void ald_usb_high_speed_enable(bool enable)
{
assert_param(enable == DISABLE);
return;
}
/**
* @brief Gets the speed of the device.
* @retval Type of the speed.
*/
uint32_t ald_usb_device_speed_get(void)
{
return ALD_USB_FULL_SPEED;
}
/**
* @brief Gets the number of the endpoint.
* @retval Number of the endpoint.
*/
uint32_t ald_usb_num_ep_get( void)
{
return ALD_NUM_USB_EP;
}
/**
* @brief Reset USB Control.
* @retval None
*/
void ald_usb_control_reset(void)
{
//ALD_RCU_USB_RESET_ENABLE();
return;
}
/**
* @brief Output USB clock, not support.
* @retval None
*/
void ald_usb_clock_output(void)
{
return;
}
/**
* @brief Starts eye diagram for high-speed host, not support.
* @param buf: Buffer for eye diagram.
* @param len: Length of the buffer.
* @retval Status, 0 means success, other values means failure.
*/
int ald_usb_eye_diagram_start(uint8_t *buf, uint16_t len)
{
return 0;
}
/**
* @brief Pull up or pull down USB dp line.
* @param pupd: USB_DPDM_FLOATING/USB_DPDM_PUSH_UP/USB_DPDM_PUSH_DOWN.
* @retval None
*/
void ald_usb_dppud_set(ald_dpdm_push_t pupd)
{
USB->DPDMCON &= (uint8_t)(~USB_DPDMCON_DPPUD_MSK);
USB->DPDMCON |= (uint8_t)(pupd << USB_DPDMCON_DPPUD_POSS);
return;
}
/**
* @brief Pull up or pull down USB dm line.
* @param pupd: USB_DPDM_FLOATING/USB_DPDM_PUSH_UP/USB_DPDM_PUSH_DOWN.
* @retval None
*/
void ald_usb_dmpud_set(ald_dpdm_push_t pupd)
{
USB->DPDMCON &= (uint8_t)(~USB_DPDMCON_DMPUD_MSK);
USB->DPDMCON |= (uint8_t)(pupd << USB_DPDMCON_DMPUD_POSS);
return;
}
/**
* @brief Switch usb mode by software.
* @param host: 0, indicates that software force to host
host: 1, indicates that software force to device.
* @retval None
*/
void ald_usb_swcid_host(uint8_t host)
{
if (host)
USB->SWCID |= (uint8_t)(USB_SWCID_HOST_MSK);
else
USB->SWCID &= (uint8_t)(~USB_SWCID_HOST_MSK);
return;
}
/**
* @brief Switch the control method of CID.
* @param cid: 0, indicates that use the hardware control
cid: 1, indicates that use the software control.
* @retval None
*/
void ald_usb_swcid_cidctrl(uint8_t cid)
{
if (cid)
USB->SWCID |= (uint8_t)(ALD_USB_SWCID_CIDCTRL);
else
USB->SWCID &= (uint8_t)(~ALD_USB_SWCID_CIDCTRL);
return;
}
/**
* @}
*/
/** @defgroup USB_Public_Functions_Group2 Device functions
* @brief Device functions
* @{
*/
/**
* @brief Gets the address.
* @retval Address.
*/
uint8_t ald_usb_dev_get_addr(void)
{
return (USB->FADDR);
}
/**
* @brief Sets the address.
* @param addr: The address which will be set.
* @retval None
*/
void ald_usb_dev_set_addr(uint8_t addr)
{
USB->FADDR = addr;
return;
}
/**
* @brief Enable connection.
* @retval None
*/
void ald_usb_dev_connect(void)
{
USB->DPDMCON |= (uint8_t)(ALD_USB_DPDMCON_PHYPWREN);
return;
}
/**
* @brief Disable connection.
* @retval None
*/
void ald_usb_dev_disconnect(void)
{
USB->DPDMCON &= (uint8_t)(~(ALD_USB_DPDMCON_PHYPWREN));
return;
}
/**
* @brief Enable the devices suspend.
* @retval None
*/
void ald_usb_dev_suspend_enable(void)
{
USB->POWER |= (uint8_t)USB_POWER_SUSPENDEN_MSK;
return;
}
/**
* @brief Configure the endpoint in device mode.
* @param ep_idx: Index of the endpoint
* @param p_max: Size of the maximum package.
* @param flags: Flags of the endpoint.
* @retval None
*/
void ald_usb_dev_ep_config(uint32_t ep_idx, uint32_t p_max, uint32_t flags)
{
uint32_t tmp = 0U;
USB->INDEX = (uint8_t)ep_idx;
if (flags & ALD_USB_EP_DEV_IN) {
USB->TXMAXP = (uint8_t)(p_max);
if (flags & ALD_USB_EP_AUTO_SET)
tmp |= USB_TXCSRH_AUTOSET_MSK;
if ((flags & ALD_USB_EP_MODE_MASK) == ALD_USB_EP_MODE_ISOC)
tmp |= USB_TXCSRH_ISO_MSK;
USB->CSR0H_TXCSRH |= (uint8_t)tmp;
USB->CSR0L_TXCSRL |= (uint8_t)USB_TXCSRL_CLRDT_MSK;
}
else {
USB->RXMAXP = (uint8_t)(p_max);
if (flags & ALD_USB_EP_AUTO_CLEAR)
tmp |= USB_RXCSRH_AUTOCLR_MSK;
if ((flags & ALD_USB_EP_MODE_MASK) == ALD_USB_EP_MODE_ISOC)
tmp |= USB_TXCSRH_ISO_MSK;
USB->RXCSRH |= (uint8_t)tmp;
USB->RXCSRL |= (uint8_t)USB_RXCSRL_CLRDT_MSK;
}
return;
}
/**
* @brief Gets the parameters of the endpoint.
* @param ep_idx: Index of the endpoint
* @param p_max: Size of the maximum package.
* @param flags: Flags of the endpoint.
* @retval None
*/
void ald_usb_dev_ep_get_config(uint32_t ep_idx, uint32_t *p_max, uint32_t *flags)
{
uint32_t tmp;
USB->INDEX = (uint8_t)ep_idx;
if (*flags & ALD_USB_EP_DEV_IN) {
*flags = ALD_USB_EP_DEV_IN;
*p_max = (uint32_t)USB->TXMAXP;
tmp = (uint32_t)USB->CSR0H_TXCSRH;
if (tmp & USB_TXCSRH_AUTOSET_MSK)
*flags |= ALD_USB_EP_AUTO_SET;
if (tmp & USB_TXCSRH_ISO_MSK)
*flags |= ALD_USB_EP_MODE_ISOC;
else
*flags |= ALD_USB_EP_MODE_BULK;
}
else {
*flags = ALD_USB_EP_DEV_OUT;
*p_max = (uint32_t)USB->RXMAXP;
tmp = (uint32_t)USB->RXCSRH;
if (tmp & USB_RXCSRH_AUTOCLR_MSK)
*flags |= ALD_USB_EP_AUTO_CLEAR;
if (tmp & USB_RXCSRH_ISO_MSK)
*flags |= ALD_USB_EP_MODE_ISOC;
else
*flags |= ALD_USB_EP_MODE_BULK;
}
return;
}
extern void printf_e(const char *fmt, ...);
/**
* @brief Acknowledge the data from host.
* @param ep_idx: Index of the endpoint
* @param last: true/false
* @retval None
*/
void ald_usb_dev_ep_data_ack(uint32_t ep_idx, bool last)
{
USB->INDEX = (uint8_t)ep_idx;
if (ep_idx == ALD_USB_EP_0)
USB->CSR0L_TXCSRL |= (uint8_t)(USB_CSR0L_RXRDYC_MSK | (last ? USB_CSR0L_DATAEND_MSK : 0));
else
USB->RXCSRL &= (uint8_t)(~USB_RXCSRL_RXRDY_MSK);
return;
}
/**
* @brief Stall the endpoint.
* @param ep_idx: Index of the endpoint
* @param flags: Flags.
* @retval None
*/
void ald_usb_dev_ep_stall(uint32_t ep_idx, uint32_t flags)
{
USB->INDEX = (uint8_t)ep_idx;
if (ep_idx == ALD_USB_EP_0)
USB->CSR0L_TXCSRL |= (uint8_t)(USB_CSR0L_RXRDYC_MSK | USB_CSR0L_STALL_MSK);
else if (flags == ALD_USB_EP_DEV_IN)
USB->CSR0L_TXCSRL |= (uint8_t)USB_TXCSRL_STALL_MSK;
else
USB->RXCSRL |= (uint8_t)USB_RXCSRL_STALL_MSK;
return;
}
/**
* @brief Cancel the stall status.
* @param ep_idx: Index of the endpoint
* @param flags: Flags.
* @retval None
*/
void ald_usb_dev_ep_stall_clear(uint32_t ep_idx, uint32_t flags)
{
USB->INDEX = (uint8_t)ep_idx;
if (ep_idx == ALD_USB_EP_0)
USB->CSR0L_TXCSRL &= (uint8_t)(~USB_CSR0L_STALLED_MSK);
else if (flags == ALD_USB_EP_DEV_IN){
USB->CSR0L_TXCSRL &= (uint8_t)(~(USB_TXCSRL_STALL_MSK | USB_TXCSRL_STALLED_MSK));
USB->CSR0L_TXCSRL |= (uint8_t)USB_TXCSRL_CLRDT_MSK;
}
else{
USB->RXCSRL &= (uint8_t)(~(USB_RXCSRL_STALL_MSK | USB_RXCSRL_STALLED_MSK));
USB->RXCSRL |= (uint8_t)USB_RXCSRL_CLRDT_MSK;
}
return;
}
/**
* @brief Clear the status of the endpoint.
* @param ep_idx: Index of the endpoint
* @param flags: Flags.
* @retval None
*/
void ald_usb_dev_ep_status_clear(uint32_t ep_idx, uint32_t flags)
{
USB->INDEX = (uint8_t)ep_idx;
if (ep_idx == ALD_USB_EP_0) {
if (flags & ALD_USB_DEV_EP0_OUT_PKTRDY)
USB->CSR0L_TXCSRL |= (uint8_t)USB_CSR0L_RXRDYC_MSK;
if (flags & ALD_USB_DEV_EP0_SETUP_END)
USB->CSR0L_TXCSRL |= (uint8_t)USB_CSR0L_SETENDC_MSK;
if (flags & ALD_USB_DEV_EP0_SENT_STALL)
USB->CSR0L_TXCSRL &= (uint8_t)(~USB_CSR0L_STALLED_MSK);
}
else {
USB->CSR0L_TXCSRL &= (uint8_t)(~(flags & (ALD_USB_DEV_TX_SENT_STALL | ALD_USB_DEV_TX_UNDERRUN)));
USB->RXCSRL &= (uint8_t)(~((flags & (ALD_USB_DEV_RX_SENT_STALL | ALD_USB_DEV_RX_DATA_ERROR
| ALD_USB_DEV_RX_OVERRUN)) >> ALD_USB_RX_EPSTATUS_SHIFT));
}
return;
}
/**
* @}
*/
/** @defgroup USB_Public_Functions_Group3 Host functions
* @brief Host functions
* @{
*/
/**
* @brief Gets the device's address.
* @param ep_idx: Index of the endpoint
* @param flags: Flags.
* @retval Address
*/
uint32_t ald_usb_host_addr_get(uint32_t ep_idx, uint32_t flags)
{
USB->INDEX = (uint8_t)ep_idx;
return (USB->FADDR);
}
/**
* @brief Sets the device's address.
* @param ep_idx: Index of the endpoint.
* @param addr: The device's address.
* @param flags: Flags.
* @retval None
*/
void ald_usb_host_addr_set(uint32_t ep_idx, uint32_t addr, uint32_t flags)
{
USB->INDEX = (uint8_t)ep_idx;
USB->FADDR = (uint8_t)addr;
return;
}
/**
* @brief Configure the endpoint in host mode.
* @param ep_idx: Index of the endpoint.
* @param p_max: Size of the maximum package.
* @param nak_val: Value of the nack.
* @param t_ep: Target endpoint.
* @param flags: Flags.
* @retval None
*/
void ald_usb_host_ep_config(uint32_t ep_idx, uint32_t p_max, uint32_t nak_val, uint32_t t_ep, uint32_t flags)
{
uint32_t tmp = 0U;
USB->INDEX = (uint8_t)ep_idx;
if (ep_idx == ALD_USB_EP_0) {
USB->NAKLIMIT0_TXINTERVAL = (uint8_t)nak_val;
if (flags & ALD_USB_EP_SPEED_HIGH)
;
else if (flags & ALD_USB_EP_SPEED_FULL)
;
else
;
}
else {
tmp = t_ep;
if (flags & ALD_USB_EP_SPEED_HIGH)
;
else if (flags & ALD_USB_EP_SPEED_FULL)
;
else
;
switch (flags & ALD_USB_EP_MODE_MASK) {
case ALD_USB_EP_MODE_BULK:
tmp |= ALD_USB_TXTYPE1_PROTO_BULK;
break;
case ALD_USB_EP_MODE_ISOC:
tmp |= ALD_USB_TXTYPE1_PROTO_ISOC;
break;
case ALD_USB_EP_MODE_INT:
tmp |= ALD_USB_TXTYPE1_PROTO_INT;
break;
case ALD_USB_EP_MODE_CTRL:
tmp |= ALD_USB_TXTYPE1_PROTO_CTRL;
break;
}
if (flags & ALD_USB_EP_HOST_OUT) {
USB->TXTYPE = (uint8_t)tmp;
USB->NAKLIMIT0_TXINTERVAL = (uint8_t)nak_val;
USB->TXMAXP = (uint8_t)p_max;
tmp = 0;
if (flags & ALD_USB_EP_AUTO_SET)
tmp = (uint8_t)ALD_USB_TXCSRH_AUTOSET;
USB->CSR0H_TXCSRH |= (uint8_t)tmp;
}
else {
USB->RXTYPE = (uint8_t)tmp;
USB->RXINTERVAL = (uint8_t)nak_val;
USB->RXMAXP = (uint8_t)p_max;
tmp = 0;
if (flags & ALD_USB_EP_AUTO_CLEAR)
tmp |= (uint8_t)USB_RXCSRH_AUTOCLR_MSK;
if (flags & ALD_USB_EP_AUTO_REQUEST)
tmp |= (uint8_t)USB_RXCSRH_AUTOREQ_MSK;
USB->RXCSRH |= (uint8_t)tmp;
}
}
return;
}
/**
* @brief Acknowledge the data in host mode.
* @param ep_idx: Index of the endpoint.
* @retval None
*/
void ald_usb_host_ep_data_ack(uint32_t ep_idx)
{
USB->INDEX = (uint8_t)ep_idx;
if (ep_idx == ALD_USB_EP_0)
USB->CSR0L_TXCSRL &= (uint8_t)(~(USB_CSR0L_RXRDY_MSK));
else
USB->RXCSRL &= (uint8_t)(~(USB_RXCSRL_RXRDY_MSK));
return;
}
/**
* @brief Toggle the data in host mode.
* The function is used to force the state of the data toggle in host mode.
* If the value passed in the bDataToggle parameter is false, then the data
* toggle is set to the DATA0 state, and if it is true it is set to the DATA1
* state.
* @param ep_idx: Index of the endpoint.
* @param toggle: true/false.
* @param flags: can be USB_EP_HOST_IN or USB_EP_HOST_OUT.
* @retval None
*/
void ald_usb_host_ep_data_toggle(uint32_t ep_idx, bool toggle, uint32_t flags)
{
/* not supported by es32f0271. */
return;
}
/**
* @brief Clear the status of endpoint in host mode.
* @param ep_idx: Index of the endpoint.
* @param flags: Flags.
* @retval None
*/
void ald_usb_host_ep_status_clear(uint32_t ep_idx, uint32_t flags)
{
USB->INDEX = (uint8_t)ep_idx;
if (ep_idx == ALD_USB_EP_0) {
USB->CSR0L_TXCSRL &= (uint8_t)(~flags & 0xFF);
}
else {
USB->CSR0L_TXCSRL &= (uint8_t)(~flags & 0xFF);
USB->RXCSRL &= (uint8_t)(~(flags >> 16) & 0xFF);
}
return;
}
/**
* @brief Gets the HUB's address.
* @param ep_idx: Index of the endpoint.
* @param flags: Flags.
* @retval Address
*/
uint32_t ald_usb_host_hub_addr_get(uint32_t ep_idx, uint32_t flags)
{
/* not supported by es32f0271. */
return 0;
}
/**
* @brief Sets the HUB's address.
* @param ep_idx: Index of the endpoint.
* @param addr: HUB's address which will be set.
* @param flags: Flags.
* @retval Address
*/
void ald_usb_host_hub_addr_set(uint32_t ep_idx, uint32_t addr, uint32_t flags)
{
/* not supported by es32f0271. */
return;
}
/**
* @brief Disable power.
* @retval None
*/
void ald_usb_host_pwr_disable(void)
{
USB->DPDMCON &= (uint8_t)(~(ALD_USB_DPDMCON_PHYPWREN));
return;
}
/**
* @brief Enable power.
* @retval None
*/
void ald_usb_host_pwr_enable(void)
{
USB->DPDMCON |= (uint8_t)(ALD_USB_DPDMCON_PHYPWREN);
return;
}
/**
* @brief Configure power in host mode.
* @param flags: Flags
* @retval None
*/
void ald_usb_host_pwr_config(uint32_t flags)
{
return;
}
/**
* @brief Disable the fault parameters of the power.
* @retval None
*/
void ald_usb_host_pwr_fault_disable(void)
{
return;
}
/**
* @brief Enable the fault parameters of the power.
* @retval None
*/
void ald_usb_host_pwr_fault_enable(void)
{
return;
}
/**
* @brief Request data IN(from device to host)
* @param ep_idx: Index of the endpoint.
* @retval None
*/
void ald_usb_host_request_in(uint32_t ep_idx)
{
USB->INDEX = (uint8_t)ep_idx;
if (ep_idx == ALD_USB_EP_0)
USB->CSR0L_TXCSRL |= (uint8_t)USB_CSR0L_REQPKT_MSK;
else
USB->RXCSRL |= (uint8_t)USB_RXCSRL_REQPKT_MSK;
return;
}
/**
* @brief Clear the status of request IN.
* @param ep_idx: Index of the endpoint.
* @retval None
*/
void ald_usb_host_request_in_clear(uint32_t ep_idx)
{
USB->INDEX = (uint8_t)ep_idx;
if (ep_idx == ALD_USB_EP_0)
USB->CSR0L_TXCSRL &= (uint8_t)(~USB_CSR0L_REQPKT_MSK);
else
USB->RXCSRL &= (uint8_t)(~USB_RXCSRL_REQPKT_MSK);
return;
}
/**
* @brief Set the request for a status IN transaction.
* @retval None
*/
void ald_usb_host_request_status(void)
{
USB->CSR0L_TXCSRL |= (uint8_t)(ALD_USB_CSR0L_REQPKT | ALD_USB_CSR0L_STATUSPKT);
return;
}
/**
* @brief Reset the USB's bus.
* @param start: true/false.
* @retval None
*/
void ald_usb_host_reset(bool start)
{
if (start)
USB->POWER |= ALD_USB_POWER_RESET;
else
USB->POWER &= ~(ALD_USB_POWER_RESET);
return;
}
/**
* @brief Resume the devices.
* @param start: true/false.
* @retval None
*/
void ald_usb_host_resume(bool start)
{
if (start)
USB->POWER |= (uint8_t)USB_POWER_RESUME_MSK;
else
USB->POWER &= (uint8_t)(~(ALD_USB_POWER_RESUME));
return;
}
/**
* @brief Suspend the devices.
* @retval None
*/
void ald_usb_host_suspend(void)
{
USB->POWER |= (uint8_t)USB_POWER_SUSPEND_MSK;
return;
}
/**
* @brief Gets the device's speed.
* @retval Type of the speed.
*/
uint32_t ald_usb_host_speed_get(void)
{
if (USB->DEVCON & ALD_USB_DEVCON_FSDEV)
return ALD_USB_FULL_SPEED;
if (USB->DEVCON & ALD_USB_DEVCON_LSDEV)
return ALD_USB_LOW_SPEED;
return ALD_USB_UNDEF_SPEED;
}
/**
* @brief Sets the endpoint speed.
* @param ep_idx: Index of the endpoint.
* @param flags: Type of the speed.
* @retval None
*/
void ald_usb_host_ep_speed_set(uint32_t ep_idx, uint32_t flags)
{
/* not support in F0271 */
return;
}
/**
* @brief Ping the endpoint.
* @param ep_idx: Index of the endpoint.
* @param enable: ENABLE/DISABLE.
* @retval None
*/
void ald_usb_host_ep_ping(uint32_t ep_idx, bool enable)
{
/* not support in F0271 */
return;
}
/**
* @}
*/
/** @defgroup USB_Public_Functions_Group4 Endpoint functions
* @brief Endpoint functions
* @{
*/
/**
* @brief Gets the size of the available data.
* @param ep_idx: Index of the endpoint
* @retval Size in bytes.
*/
uint32_t ald_usb_ep_data_avail(uint32_t ep_idx)
{
USB->INDEX = (uint8_t)ep_idx;
if (ep_idx == ALD_USB_EP_0) {
if ((USB->CSR0L_TXCSRL & ALD_USB_CSR0L_RXRDY) == 0){
return 0;
}
return USB->COUNT0_RX1;
}
else {
if ((USB->RXCSRL & ALD_USB_CSR0L_RXRDY) == 0)
{
return 0;
}
return (USB->COUNT0_RX1 + (((uint32_t)USB->RXCOUNT2) << 8));
}
}
/**
* @brief Gets the data from FIFO.
* @param ep_idx: Index of the endpoint
* @param data: Pointer to the buffer.
* @param size: Size of the data.
* @retval Status.
*/
int32_t ald_usb_ep_data_get(uint32_t ep_idx, uint8_t *data, uint32_t *size)
{
uint32_t i, rx_fifo_addr;
USB->INDEX = (uint8_t)ep_idx;
if (ep_idx == ALD_USB_EP_0) {
if ((USB->CSR0L_TXCSRL & ALD_USB_CSR0L_RXRDY) == 0) {
*size = 0;
return -1;
}
i = USB->COUNT0_RX1;
}
else {
if ((USB->RXCSRL & ALD_USB_CSR0L_RXRDY) == 0) {
*size = 0;
return -1;
}
i = USB->COUNT0_RX1 + (((uint32_t)USB->RXCOUNT2) << 8);
}
i = (i < *size) ? i : *size;
*size = i;
rx_fifo_addr = (uint32_t)(&USB->EP0FIFO) + 4 * ep_idx;
for (; i > 0; i--)
*data++ = *(volatile uint8_t *)(rx_fifo_addr);
return 0;
}
/**
* @brief Puts data to the FIFO.
* @param ep_idx: Index of the endpoint
* @param data: Pointer to the data.
* @param size: Size of the data.
* @retval Status.
*/
int32_t ald_usb_ep_data_put(uint32_t ep_idx, uint8_t *data, uint32_t size)
{
uint32_t tx_fifo_addr;
USB->INDEX = (uint8_t)ep_idx;
if (ep_idx == ALD_USB_EP_0) {
if (USB->CSR0L_TXCSRL & ALD_USB_CSR0L_TXRDY)
return -1;
}
else {
if (USB->CSR0L_TXCSRL & ALD_USB_TXCSRL_TXRDY)
return -1;
}
tx_fifo_addr = (uint32_t)(&USB->EP0FIFO) + 4 * ep_idx;
for (; size > 0; size--)
*(volatile uint8_t *)tx_fifo_addr = *data++;
return 0;
}
/**
* @brief Send data.
* @param ep_idx: Index of the endpoint
* @param tx_type: Type.
* @retval Status.
*/
int32_t ald_usb_ep_data_send(uint32_t ep_idx, uint32_t tx_type)
{
uint32_t tmp;
USB->INDEX = (uint8_t)ep_idx;
if (ep_idx == ALD_USB_EP_0) {
if (USB->CSR0L_TXCSRL & ALD_USB_CSR0L_TXRDY)
return -1;
tmp = tx_type & 0xFF;
}
else {
if (USB->CSR0L_TXCSRL & ALD_USB_TXCSRL_TXRDY)
return -1;
tmp = (tx_type >> 8) & 0xff;
}
USB->CSR0L_TXCSRL = tmp;
return 0;
}
/**
* @brief Clear the status of the toggle.
* @param ep_idx: Index of the endpoint
* @param flags: Flags.
* @retval None
*/
void ald_usb_ep_data_toggle_clear(uint32_t ep_idx, uint32_t flags)
{
USB->INDEX = (uint8_t)ep_idx;
if (flags & (ALD_USB_EP_HOST_OUT | ALD_USB_EP_DEV_IN))
USB->CSR0L_TXCSRL |= (uint8_t)ALD_USB_TXCSRL_CLRDT;
else
USB->RXCSRL |= (uint8_t)ALD_USB_RXCSRL_CLRDT;
return;
}
/**
* @brief Sets the size of request data IN
* @param ep_idx: Index of the endpoint
* @param count: Size of request data IN.
* @retval None
*/
void ald_usb_ep_req_packet_count(uint32_t ep_idx, uint32_t count)
{
/* not support in f0271 */
return;
}
/**
* @brief Gets the status of the endpoint.
* @param ep_idx: Index of the endpoint
* @retval Status.
*/
uint32_t ald_usb_ep_status(uint32_t ep_idx)
{
uint32_t status;
USB->INDEX = (uint8_t)ep_idx;
status = (ep_idx == ALD_USB_EP_0) ? (USB->CSR0L_TXCSRL): (USB->CSR0L_TXCSRL | (USB->RXCSRL << 16));
return status;
}
/**
* @brief Configure the endpoint in DMA mode.
* @param ep_idx: Index of the endpoint
* @param flag: Flags.
* @param en: ENABLE/DISABLE.
* @retval None
*/
void ald_usb_ep_dma_config(uint32_t ep_idx, uint32_t flag, type_func_t en)
{
/* Not supported in F0271 */
return;
}
/**
* @}
*/
/** @defgroup USB_Public_Functions_Group5 FIFO functions
* @brief FIFO functions
* @{
*/
/**
* @brief Gets the address of the FIFO.
* @param ep_idx: Index of the endpoint
* @retval Address
*/
uint32_t ald_usb_fifo_addr_get(uint32_t ep_idx)
{
/* Not supported in F0271 */
return 0;
}
/**
* @brief Gets the parameters of the FIFO.
* @param ep_idx: Index of the endpoint
* @param addr: Address.
* @param size: Size of FIFO.
* @param flags: Flags.
* @retval None
*/
void ald_usb_fifo_config_get(uint32_t ep_idx, uint32_t *addr, uint32_t *size, uint32_t flags)
{
USB->INDEX = ep_idx;
if (flags & (ALD_USB_EP_HOST_OUT | ALD_USB_EP_DEV_IN)) {
*addr = ((uint32_t)USB->TXFIFO1 | ((uint32_t)USB->TXFIFO2 & 0x03)) << 3;
*size = ((USB->TXFIFO2 & ALD_USB_TXFIFO2_DPB) == 0x00) ? (USB->TXFIFO2 & ALD_USB_TXFIFO2_MAXPKTSIZE_1024) : (2 * (USB->TXFIFO2 & ALD_USB_TXFIFO2_MAXPKTSIZE_1024));
}
else {
*addr = ((uint32_t)USB->RXFIFO1 | ((uint32_t)USB->RXFIFO2 & 0x03)) << 3;
*size = ((USB->RXFIFO2 & ALD_USB_RXFIFO2_DPB) == 0x00) ? (USB->RXFIFO2 & ALD_USB_RXFIFO2_MAXPKTSIZE_1024) : (2 * (USB->RXFIFO2 & ALD_USB_RXFIFO2_MAXPKTSIZE_1024));
}
return;
}
/**
* @brief Sets the parameters of the FIFO.
* @param ep_idx: Index of the endpoint
* @param addr: Address.
* @param size: Size of FIFO, valid parameter has defined in usb_lowlayer_api.h.
* @param flags: Flags.
* @retval None
*/
void ald_usb_fifo_config_set(uint32_t ep_idx, uint32_t addr, uint32_t size, uint32_t flags)
{
USB->INDEX = ep_idx;
if (flags & (ALD_USB_EP_HOST_OUT | ALD_USB_EP_DEV_IN)) {
USB->TXFIFO1 = (uint8_t)((addr >> 3) & 0xFF);
USB->TXFIFO2 |= (uint8_t)(((addr >> 3) >> 8) & 0x0F);
USB->TXFIFO2 |= (uint8_t)(size << USB_TXFIFO2_MAXPKTSIZE_POSS);
USB->CSR0L_TXCSRL |= (uint8_t)ALD_USB_TXCSRL_FLUSH;
}
else {
USB->RXFIFO1 = (uint8_t)((addr >> 3) & 0xFF);
USB->RXFIFO2 |= (uint8_t)(((addr >> 3) >> 8) & 0x0F);
USB->RXFIFO2 |= (uint8_t)(size << USB_RXFIFO2_MAXPKTSIZE_POSS);
USB->RXCSRL |= (uint8_t)ALD_USB_RXCSRL_FLUSH;
}
return;
}
/**
* @brief Flush the FIFO
* @param ep_idx: Index of the endpoint
* @param flags: Flags.
* @retval None
*/
void ald_usb_fifo_flush(uint32_t ep_idx, uint32_t flags)
{
USB->INDEX = ep_idx;
if (ep_idx == ALD_USB_EP_0) {
if ((USB->CSR0L_TXCSRL & (ALD_USB_CSR0L_RXRDY | ALD_USB_CSR0L_TXRDY)) != 0)
USB->CSR0H_TXCSRH |= ALD_USB_CSR0H_FLUSH;
}
else {
if (flags & (ALD_USB_EP_HOST_OUT | ALD_USB_EP_DEV_IN)) {
if (USB->CSR0L_TXCSRL & ALD_USB_TXCSRL_TXRDY)
USB->CSR0L_TXCSRL |= ALD_USB_TXCSRL_FLUSH;
}
else {
if (USB->RXCSRL & ALD_USB_RXCSRL_RXRDY)
USB->RXCSRL |= ALD_USB_RXCSRL_FLUSH;
}
}
return;
}
/**
* @}
*/
/** @defgroup USB_Public_Functions_Group6 Interrupt functions
* @brief Interrupt functions
* @{
*/
/**
* @brief Disable interrupt.
* @param flags: Type of the interrupt.
* @retval None
*/
void ald_usb_int_disable(uint32_t flags)
{
if (flags & ALD_USB_IDR_STATUS)
USB->IDR |= (uint8_t)(flags & ALD_USB_IDR_STATUS);
return;
}
/**
* @brief Enable interrupt.
* @param flags: Type of the interrupt.
* @retval None
*/
void ald_usb_int_enable(uint32_t flags)
{
if (flags & ALD_USB_IER_STATUS)
USB->IER |= (uint8_t)(flags & ALD_USB_IER_STATUS);
return;
}
/**
* @brief Gets the status of the interrupt.
* @retval Status.
*/
uint32_t ald_usb_int_status_get(void)
{
uint32_t Status;
Status = USB->IFM & 0x7F;
USB->ICR |= Status;
return Status;
}
/**
* @brief Disable interrupt of the endpoint.
* @param flags: Type of the interrupt.
* @retval None
*/
void ald_usb_int_disable_ep(uint32_t flags)
{
USB->TXIDR |= (uint8_t)(flags & (ALD_USB_INTEP_HOST_OUT | ALD_USB_INTEP_DEV_IN | ALD_USB_INTEP_0));
USB->RXIDR |= (uint8_t)((flags & (ALD_USB_INTEP_HOST_IN | ALD_USB_INTEP_DEV_OUT)) >> ALD_USB_INTEP_RX_SHIFT);
return;
}
/**
* @brief Enable interrupt of the endpoint.
* @param flags: Type of the interrupt.
* @retval None
*/
void ald_usb_int_enable_ep(uint32_t flags)
{
USB->TXIER |= (uint8_t)(flags & (ALD_USB_INTEP_HOST_OUT | ALD_USB_INTEP_DEV_IN | ALD_USB_INTEP_0));
USB->RXIER |= (uint8_t)((flags & (ALD_USB_INTEP_HOST_IN | ALD_USB_INTEP_DEV_OUT)) >> ALD_USB_INTEP_RX_SHIFT);
return;
}
/**
* @brief Gets the ststus of the endpoint interrupt.
* @retval Status.
*/
uint32_t ald_usb_int_status_ep_get(void)
{
uint32_t status;
status = USB->TXIFM;
status |= (USB->RXIFM << ALD_USB_INTEP_RX_SHIFT);
USB->TXICR |= (uint8_t)(status & 0xFF);
USB->RXICR |= (uint8_t)((status >> ALD_USB_INTEP_RX_SHIFT) & 0xFF);
return status;
}
/**
* @brief Register USB's interrupt.
* @retval None
*/
void ald_usb_int_register(void)
{
//ald_mcu_irq_config(USB_IRQn, 2, ENABLE);
csi_vic_set_prio(USB_IRQn, 3);
csi_vic_enable_sirq(USB_IRQn);
return;
}
/**
* @brief Unregister USB's interrupt.
* @retval None
*/
void ald_usb_int_unregister(void)
{
//ald_mcu_irq_config(USB_IRQn, 2, DISABLE);
csi_vic_disable_sirq(USB_IRQn);
return;
}
/**
* @brief Get USB's interrupt number.
* @retval None
*/
uint32_t ald_usb_int_num_get(void)
{
return USB_IRQn;
}
/**
* @}
*/
/** @defgroup USB_Public_Functions_Group7 DMA functions
* @brief DMA functions
* @{
*/
#if defined(__ALD_MDA_H__)
/**
* @brief Configure DMA's channel.
* @param ch: Channel.
* @param addr: Address.
* @param count: Size of the data to be moved.
* @param ctrl: Parameters of the DMA's controler
* @retval None
*/
void ald_usb_dma_channel_config(uint8_t ch, uint32_t addr, uint32_t count, uint32_t ctrl)
{
return;
}
/**
* @brief Start multiple receive.
* @param ep_idx: Index of the endpoint
* @retval None
*/
void ald_usb_dma_mult_recv_start(uint32_t ep_idx)
{
return;
}
/**
* @brief Start DMA's machine.
* @param ch: Channel.
* @retval None
*/
void ald_usb_dma_channel_start(uint8_t ch)
{
return;
}
/**
* @brief Stop DMA's machine.
* @param ch: Channel.
* @retval None
*/
void ald_usb_dma_channel_stop(uint8_t ch)
{
return;
}
/**
* @brief Gets flags of the interrupt.
* @retval Flags of the interrupt.
*/
uint32_t ald_usb_dma_get_interrupt_flag(void)
{
return;
}
/**
* @brief Gets the status of the error.
* @param ch: Channel.
* @retval Status.
*/
uint32_t ald_usb_dma_get_channel_error(uint8_t ch)
{
return 0;
}
/**
* @brief Clear the status of the error.
* @param ch: Channel.
* @retval None
*/
void ald_usb_dma_clear_channel_error(uint8_t ch)
{
return;
}
#endif
/**
* @}
*/
/** @defgroup USB_Public_Functions_Group8 LPM functions
* @brief LPM functions
* @{
*/
/**
* @brief Gets status of remote wakeup.
* @retval Status.
*/
uint32_t ald_usb_lpm_remote_wake_is_enable(void)
{
return 1;
}
/**
* @brief Gets the link status
* @retval Status
*/
uint32_t ald_usb_lpm_link_status_get(void)
{
return 0;
}
/**
* @brief Gets the index of the endpoint.
* @retval Index of the endpoint.
*/
uint32_t ald_usb_lpm_ep_get(void)
{
return 0;
}
/**
* @brief Gets the status of the interrupt.
* @retval Status.
*/
uint32_t ald_usb_lpm_int_status_get(void)
{
return 0;
}
/**
* @brief Disable the LPM interrupt.
* @retval None
*/
void ald_usb_lpm_int_disable(uint32_t ints)
{
return;
}
/**
* @brief Enable the LPM interrupt.
* @retval None
*/
void ald_usb_lpm_int_enable(uint32_t ints)
{
return;
}
/**
* @brief Transmit a LPM transaction in host mode.
* @param addr: Address.
* @param ep_idx: Index of the endpoint.
* @retval None
*/
void ald_usb_host_lpm_send(uint32_t addr, uint32_t ep_idx)
{
return;
}
/**
* @brief Configure the LPM parameters in host mode.
* @param resume_time: Resume time.
* @param config: Parameters
* @retval None
*/
void ald_usb_host_lpm_config(uint32_t resume_time, uint32_t config)
{
return;
}
/**
* @brief Initiate a RESUME from the L1 state in host mode.
* @retval None
*/
void ald_usb_host_lpm_resume(void)
{
return;
}
/**
* @brief Enable remote wakeup in device mode.
* @retval None
*/
void ald_usb_dev_lpm_remote_wake(void)
{
return;
}
/**
* @brief Enable remote wakeup in device mode.
* @retval None
*/
void ald_usb_dev_lpm_config(uint32_t config)
{
return;
}
/**
* @brief Enable LPM in device mode.
* @retval None
*/
void ald_usb_dev_lpm_enable(void)
{
return;
}
/**
* @brief Disable LPM in device mode.
* @retval None
*/
void ald_usb_dev_lpm_disable(void)
{
return;
}
/**
* @}
*/
/** @defgroup USB_Public_Functions_Group10 LPM functions
* @brief USB SWVBUS control
* @{
*/
/**
* @brief Selet the control mode of VBUS.
* @param sigctl: 0, indicates that use the hardware control
* sigctl: 1, indicates that use the software control.
* @retval Status.
*/
void ald_usb_swvbus_sigctl_set(uint8_t sigctl)
{
if (sigctl == 1)
{
USB->SWVBUS |= 0x01 << 0;
}
else
{
USB->SWVBUS &= ~(0x01 << 0);
}
return;
}
/**
* @brief Selet the control mode of VBUS.
* @param None.
* @retval Status: 0, indicates that use the hardware control
* Status: 1, indicates that use the software control.
*/
uint8_t ald_usb_swvbus_sigctl_get(void)
{
return (USB->SWVBUS & (0x01 << 0));
}
/**
* @brief Set session end threshold.
* @param thd: 0, indicates that lower than the session end threshold
* thd: 1, indicates that higher than the session end threshold.
* @retval Status.
*/
void ald_usb_swvbus_sesendth_set(uint8_t thd)
{
if (thd == 1)
{
USB->SWVBUS |= 0x01 << 1;
}
else
{
USB->SWVBUS &= ~(0x01 << 1);
}
return;
}
/**
* @brief Set session end threshold.
* @param None.
* @retval Status: 0, indicates that lower than the session end threshold
* Status: 1, indicates that higher than the session end threshold.
*/
uint8_t ald_usb_swvbus_sesendth_get(void)
{
return (USB->SWVBUS & (0x01 << 1));
}
/**
* @brief Set session valid threshold.
* @param thd: 0, indicates that lower than the session valid threshold
thd: 1, indicates that higher than the session valid threshold.
* @retval Status.
*/
void ald_usb_swvbus_sesvalth_set(uint8_t thd)
{
if (thd == 1)
{
USB->SWVBUS |= 0x01 << 2;
}
else
{
USB->SWVBUS &= ~(0x01 << 2);
}
return;
}
/**
* @brief Set session valid threshold.
* @param None.
* @retval Status: 0, indicates that lower than the session valid threshold
Status: 1, indicates that higher than the session valid threshold.
*/
uint8_t ald_usb_swvbus_sesvalth_get(void)
{
return (USB->SWVBUS & (0x01 << 2));
}
/**
* @brief Set VBUS valid threshold.
* @param thd: 0, indicates that lower than the vbus valid threshold
thd: 1, indicates that higher than the vbus valid threshold.
* @retval Status.
*/
void ald_usb_swvbus_valth_set(uint8_t thd)
{
if (thd == 1)
{
USB->SWVBUS |= 0x01 << 3;
}
else
{
USB->SWVBUS &= ~(0x01 << 3);
}
return;
}
/**
* @brief Set VBUS valid threshold.
* @param None.
* @retval Status: 0, indicates that lower than the vbus valid threshold
Status:thd: 1, indicates that higher than the vbus valid threshold.
*/
uint8_t ald_usb_swvbus_valth_get(void)
{
return (USB->SWVBUS & (0x01 << 3));
}
/**
* @}
*/
/** @defgroup USB_Public_Functions_Group11 components initialization functions
* @brief USB components initialization
* @{
*/
/**
* @brief Initialize usb host components.
* @retval None
*/
void ald_usb_host_components_init(void)
{
WRITE_REG(SYSCFG->PROT, 0x55AA6996U);
RMU->AHB1RSTR |= 0x40;
RMU->AHB1RSTR &= 0xFFFFFFBF;
WRITE_REG(SYSCFG->PROT, 0x0);
/* Config EP0 */
ald_usb_host_ep_config(ALD_USB_EP_0, 64, 0, 0, (ALD_USB_EP_MODE_CTRL | ALD_USB_EP_SPEED_FULL | ALD_USB_EP_HOST_OUT));
/* Clear interrupts */
/* Enable PHY power */
ald_usb_host_pwr_enable();
/* clear hnp session */
ald_usb_otg_session_request(false);
/* set vbus control mode and threshold value */
ald_usb_swvbus_sigctl_set(1);
ald_usb_swvbus_sesendth_set(1);
ald_usb_swvbus_sesvalth_set(1);
ald_usb_swvbus_valth_set(1);
/* Pull down DP and DM */
ald_usb_dppud_set(ALD_USB_DPDM_PUSH_DOWN);
ald_usb_dmpud_set(ALD_USB_DPDM_PUSH_DOWN);
/* software control CID */
ald_usb_swcid_cidctrl(1);
/* force to host mode */
ald_usb_swcid_host(0);
/* start host request */
ald_usb_mode_host_req();
/* Start hnp */
ald_usb_otg_session_request(true);
/* Clear interrupts */
ald_usb_int_status_get();
/* Init interrupts */
ald_usb_int_enable(ALD_USB_INTCTRL_SESSION | ALD_USB_INTCTRL_DISCONNECT | ALD_USB_INTCTRL_CONNECT | ALD_USB_INTCTRL_SOF |
ALD_USB_INTCTRL_BABBLE | ALD_USB_INTCTRL_RESUME);
ald_usb_int_enable_ep(ALD_USB_INTEP_ALL);
ald_usb_int_register();
return;
}
/**
* @brief Initialize usb device components.
* @retval None
*/
void ald_usb_device_components_init(void)
{
ald_usb_otg_session_request(true);
#if (defined(ES32VF2264))
ald_usb_dppud_set(ALD_USB_DPDM_FLOATING);/*测试板的USBDP(PA12)上拉到3.3V。具体情况请参考ReadMe.txt*/
#else
ald_usb_dppud_set(USB_DPDM_PUSH_UP);
#endif /* ES32VF2264 */
/* software control CID */
ald_usb_swcid_cidctrl(1);
/* force to dev mode */
ald_usb_swcid_host(1);
ald_usb_dev_suspend_enable();
return;
}
/**
* @}
*/
/**
* @}
*/
#endif /* ALD_USB */
/**
* @}
*/
/**
* @}
*/