stm32: Improve usbfs epr register handling

Replace the set_stat_x_bits() functions with a single calc_epr_bits()
function.  This new function supports setting bits other than the stat
field in the epr register.

Signed-off-by: Kevin O'Connor <kevin@koconnor.net>
This commit is contained in:
Kevin O'Connor 2023-09-23 12:26:21 -04:00
parent aa726cb7cb
commit 1e3ace2170
1 changed files with 41 additions and 40 deletions

View File

@ -1,6 +1,6 @@
// Hardware interface to "fullspeed USB controller" // Hardware interface to "fullspeed USB controller"
// //
// Copyright (C) 2018-2021 Kevin O'Connor <kevin@koconnor.net> // Copyright (C) 2018-2023 Kevin O'Connor <kevin@koconnor.net>
// //
// This file may be distributed under the terms of the GNU GPLv3 license. // This file may be distributed under the terms of the GNU GPLv3 license.
@ -175,26 +175,19 @@ btable_write_packet(int ep, const uint8_t *src, int count)
#define USB_EPR ((volatile uint32_t *)USB_BASE) #define USB_EPR ((volatile uint32_t *)USB_BASE)
#define EPR_TBITS (USB_EP_DTOG_RX | USB_EPRX_STAT \
| USB_EP_DTOG_TX | USB_EPTX_STAT)
#define EPR_RWBITS (USB_EPADDR_FIELD | USB_EP_KIND | USB_EP_TYPE_MASK) #define EPR_RWBITS (USB_EPADDR_FIELD | USB_EP_KIND | USB_EP_TYPE_MASK)
#define EPR_RWCBITS (USB_EP_CTR_RX | USB_EP_CTR_TX) #define EPR_RWCBITS (USB_EP_CTR_RX | USB_EP_CTR_TX)
// Calculate the memory update needed to set the epr register
static uint32_t static uint32_t
set_stat_rx_bits(uint32_t epr, uint32_t bits) calc_epr_bits(uint32_t epr, uint32_t mask, uint32_t value)
{ {
return ((epr & (EPR_RWBITS | USB_EPRX_STAT)) ^ bits) | EPR_RWCBITS; uint32_t tmask = mask & EPR_TBITS, tvalue = value & tmask;
} uint32_t rwmask = mask & EPR_RWBITS, rwbits = value & rwmask;
uint32_t cbits = EPR_RWCBITS & ~mask;
static uint32_t return (((epr & (EPR_RWBITS | tmask)) ^ tvalue) & ~rwmask) | rwbits | cbits;
set_stat_tx_bits(uint32_t epr, uint32_t bits)
{
return ((epr & (EPR_RWBITS | USB_EPTX_STAT)) ^ bits) | EPR_RWCBITS;
}
static uint32_t
set_stat_rxtx_bits(uint32_t epr, uint32_t bits)
{
uint32_t mask = EPR_RWBITS | USB_EPRX_STAT | USB_EPTX_STAT;
return ((epr & mask) ^ bits) | EPR_RWCBITS;
} }
@ -205,36 +198,37 @@ set_stat_rxtx_bits(uint32_t epr, uint32_t bits)
int_fast8_t int_fast8_t
usb_read_bulk_out(void *data, uint_fast8_t max_len) usb_read_bulk_out(void *data, uint_fast8_t max_len)
{ {
uint32_t epr = USB_EPR[USB_CDC_EP_BULK_OUT]; uint32_t ep = USB_CDC_EP_BULK_OUT, epr = USB_EPR[ep];
if ((epr & USB_EPRX_STAT) == USB_EP_RX_VALID) if ((epr & USB_EPRX_STAT) == USB_EP_RX_VALID)
// No data ready // No data ready
return -1; return -1;
uint32_t count = btable_read_packet(USB_CDC_EP_BULK_OUT, data, max_len); uint32_t count = btable_read_packet(ep, data, max_len);
USB_EPR[USB_CDC_EP_BULK_OUT] = set_stat_rx_bits(epr, USB_EP_RX_VALID); USB_EPR[ep] = calc_epr_bits(epr, USB_EPRX_STAT, USB_EP_RX_VALID);
return count; return count;
} }
int_fast8_t int_fast8_t
usb_send_bulk_in(void *data, uint_fast8_t len) usb_send_bulk_in(void *data, uint_fast8_t len)
{ {
uint32_t epr = USB_EPR[USB_CDC_EP_BULK_IN]; uint32_t ep = USB_CDC_EP_BULK_IN, epr = USB_EPR[ep];
if ((epr & USB_EPTX_STAT) != USB_EP_TX_NAK) if ((epr & USB_EPTX_STAT) != USB_EP_TX_NAK)
// No buffer space available // No buffer space available
return -1; return -1;
btable_write_packet(USB_CDC_EP_BULK_IN, data, len); btable_write_packet(ep, data, len);
USB_EPR[USB_CDC_EP_BULK_IN] = set_stat_tx_bits(epr, USB_EP_TX_VALID); USB_EPR[ep] = calc_epr_bits(epr, USB_EPTX_STAT, USB_EP_TX_VALID);
return len; return len;
} }
int_fast8_t int_fast8_t
usb_read_ep0(void *data, uint_fast8_t max_len) usb_read_ep0(void *data, uint_fast8_t max_len)
{ {
uint32_t epr = USB_EPR[0]; uint32_t ep = 0, epr = USB_EPR[ep];
if ((epr & USB_EPRX_STAT) != USB_EP_RX_NAK) if ((epr & USB_EPRX_STAT) != USB_EP_RX_NAK)
// No data ready // No data ready
return -1; return -1;
uint32_t count = btable_read_packet(0, data, max_len); uint32_t count = btable_read_packet(ep, data, max_len);
USB_EPR[0] = set_stat_rxtx_bits(epr, USB_EP_RX_VALID | USB_EP_TX_NAK); USB_EPR[ep] = calc_epr_bits(epr, USB_EPRX_STAT | USB_EPTX_STAT
, USB_EP_RX_VALID | USB_EP_TX_NAK);
return count; return count;
} }
@ -247,23 +241,24 @@ usb_read_ep0_setup(void *data, uint_fast8_t max_len)
int_fast8_t int_fast8_t
usb_send_ep0(const void *data, uint_fast8_t len) usb_send_ep0(const void *data, uint_fast8_t len)
{ {
uint32_t epr = USB_EPR[0]; uint32_t ep = 0, epr = USB_EPR[ep];
if ((epr & USB_EPRX_STAT) != USB_EP_RX_VALID) if ((epr & USB_EPRX_STAT) != USB_EP_RX_VALID)
// Transfer interrupted // Transfer interrupted
return -2; return -2;
if ((epr & USB_EPTX_STAT) != USB_EP_TX_NAK) if ((epr & USB_EPTX_STAT) != USB_EP_TX_NAK)
// No buffer space available // No buffer space available
return -1; return -1;
btable_write_packet(0, data, len); btable_write_packet(ep, data, len);
USB_EPR[0] = set_stat_tx_bits(epr, USB_EP_TX_VALID); USB_EPR[ep] = calc_epr_bits(epr, USB_EPTX_STAT, USB_EP_TX_VALID);
return len; return len;
} }
void void
usb_stall_ep0(void) usb_stall_ep0(void)
{ {
USB_EPR[0] = set_stat_rxtx_bits(USB_EPR[0] uint32_t ep = 0, epr = USB_EPR[ep];
, USB_EP_RX_STALL | USB_EP_TX_STALL); USB_EPR[ep] = calc_epr_bits(epr, USB_EPRX_STAT | USB_EPTX_STAT
, USB_EP_RX_STALL | USB_EP_TX_STALL);
} }
static uint8_t set_address; static uint8_t set_address;
@ -289,13 +284,20 @@ usb_set_configure(void)
static void static void
usb_reset(void) usb_reset(void)
{ {
USB_EPR[0] = 0 | USB_EP_CONTROL | USB_EP_RX_VALID | USB_EP_TX_NAK; uint32_t ep = 0;
USB_EPR[USB_CDC_EP_ACM] = (USB_CDC_EP_ACM | USB_EP_INTERRUPT USB_EPR[ep] = 0 | USB_EP_CONTROL | USB_EP_RX_VALID | USB_EP_TX_NAK;
| USB_EP_RX_NAK | USB_EP_TX_NAK);
USB_EPR[USB_CDC_EP_BULK_OUT] = (USB_CDC_EP_BULK_OUT | USB_EP_BULK ep = USB_CDC_EP_ACM;
| USB_EP_RX_VALID | USB_EP_TX_NAK); USB_EPR[ep] = (USB_CDC_EP_ACM | USB_EP_INTERRUPT
USB_EPR[USB_CDC_EP_BULK_IN] = (USB_CDC_EP_BULK_IN | USB_EP_BULK | USB_EP_RX_NAK | USB_EP_TX_NAK);
| USB_EP_RX_NAK | USB_EP_TX_NAK);
ep = USB_CDC_EP_BULK_OUT;
USB_EPR[ep] = (USB_CDC_EP_BULK_OUT | USB_EP_BULK
| USB_EP_RX_VALID | USB_EP_TX_NAK);
ep = USB_CDC_EP_BULK_IN;
USB_EPR[ep] = (USB_CDC_EP_BULK_IN | USB_EP_BULK
| USB_EP_RX_NAK | USB_EP_TX_NAK);
USB->CNTR = USB_CNTR_CTRM | USB_CNTR_RESETM; USB->CNTR = USB_CNTR_CTRM | USB_CNTR_RESETM;
USB->DADDR = USB_DADDR_EF; USB->DADDR = USB_DADDR_EF;
@ -308,9 +310,8 @@ USB_IRQHandler(void)
uint32_t istr = USB->ISTR; uint32_t istr = USB->ISTR;
if (istr & USB_ISTR_CTR) { if (istr & USB_ISTR_CTR) {
// Endpoint activity // Endpoint activity
uint32_t ep = istr & USB_ISTR_EP_ID; uint32_t ep = istr & USB_ISTR_EP_ID, epr = USB_EPR[ep];
uint32_t epr = USB_EPR[ep]; USB_EPR[ep] = calc_epr_bits(epr, USB_EP_CTR_RX | USB_EP_CTR_TX, 0);
USB_EPR[ep] = epr & EPR_RWBITS;
if (ep == 0) { if (ep == 0) {
usb_notify_ep0(); usb_notify_ep0();
if (epr & USB_EP_CTR_TX && set_address) { if (epr & USB_EP_CTR_TX && set_address) {