/* * Copyright (C) 2021 Mesotic SAS * * This file is subject to the terms and conditions of the GNU Lesser * General Public License v2.1. See the file LICENSE in the top level * directory for more details. */ /** * @ingroup cpu_stm32 * @{ * @file * @brief Low level USB interface functions for the stm32 FS devices * * @author Dylan Laduranty * @} */ #define USB_H_USER_IS_RIOT_INTERNAL #include #include #include #include #include "bitarithm.h" #include "cpu.h" #include "cpu_conf.h" #include "periph/pm.h" #include "periph/gpio.h" #include "periph/usbdev.h" #include "usbdev_stm32.h" #include "pm_layered.h" #include /** * Be careful with enabling debug here. As with all timing critical systems it * is able to interfere with USB functionality and you might see different * errors than debug disabled */ #define ENABLE_DEBUG 0 #include "debug.h" #ifndef USB1_PMAADDR #define USB1_PMAADDR USB_PMAADDR #endif /* ndef USB1_PMAADDR */ #define _ENDPOINT_NUMOF (8) /* IP specific */ #define STM32_FS_BUF_SPACE (1024) /* Base Buffer address located in USB SRAM area */ #define BUF_BASE_OFFSET (_ENDPOINT_NUMOF * 8) /* List of instantiated USB peripherals */ static stm32_usbdev_fs_t _usbdevs[USBDEV_NUMOF]; static usbdev_ep_t _ep_in[_ENDPOINT_NUMOF]; static usbdev_ep_t _ep_out[_ENDPOINT_NUMOF]; static uint8_t* _ep_out_buf[_ENDPOINT_NUMOF]; static uint8_t* _app_pbuf[_ENDPOINT_NUMOF]; static uint32_t _ep_in_buf[_ENDPOINT_NUMOF]; /* Forward declaration for the usb device driver */ const usbdev_driver_t driver; static USB_TypeDef *_global_regs(const stm32_usbdev_fs_config_t *conf) { return (USB_TypeDef *)conf->base_addr; } /* Endpoint Buffer Descriptor */ typedef struct { uint16_t addr_tx; uint16_t count_tx; uint16_t addr_rx; uint16_t count_rx; } ep_buf_desc_t; #define EP_DESC ((ep_buf_desc_t*)USB1_PMAADDR) #define EP_REG(x) (*((volatile uint32_t*) (((uintptr_t)(USB)) + (4*x)))) usbdev_t *usbdev_get_ctx(unsigned num) { assert(num < USBDEV_NUMOF); return &_usbdevs[num].usbdev; } static inline void _enable_irq(void) { stm32_usbdev_fs_t *usbdev = &_usbdevs[0]; const stm32_usbdev_fs_config_t *conf = usbdev->config; _global_regs(conf)->CNTR = USB_CNTR_CTRM | USB_CNTR_RESETM | USB_CNTR_ERRM | USB_CNTR_PMAOVRM; } static void _setup(stm32_usbdev_fs_t *usbdev, const stm32_usbdev_fs_config_t *config) { usbdev->usbdev.driver = &driver; usbdev->config = config; usbdev->out = _ep_out; usbdev->in = _ep_in; } static uint32_t _type_to_reg(usb_ep_type_t type) { switch (type) { case USB_EP_TYPE_CONTROL: return USB_EP_CONTROL; case USB_EP_TYPE_ISOCHRONOUS: return USB_EP_ISOCHRONOUS; case USB_EP_TYPE_BULK: return USB_EP_BULK; case USB_EP_TYPE_INTERRUPT: return USB_EP_INTERRUPT; default: assert(false); return 0; } } void usbdev_init_lowlevel(void) { for (size_t i = 0; i < USBDEV_NUMOF; i++) { _setup(&_usbdevs[i], &stm32_usbdev_fs_config[i]); } } /*TODO: rework this function (maybe move it outside this file) for other ST families support */ static void _enable_usb_clk(void) { #if defined(PWR_CR2_USV) /* Validate USB Supply */ PWR->CR2 |= PWR_CR2_USV; #endif #if defined(RCC_APB1SMENR_USBSMEN) RCC->APB1SMENR |= RCC_APB1SMENR_USBSMEN; #elif defined(RCC_APB1SMENR_USBSMEN) RCC->APB1SMENR1 |= RCC_APB1SMENR1_USBSMEN; #endif /* Enable CRS with auto trim enabled */ CRS->CR |= (CRS_CR_AUTOTRIMEN | CRS_CR_CEN); } static void _enable_gpio(const stm32_usbdev_fs_config_t *conf) { gpio_init(conf->dp, GPIO_IN); gpio_init(conf->dm, GPIO_IN); /* Configure AF for the pins */ gpio_init_af(conf->dp, conf->af); gpio_init_af(conf->dm, conf->af); } static void _set_ep_in_status(uint16_t *val, uint16_t mask) { /* status endpoints bits can only be toggled, writing 0 as no effect. Thus, check which bits should be set then XOR it with their current value */ if (mask & USB_EPTX_DTOG1) { *val ^= USB_EPTX_DTOG1; } if (mask & USB_EPTX_DTOG2) { *val ^= USB_EPTX_DTOG2; } } static void _set_ep_out_status(uint16_t *val, uint16_t mask) { /* status endpoints bits can only be toggled, writing 0 as no effect. Thus, check which bits should be set then XOR it with their current value */ if (mask & USB_EPRX_DTOG1) { *val ^= USB_EPRX_DTOG1; } if (mask & USB_EPRX_DTOG2) { *val ^= USB_EPRX_DTOG2; } } static inline void _usb_attach(stm32_usbdev_fs_t *usbdev) { /* Some ST USB IP doesn't have this feature */ #ifdef USB_BCDR_DPPU const stm32_usbdev_fs_config_t *conf = usbdev->config; /* Enable DP pullup to signal connection */ _global_regs(conf)->BCDR |= USB_BCDR_DPPU; while (!(CRS->ISR & CRS_ISR_ESYNCF)); #else (void)usbdev; #endif /* USB_BCDR_DPPU */ } static inline void _usb_detach(stm32_usbdev_fs_t *usbdev) { /* Some ST USB IP doesn't have this feature */ #ifdef USB_BCDR_DPPU const stm32_usbdev_fs_config_t *conf = usbdev->config; DEBUG_PUTS("usbdev_fs: Detaching from host"); /* Disable DP pullup to signal disconnection */ CLRBIT(_global_regs(conf)->BCDR, USB_BCDR_DPPU); #else (void)usbdev; #endif } static void _set_address(stm32_usbdev_fs_t *usbdev, uint8_t address) { const stm32_usbdev_fs_config_t *conf = usbdev->config; _global_regs(conf)->DADDR = USB_DADDR_EF | (address & USB_DADDR_ADD); } void USBDEV_ISR(void) { stm32_usbdev_fs_t *usbdev = &_usbdevs[0]; const stm32_usbdev_fs_config_t *conf = usbdev->config; int irq_status = _global_regs(conf)->ISTR; /* Disable interrupts temporarily */ _global_regs(conf)->CNTR = 0; /* Correct transfer Interrupt */ if (_global_regs(conf)->ISTR & USB_ISTR_CTR) { /* Clear pending interrupts */ _global_regs(conf)->ISTR = 0; /* Get endpoint number and associated ep pointer */ unsigned epnum = irq_status & 0x000F; usbdev_ep_t *ep = (irq_status & USB_ISTR_DIR) ? &_ep_out[epnum] : &_ep_in[epnum]; /* get type and add endpoint address */ uint16_t reg = _type_to_reg(ep->type) | ep->num; if (irq_status & USB_ISTR_DIR) { /* Clear RX CTR by writing 0, avoid clear TX CTR so leave it to one */ EP_REG(epnum) = reg | USB_EP_CTR_TX; usbdev->usbdev.epcb(ep, USBDEV_EVENT_ESR); } else { /* Clear TX CTR by writing 0, avoid clear RX CTR so leave it to one */ EP_REG(epnum) = reg | USB_EP_CTR_RX; usbdev->usbdev.epcb(ep, USBDEV_EVENT_ESR); } } else if (_global_regs(conf)->ISTR & USB_ISTR_RESET) { usbdev->usbdev.cb(&usbdev->usbdev, USBDEV_EVENT_ESR); } else if (_global_regs(conf)->ISTR & USB_ISTR_PMAOVR) { _global_regs(conf)->ISTR = 0; } else { _global_regs(conf)->ISTR = 0; } cortexm_isr_end(); } static void _usbdev_init(usbdev_t *dev) { stm32_usbdev_fs_t *usbdev = (stm32_usbdev_fs_t *)dev; const stm32_usbdev_fs_config_t *conf = usbdev->config; DEBUG_PUTS("usbdev_fs: initialization"); /* Block STOP/STANDBY */ pm_block(STM32_PM_STOP); pm_block(STM32_PM_STANDBY); /* Enable the clock to the peripheral */ periph_clk_en(conf->apb, conf->rcc_mask); /* Enable USB clock */ _enable_usb_clk(); /* Configure GPIOs */ _enable_gpio(conf); /* Reset USB IP */ _global_regs(conf)->CNTR = USB_CNTR_FRES; /* Clear reset */ _global_regs(conf)->CNTR = 0; /* Clear interrupt register */ _global_regs(conf)->ISTR = 0x0000; /* Set BTABLE at start of USB SRAM */ _global_regs(conf)->BTABLE = 0x0000; /* Set default address after reset */ _global_regs(conf)->DADDR = USB_DADDR_EF; /* Unmask the interrupt in the NVIC */ _enable_irq(); /* Enable USB IRQ */ NVIC_EnableIRQ(conf->irqn); /* fill USB SRAM with zeroes */ memset((uint8_t*)USB1_PMAADDR, 0, 1024); } static usbdev_ep_t *_get_ep(stm32_usbdev_fs_t *usbdev, unsigned num, usb_ep_dir_t dir) { if (num >= _ENDPOINT_NUMOF) { return NULL; } return dir == USB_EP_DIR_IN ? &usbdev->in[num] : &usbdev->out[num]; } static bool _ep_check(usbdev_ep_t* ep) { /* Check if endpoint already configured */ if (ep->dir == USB_EP_DIR_IN) { if ((EP_REG(ep->num) & USB_EPTX_STAT) == USB_EP_TX_DIS) { return true; } } else { if ((EP_REG(ep->num) & USB_EPRX_STAT) == USB_EP_RX_DIS) { return true; } } return false; } static void _ep_enable(usbdev_ep_t *ep) { uint16_t reg = EP_REG(ep->num); /* Avoid modification of the following registers */ SETBIT(reg, USB_EP_CTR_RX | USB_EP_CTR_TX); if (ep->dir == USB_EP_DIR_OUT) { _set_ep_out_status(®, USB_EP_RX_NAK); /* Avoid toggling the other direction */ CLRBIT(reg, USB_EP_TX_VALID); } else { _set_ep_in_status(®, USB_EP_TX_NAK); /* Avoid toggling the other direction */ CLRBIT(reg, USB_EP_RX_VALID); } EP_REG(ep->num) = reg; } static void _ep_disable(usbdev_ep_t *ep) { uint16_t reg = EP_REG(ep->num); /* Avoid modification of the following registers */ SETBIT(reg, USB_EP_CTR_RX | USB_EP_CTR_TX); if (ep->dir == USB_EP_DIR_OUT) { _set_ep_out_status(®, USB_EP_RX_DIS); /* Avoid toggling the other direction */ CLRBIT(reg, USB_EP_TX_VALID); } else { _set_ep_in_status(®, USB_EP_TX_DIS); /* Avoid toggling the other direction */ CLRBIT(reg, USB_EP_RX_VALID); } EP_REG(ep->num) = reg; } static usbdev_ep_t *_usbdev_new_ep(usbdev_t *dev, usb_ep_type_t type, usb_ep_dir_t dir, size_t len) { stm32_usbdev_fs_t *usbdev = (stm32_usbdev_fs_t *)dev; /* The IP supports all types for all endpoints */ usbdev_ep_t *new_ep = NULL; /* Always return endpoint 0 for control types as requested by IP */ if (type == USB_EP_TYPE_CONTROL) { new_ep = _get_ep(usbdev, 0, dir); new_ep->num = 0; } else { /* Find the first unassigned ep with proper dir */ for (unsigned idx = 1; idx < _ENDPOINT_NUMOF && !new_ep; idx++) { usbdev_ep_t *ep = _get_ep(usbdev, idx, dir); /* Check if endpoint is available */ bool avail = _ep_check(ep); if (ep->type == USB_EP_TYPE_NONE && avail) { new_ep = ep; new_ep->num = idx; } } } if (new_ep) { if (usbdev->used + len < STM32_FS_BUF_SPACE) { if (dir == USB_EP_DIR_IN) { _ep_in_buf[new_ep->num] = (BUF_BASE_OFFSET + usbdev->used); } else { _ep_out_buf[new_ep->num] = (uint8_t*)(BUF_BASE_OFFSET + usbdev->used); } usbdev->used += len; new_ep->len = len; new_ep->type = type; new_ep->dev = dev; new_ep->dir = dir; DEBUG("usbdev_fs: register new %s endpoint as EP%d\n", new_ep->dir == USB_EP_DIR_IN ? "IN": "OUT", new_ep->num); DEBUG("usbdev_fs: %d bytes used\n", usbdev->used); } else { assert(0); } } return new_ep; } static int _usbdev_get(usbdev_t *usbdev, usbopt_t opt, void *value, size_t max_len) { (void)usbdev; (void)max_len; int res = -ENOTSUP; switch (opt) { case USBOPT_MAX_VERSION: assert(max_len == sizeof(usb_version_t)); *(usb_version_t *)value = USB_VERSION_20; res = sizeof(usb_version_t); break; case USBOPT_MAX_SPEED: assert(max_len == sizeof(usb_speed_t)); *(usb_speed_t *)value = USB_SPEED_FULL; res = sizeof(usb_speed_t); break; default: DEBUG("Unhandled get call: 0x%x\n", opt); break; } return res; } static int _usbdev_set(usbdev_t *dev, usbopt_t opt, const void *value, size_t value_len) { (void)value_len; stm32_usbdev_fs_t *usbdev = (stm32_usbdev_fs_t *)dev; int res = -ENOTSUP; switch (opt) { case USBOPT_ADDRESS: assert(value_len == sizeof(uint8_t)); uint8_t addr = (*((uint8_t *)value)); _set_address(usbdev, addr); break; case USBOPT_ATTACH: assert(value_len == sizeof(usbopt_enable_t)); if (*((usbopt_enable_t *)value)) { _usb_attach(usbdev); } else { _usb_detach(usbdev); } res = sizeof(usbopt_enable_t); break; default: DEBUG("usbdev_fs: Unhandled set call: 0x%x\n", opt); break; } return res; } static void _usbdev_esr(usbdev_t *dev) { stm32_usbdev_fs_t *usbdev = (stm32_usbdev_fs_t *)dev; const stm32_usbdev_fs_config_t *conf = usbdev->config; if (_global_regs(conf)->ISTR & USB_ISTR_RESET) { usbdev->usbdev.cb(&usbdev->usbdev, USBDEV_EVENT_RESET); _global_regs(conf)->ISTR = (uint16_t)~USB_ISTR_RESET; } else { _global_regs(conf)->ISTR = 0x0000; } _enable_irq(); } static void _usbdev_ep_init(usbdev_ep_t *ep) { /* Set endpoint type */ uint16_t reg = _type_to_reg(ep->type); /* Keep previous state if already set */ reg |= EP_REG(ep->num) & (USB_EPTX_STAT | USB_EPRX_STAT); _set_ep_out_status(®, USB_EP_RX_VALID); /* Assign EP address */ reg |= (ep->num & 0x000F); /* Ensure interrupts are cleared */ reg |= USB_EP_CTR_RX | USB_EP_CTR_TX; EP_REG(ep->num) = reg; if (ep->dir == USB_EP_DIR_IN) { EP_DESC[ep->num].addr_tx = _ep_in_buf[ep->num]; EP_DESC[ep->num].count_tx = 64; } else { EP_DESC[ep->num].addr_rx = (uint32_t)_ep_out_buf[ep->num]; /* Set BLSIZE + NUM_BLOCK (1) */ EP_DESC[ep->num].count_rx = (1 << 10) | (1 << 15); } } static usbopt_enable_t _ep_get_stall(usbdev_ep_t *ep) { usbopt_enable_t res; if (ep->dir == USB_EP_DIR_IN) { res = ((EP_REG(ep->num) & USB_EPTX_STAT) == USB_EP_TX_STALL) ? USBOPT_ENABLE : USBOPT_DISABLE; } else { res = ((EP_REG(ep->num) & USB_EPRX_STAT) == USB_EP_RX_STALL) ? USBOPT_ENABLE : USBOPT_DISABLE; } return res; } static size_t _ep_get_available(usbdev_ep_t *ep) { size_t size; if (ep->dir == USB_EP_DIR_OUT) { size = EP_DESC[ep->num].count_rx & 0x03FF; } else { size = EP_DESC[ep->num].count_tx; } return size; } static int _usbdev_ep_get(usbdev_ep_t *ep, usbopt_ep_t opt, void *value, size_t max_len) { (void)max_len; int res = -ENOTSUP; switch (opt) { case USBOPT_EP_STALL: assert(max_len == sizeof(usbopt_enable_t)); *(usbopt_enable_t *)value = _ep_get_stall(ep); res = sizeof(usbopt_enable_t); break; case USBOPT_EP_AVAILABLE: assert(max_len == sizeof(size_t)); *(size_t *)value = _ep_get_available(ep); res = sizeof(size_t); break; default: DEBUG("usbdev_fs: Unhandled get call: 0x%x\n", opt); break; } return res; } static void _ep_set_stall(usbdev_ep_t *ep, usbopt_enable_t enable) { uint16_t reg = EP_REG(ep->num); /* Avoid modification of the following registers */ SETBIT(reg, USB_EP_CTR_RX | USB_EP_CTR_TX); if (ep->dir == USB_EP_DIR_IN) { _set_ep_in_status(®, enable ? USB_EP_TX_STALL : USB_EP_TX_NAK); /* Avoid toggling the other direction */ CLRBIT(reg, USB_EP_RX_VALID); } else { _set_ep_out_status(®, enable ? USB_EP_RX_STALL : USB_EP_RX_NAK); /* Avoid toggling the other direction */ CLRBIT(reg, USB_EP_TX_VALID); } EP_REG(ep->num) = reg; } static int _usbdev_ep_set(usbdev_ep_t *ep, usbopt_ep_t opt, const void *value, size_t value_len) { (void)value_len; int res = -ENOTSUP; switch (opt) { case USBOPT_EP_ENABLE: assert(value_len == sizeof(usbopt_enable_t)); if (*((usbopt_enable_t *)value)) { _usbdev_ep_init(ep); _ep_enable(ep); } else { _ep_disable(ep); } res = sizeof(usbopt_enable_t); break; case USBOPT_EP_STALL: assert(value_len == sizeof(usbopt_enable_t)); _ep_set_stall(ep, *(usbopt_enable_t *)value); res = sizeof(usbopt_enable_t); break; default: DEBUG("usbdev_fs: Unhandled endpoint set call: 0x%x\n", opt); break; } return res; } static void _usbdev_ep_esr(usbdev_ep_t *ep) { if (ep->dir == USB_EP_DIR_OUT) { /* Copy SRAM buffer to OUT buffer */ uint8_t* tmp_ptr = (uint8_t*)(USB1_PMAADDR + EP_DESC[ep->num].addr_rx); for (uint8_t cpt=0; cpt<64; cpt++) { _app_pbuf[ep->num][cpt] = tmp_ptr[cpt]; } } ep->dev->epcb(ep, USBDEV_EVENT_TR_COMPLETE); _enable_irq(); } static int _usbdev_ep_xmit(usbdev_ep_t *ep, uint8_t* buf, size_t len) { unsigned irq = irq_disable(); uint16_t reg = EP_REG(ep->num); /* Avoid modification of the following registers */ SETBIT(reg, USB_EP_CTR_RX | USB_EP_CTR_TX); if (ep->dir == USB_EP_DIR_OUT) { _set_ep_out_status(®, USB_EP_RX_VALID); _set_ep_in_status(®, USB_EP_TX_NAK); if (len == 0) { len = ep->len; } if (len < 63) { /* Should write len / 2 in count_rx if len <= 62 */ EP_DESC[ep->num].count_rx = (len >> 1) << 10; } else { /* BL_SIZE = 1, NUM_BLOCK = 1 */ EP_DESC[ep->num].count_rx = 1 << 15 | 1 << 10; } CLRBIT(reg, USB_EP_DTOG_TX | USB_EP_DTOG_RX); _app_pbuf[ep->num] = buf; } else { _set_ep_in_status(®, USB_EP_TX_VALID); _set_ep_out_status(®, USB_EP_RX_VALID); /* Transfer IN buffer content to USB SRAM */ uint16_t* ptr = (uint16_t*) (USB1_PMAADDR + EP_DESC[ep->num].addr_tx); for (unsigned tmp=0; tmp < len; tmp+=2) { size_t index = tmp/2; size_t off = tmp; ptr[index] = (buf[off+1] << 8) | buf[off]; } /* Manage odd transfer size */ if (len % 2 == 1) { size_t off = (len/2) + 1; ptr[off] = 0x00FF & buf[len-1]; } EP_DESC[ep->num].count_tx = len; CLRBIT(reg, USB_EP_DTOG_TX | USB_EP_DTOG_RX); } EP_REG(ep->num) = reg; irq_restore(irq); return 0; } const usbdev_driver_t driver = { .init = _usbdev_init, .new_ep = _usbdev_new_ep, .get = _usbdev_get, .set = _usbdev_set, .esr = _usbdev_esr, .ep_init = _usbdev_ep_init, .ep_get = _usbdev_ep_get, .ep_set = _usbdev_ep_set, .ep_esr = _usbdev_ep_esr, .xmit = _usbdev_ep_xmit, };