| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425 |
- /*
- * Copyright (c) 2020, Alex Taradov <alex@taradov.com>
- * 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. The name of the author may not 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.
- */
- /*- Includes ----------------------------------------------------------------*/
- #include <string.h>
- #include <stdbool.h>
- #include <stdalign.h>
- #include "same70.h"
- #include "hal_gpio.h"
- #include "utils.h"
- #include "usb.h"
- #include "usb_std.h"
- #include "usb_descriptors.h"
- /*- Definitions -------------------------------------------------------------*/
- #define USBHS_RAM(ep) ((volatile uint8_t *)(0xa0100000 + 0x8000 * (ep)))
- /*- Constants ---------------------------------------------------------------*/
- static const int usb_ep_banks[USB_EP_NUM] = USB_EP_BANKS;
- /*- Variables ---------------------------------------------------------------*/
- static uint8_t usb_ctrl_out_buf[64];
- static void (*usb_control_recv_callback)(uint8_t *data, int size);
- static uint8_t *usb_ep_data[USB_EP_NUM];
- static int usb_ep_size[USB_EP_NUM];
- /*- Implementations ---------------------------------------------------------*/
- //-----------------------------------------------------------------------------
- void usb_hw_init(void)
- {
- PMC->PMC_PCER1 = PMC_PCER1_PID34/*ID_USBHS*/;
- PMC->CKGR_UCKR = CKGR_UCKR_UPLLEN | CKGR_UCKR_UPLLCOUNT(3);
- while (0 == (PMC->PMC_SR & PMC_SR_LOCKU));
- PMC->PMC_USB = PMC_USB_USBS;
- USBHS->USBHS_CTRL = USBHS_CTRL_UIMOD_DEVICE | USBHS_CTRL_USBE;
- while (0 == (USBHS->USBHS_SR & USBHS_SR_CLKUSABLE));
- for (int i = 0; i < USB_EP_NUM; i++)
- usb_reset_endpoint(i, 0);
- usb_detach();
- usb_attach();
- #ifdef USB_ASYNC
- NVIC_EnableIRQ(USBHS_IRQn);
- #endif
- }
- //-----------------------------------------------------------------------------
- void usb_attach(void)
- {
- while (0 == (USBHS->USBHS_SR & USBHS_SR_CLKUSABLE));
- USBHS->USBHS_DEVICR = USBHS_DEVICR_SUSPC | USBHS_DEVICR_MSOFC |
- USBHS_DEVICR_SOFC | USBHS_DEVICR_EORSTC | USBHS_DEVICR_WAKEUPC |
- USBHS_DEVICR_EORSMC | USBHS_DEVICR_UPRSMC;
- USBHS->USBHS_DEVIER = USBHS_DEVIER_EORSTES;
- USBHS->USBHS_DEVCTRL &= ~USBHS_DEVCTRL_DETACH;
- }
- //-----------------------------------------------------------------------------
- void usb_detach(void)
- {
- USBHS->USBHS_DEVCTRL |= USBHS_DEVCTRL_DETACH;
- }
- //-----------------------------------------------------------------------------
- void usb_reset_endpoint(int ep, int dir)
- {
- USBHS->USBHS_DEVEPT |= (USBHS_DEVEPT_EPRST0 << ep);
- USBHS->USBHS_DEVEPT &= ~(USBHS_DEVEPT_EPRST0 << ep);
- USBHS->USBHS_DEVEPTIER[ep] = USBHS_DEVEPTIER_RSTDTS;
- USBHS->USBHS_DEVEPTIDR[ep] = USBHS_DEVEPTIDR_RXOUTEC | USBHS_DEVEPTIDR_TXINEC;
- usb_ep_size[ep] = 0;
- usb_ep_data[ep] = NULL;
- (void)dir;
- }
- //-----------------------------------------------------------------------------
- void usb_configure_endpoint(usb_endpoint_descriptor_t *desc)
- {
- int ep, dir, type, size, banks;
- ep = desc->bEndpointAddress & USB_INDEX_MASK;
- dir = desc->bEndpointAddress & USB_DIRECTION_MASK;
- type = desc->bmAttributes & 0x03;
- size = desc->wMaxPacketSize;
- usb_reset_endpoint(desc->bEndpointAddress, dir);
- if (USB_ISOCHRONOUS_ENDPOINT != type)
- USBHS->USBHS_DEVEPTIER[ep] = USBHS_DEVEPTIER_RSTDTS;
- if (size <= 8)
- size = USBHS_DEVEPTCFG_EPSIZE_8_BYTE;
- else if (size <= 16)
- size = USBHS_DEVEPTCFG_EPSIZE_16_BYTE;
- else if (size <= 32)
- size = USBHS_DEVEPTCFG_EPSIZE_32_BYTE;
- else if (size <= 64)
- size = USBHS_DEVEPTCFG_EPSIZE_64_BYTE;
- else if (size <= 128)
- size = USBHS_DEVEPTCFG_EPSIZE_128_BYTE;
- else if (size <= 256)
- size = USBHS_DEVEPTCFG_EPSIZE_256_BYTE;
- else if (size <= 512)
- size = USBHS_DEVEPTCFG_EPSIZE_512_BYTE;
- else if (size <= 1024)
- size = USBHS_DEVEPTCFG_EPSIZE_1024_BYTE;
- else
- while (1);
- if (USB_CONTROL_ENDPOINT == type)
- type = USBHS_DEVEPTCFG_EPTYPE_CTRL;
- else if (USB_ISOCHRONOUS_ENDPOINT == type)
- type = USBHS_DEVEPTCFG_EPTYPE_ISO;
- else if (USB_BULK_ENDPOINT == type)
- type = USBHS_DEVEPTCFG_EPTYPE_BLK;
- else
- type = USBHS_DEVEPTCFG_EPTYPE_INTRPT;
- if (USB_IN_ENDPOINT == dir)
- dir = USBHS_DEVEPTCFG_EPDIR_IN;
- else
- dir = USBHS_DEVEPTCFG_EPDIR_OUT;
- if (1 == usb_ep_banks[ep])
- banks = USBHS_DEVEPTCFG_EPBK_1_BANK;
- else if (2 == usb_ep_banks[ep])
- banks = USBHS_DEVEPTCFG_EPBK_2_BANK;
- else if (3 == usb_ep_banks[ep])
- banks = USBHS_DEVEPTCFG_EPBK_3_BANK;
- else
- while (1);
- USBHS->USBHS_DEVEPTCFG[ep] = type | size | dir | banks |
- USBHS_DEVEPTCFG_NBTRANS_1_TRANS | USBHS_DEVEPTCFG_ALLOC;
- while (0 == (USBHS->USBHS_DEVEPTISR[ep] & USBHS_DEVEPTISR_CFGOK));
- USBHS->USBHS_DEVEPT |= (USBHS_DEVEPT_EPEN0 << ep);
- USBHS->USBHS_DEVIER = (USBHS_DEVIER_PEP_0 << ep);
- }
- //-----------------------------------------------------------------------------
- bool usb_endpoint_configured(int ep, int dir)
- {
- (void)dir;
- return (0 != (USBHS->USBHS_DEVEPT & (USBHS_DEVEPT_EPEN0 << ep)));
- }
- //-----------------------------------------------------------------------------
- int usb_endpoint_get_status(int ep, int dir)
- {
- (void)dir;
- return (0 != (USBHS->USBHS_DEVEPTIMR[ep] & USBHS_DEVEPTIMR_STALLRQ));
- }
- //-----------------------------------------------------------------------------
- void usb_endpoint_set_feature(int ep, int dir)
- {
- USBHS->USBHS_DEVEPTICR[ep] = USBHS_DEVEPTICR_STALLEDIC;
- USBHS->USBHS_DEVEPTIER[ep] = USBHS_DEVEPTIER_STALLRQS;
- (void)dir;
- }
- //-----------------------------------------------------------------------------
- void usb_endpoint_clear_feature(int ep, int dir)
- {
- if (USBHS->USBHS_DEVEPTIMR[ep] & USBHS_DEVEPTIMR_STALLRQ)
- {
- if (USBHS->USBHS_DEVEPTISR[ep] & USBHS_DEVEPTISR_STALLEDI)
- USBHS->USBHS_DEVEPTICR[ep] = USBHS_DEVEPTICR_STALLEDIC;
- USBHS->USBHS_DEVEPTIDR[ep] = USBHS_DEVEPTIDR_STALLRQC;
- }
- USBHS->USBHS_DEVEPTIER[ep] = USBHS_DEVEPTIER_RSTDTS;
- (void)dir;
- }
- //-----------------------------------------------------------------------------
- void usb_set_address(int address)
- {
- USBHS->USBHS_DEVCTRL &= ~USBHS_DEVCTRL_ADDEN;
- USBHS->USBHS_DEVCTRL &= ~USBHS_DEVCTRL_UADD_Msk;
- USBHS->USBHS_DEVCTRL |= address;
- USBHS->USBHS_DEVCTRL |= USBHS_DEVCTRL_ADDEN;
- }
- //-----------------------------------------------------------------------------
- static bool endpoint_busy(int ep)
- {
- int busy = ((USBHS->USBHS_DEVEPTISR[ep] & USBHS_DEVEPTISR_NBUSYBK_Msk) >> USBHS_DEVEPTISR_NBUSYBK_Pos);
- return (busy == usb_ep_banks[ep]);
- }
- //-----------------------------------------------------------------------------
- static void dma_transfer(int ep, uint8_t *data, int size)
- {
- int ch = ep-1;
- USBHS->USBHS_DEVDMA[ch].USBHS_DEVDMANXTDSC = 0;
- USBHS->USBHS_DEVDMA[ch].USBHS_DEVDMAADDRESS = (uint32_t)data;
- USBHS->USBHS_DEVDMA[ch].USBHS_DEVDMACONTROL = USBHS_DEVDMACONTROL_CHANN_ENB |
- USBHS_DEVDMACONTROL_BUFF_LENGTH(size);
- while (0 == (USBHS->USBHS_DEVDMA[ch].USBHS_DEVDMASTATUS & USBHS_DEVDMASTATUS_END_BF_ST));
- }
- //-----------------------------------------------------------------------------
- void usb_send(int ep, uint8_t *data, int size)
- {
- while (endpoint_busy(ep));
- dma_transfer(ep, data, size);
- USBHS->USBHS_DEVEPTICR[ep] = USBHS_DEVEPTICR_TXINIC;
- USBHS->USBHS_DEVEPTIER[ep] = USBHS_DEVEPTIER_TXINES;
- USBHS->USBHS_DEVEPTIDR[ep] = USBHS_DEVEPTIDR_FIFOCONC;
- }
- //-----------------------------------------------------------------------------
- void usb_recv(int ep, uint8_t *data, int size)
- {
- usb_ep_size[ep] = size;
- usb_ep_data[ep] = data;
- USBHS->USBHS_DEVEPTICR[ep] = USBHS_DEVEPTICR_RXOUTIC;
- USBHS->USBHS_DEVEPTIER[ep] = USBHS_DEVEPTIER_RXOUTES;
- USBHS->USBHS_DEVEPTIDR[ep] = USBHS_DEVEPTIDR_FIFOCONC;
- }
- //-----------------------------------------------------------------------------
- void usb_control_send_zlp(void)
- {
- USBHS->USBHS_DEVEPTICR[0] = USBHS_DEVEPTICR_TXINIC;
- while (0 == (USBHS->USBHS_DEVEPTISR[0] & USBHS_DEVEPTISR_TXINI));
- }
- //-----------------------------------------------------------------------------
- void usb_control_stall(void)
- {
- USBHS->USBHS_DEVEPTIER[0] = USBHS_DEVEPTIER_STALLRQS;
- }
- //-----------------------------------------------------------------------------
- void usb_control_send(uint8_t *data, int size)
- {
- volatile uint8_t *usb_ram = USBHS_RAM(0);
- while (size)
- {
- int transfer_size = LIMIT(size, usb_device_descriptor.bMaxPacketSize0);
- for (int i = 0; i < transfer_size; i++)
- {
- usb_ram[i] = data[i];
- __DSB();
- }
- USBHS->USBHS_DEVEPTICR[0] = USBHS_DEVEPTICR_TXINIC;
- while (0 == (USBHS->USBHS_DEVEPTISR[0] & USBHS_DEVEPTISR_TXINI));
- size -= transfer_size;
- data += transfer_size;
- }
- }
- //-----------------------------------------------------------------------------
- void usb_control_recv(void (*callback)(uint8_t *data, int size))
- {
- usb_control_recv_callback = callback;
- }
- //-----------------------------------------------------------------------------
- static int get_ep_byte_count(int ep)
- {
- return ((USBHS->USBHS_DEVEPTISR[ep] & USBHS_DEVEPTISR_BYCT_Msk) >> USBHS_DEVEPTISR_BYCT_Pos);
- }
- //-----------------------------------------------------------------------------
- static void usb_handler(void)
- {
- if (USBHS->USBHS_DEVISR & USBHS_DEVISR_EORST)
- {
- for (int i = 0; i < USB_EP_NUM; i++)
- usb_reset_endpoint(i, 0);
- usb_set_address(0);
- USBHS->USBHS_DEVEPTIER[0] = USBHS_DEVEPTIER_RSTDTS;
- USBHS->USBHS_DEVEPTCFG[0] = USBHS_DEVEPTCFG_EPTYPE_CTRL |
- USBHS_DEVEPTCFG_ALLOC | USBHS_DEVEPTCFG_EPSIZE_64_BYTE |
- USBHS_DEVEPTCFG_EPBK_1_BANK | USBHS_DEVEPTCFG_NBTRANS_1_TRANS;
- while (0 == (USBHS->USBHS_DEVEPTISR[0] & USBHS_DEVEPTISR_CFGOK));
- USBHS->USBHS_DEVEPT = USBHS_DEVEPT_EPEN0;
- USBHS->USBHS_DEVIER = USBHS_DEVIER_PEP_0;
- USBHS->USBHS_DEVEPTICR[0] = USBHS_DEVEPTICR_RXSTPIC;
- USBHS->USBHS_DEVEPTIER[0] = USBHS_DEVEPTIER_RXSTPES;
- USBHS->USBHS_DEVICR = USBHS_DEVICR_EORSTC;
- }
- if (USBHS->USBHS_DEVEPTISR[0] & USBHS_DEVEPTISR_RXSTPI)
- {
- volatile uint8_t *data = USBHS_RAM(0);
- int size = get_ep_byte_count(0);
- for (int i = 0; i < (int)sizeof(usb_request_t); i++)
- usb_ctrl_out_buf[i] = data[i];
- USBHS->USBHS_DEVEPTICR[0] = USBHS_DEVEPTICR_RXSTPIC;
- if (size == sizeof(usb_request_t))
- {
- if (!usb_handle_standard_request((usb_request_t *)usb_ctrl_out_buf))
- usb_control_stall();
- }
- else
- {
- usb_control_stall();
- }
- }
- else if (USBHS->USBHS_DEVEPTISR[0] & USBHS_DEVEPTISR_RXOUTI)
- {
- USBHS->USBHS_DEVEPTICR[0] = USBHS_DEVEPTICR_RXOUTIC;
- // NOTE: This was not tested
- if (usb_control_recv_callback)
- {
- volatile uint8_t *data = USBHS_RAM(0);
- int size = LIMIT(get_ep_byte_count(0), usb_device_descriptor.bMaxPacketSize0);
- for (int i = 0; i < size; i++)
- usb_ctrl_out_buf[i] = data[i];
- usb_control_recv_callback(usb_ctrl_out_buf, size);
- usb_control_recv_callback = NULL;
- usb_control_send_zlp();
- }
- }
- for (int ep = 1; ep < USB_EP_NUM; ep++)
- {
- int sr = USBHS->USBHS_DEVEPTISR[ep] & USBHS->USBHS_DEVEPTIMR[ep];
- if (sr & USBHS_DEVEPTISR_RXOUTI)
- {
- int size = LIMIT(get_ep_byte_count(ep), usb_ep_size[ep]);
- dma_transfer(ep, usb_ep_data[ep], size);
- USBHS->USBHS_DEVEPTICR[ep] = USBHS_DEVEPTICR_RXOUTIC;
- USBHS->USBHS_DEVEPTIDR[ep] = USBHS_DEVEPTIDR_RXOUTEC;
- usb_recv_callback(ep, size);
- }
- if (sr & USBHS_DEVEPTISR_TXINI)
- {
- USBHS->USBHS_DEVEPTICR[ep] = USBHS_DEVEPTICR_TXINIC;
- USBHS->USBHS_DEVEPTIDR[ep] = USBHS_DEVEPTIDR_TXINEC;
- usb_send_callback(ep);
- }
- }
- }
- //-----------------------------------------------------------------------------
- void irq_handler_usbhs(void)
- {
- #ifdef USB_ASYNC
- usb_handler();
- #endif
- }
- //-----------------------------------------------------------------------------
- void usb_task(void)
- {
- #ifndef USB_ASYNC
- usb_handler();
- #endif
- }
|