can: mcp251xfd: ring: prepare to change order of TX and RX FIFOs
This patch improves the initialization of the TX and RX rings. The initialization functions are now called with pointers to the next free address (in the on chip RAM) and next free hardware FIFO. The rings are initialized using these values and the pointers are modified to point to the next free elements. This means the order of the mcp251xfd_ring_init_*() functions specifies the order of the rings in the hardware FIFO. This makes it possible to change the order of the TX and RX FIFOs, which is done in the next patch. This gives the opportunity to minimize the number of SPI transfers in the IRQ handler. The read of the IRQ status register and RX FIFO status registers can be combined into single SPI transfer. If the RX ring uses FIFO 1, the overall length of the transfer is smaller than in the original layout, where the RX FIFO comes after the TX FIFO. Link: https://lore.kernel.org/all/20220217103826.2299157-4-mkl@pengutronix.de Signed-off-by: Marc Kleine-Budde <mkl@pengutronix.de>
This commit is contained in:
parent
d2d5397fca
commit
617283b9c4
@ -52,7 +52,8 @@ mcp251xfd_cmd_prepare_write_reg(const struct mcp251xfd_priv *priv,
|
|||||||
return len;
|
return len;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void mcp251xfd_ring_init_tef(struct mcp251xfd_priv *priv)
|
static void
|
||||||
|
mcp251xfd_ring_init_tef(struct mcp251xfd_priv *priv, u16 *base)
|
||||||
{
|
{
|
||||||
struct mcp251xfd_tef_ring *tef_ring;
|
struct mcp251xfd_tef_ring *tef_ring;
|
||||||
struct spi_transfer *xfer;
|
struct spi_transfer *xfer;
|
||||||
@ -66,6 +67,9 @@ static void mcp251xfd_ring_init_tef(struct mcp251xfd_priv *priv)
|
|||||||
tef_ring->head = 0;
|
tef_ring->head = 0;
|
||||||
tef_ring->tail = 0;
|
tef_ring->tail = 0;
|
||||||
|
|
||||||
|
/* TEF- and TX-FIFO have same number of objects */
|
||||||
|
*base = mcp251xfd_get_tef_obj_addr(priv->tx->obj_num);
|
||||||
|
|
||||||
/* FIFO increment TEF tail pointer */
|
/* FIFO increment TEF tail pointer */
|
||||||
addr = MCP251XFD_REG_TEFCON;
|
addr = MCP251XFD_REG_TEFCON;
|
||||||
val = MCP251XFD_REG_TEFCON_UINC;
|
val = MCP251XFD_REG_TEFCON_UINC;
|
||||||
@ -127,7 +131,8 @@ mcp251xfd_tx_ring_init_tx_obj(const struct mcp251xfd_priv *priv,
|
|||||||
ARRAY_SIZE(tx_obj->xfer));
|
ARRAY_SIZE(tx_obj->xfer));
|
||||||
}
|
}
|
||||||
|
|
||||||
static void mcp251xfd_ring_init_tx(struct mcp251xfd_priv *priv)
|
static void
|
||||||
|
mcp251xfd_ring_init_tx(struct mcp251xfd_priv *priv, u16 *base, u8 *fifo_nr)
|
||||||
{
|
{
|
||||||
struct mcp251xfd_tx_ring *tx_ring;
|
struct mcp251xfd_tx_ring *tx_ring;
|
||||||
struct mcp251xfd_tx_obj *tx_obj;
|
struct mcp251xfd_tx_obj *tx_obj;
|
||||||
@ -139,9 +144,12 @@ static void mcp251xfd_ring_init_tx(struct mcp251xfd_priv *priv)
|
|||||||
tx_ring = priv->tx;
|
tx_ring = priv->tx;
|
||||||
tx_ring->head = 0;
|
tx_ring->head = 0;
|
||||||
tx_ring->tail = 0;
|
tx_ring->tail = 0;
|
||||||
tx_ring->base = mcp251xfd_get_tef_obj_addr(tx_ring->obj_num);
|
tx_ring->base = *base;
|
||||||
tx_ring->nr = 0;
|
tx_ring->nr = 0;
|
||||||
tx_ring->fifo_nr = MCP251XFD_TX_FIFO;
|
tx_ring->fifo_nr = *fifo_nr;
|
||||||
|
|
||||||
|
*base = mcp251xfd_get_tx_obj_addr(tx_ring, tx_ring->obj_num);
|
||||||
|
*fifo_nr += 1;
|
||||||
|
|
||||||
/* FIFO request to send */
|
/* FIFO request to send */
|
||||||
addr = MCP251XFD_REG_FIFOCON(tx_ring->fifo_nr);
|
addr = MCP251XFD_REG_FIFOCON(tx_ring->fifo_nr);
|
||||||
@ -153,33 +161,25 @@ static void mcp251xfd_ring_init_tx(struct mcp251xfd_priv *priv)
|
|||||||
mcp251xfd_tx_ring_init_tx_obj(priv, tx_ring, tx_obj, len, i);
|
mcp251xfd_tx_ring_init_tx_obj(priv, tx_ring, tx_obj, len, i);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void mcp251xfd_ring_init_rx(struct mcp251xfd_priv *priv)
|
static void
|
||||||
|
mcp251xfd_ring_init_rx(struct mcp251xfd_priv *priv, u16 *base, u8 *fifo_nr)
|
||||||
{
|
{
|
||||||
struct mcp251xfd_rx_ring *rx_ring, *prev_rx_ring = NULL;
|
struct mcp251xfd_rx_ring *rx_ring;
|
||||||
struct mcp251xfd_tx_ring *tx_ring;
|
|
||||||
struct spi_transfer *xfer;
|
struct spi_transfer *xfer;
|
||||||
u32 val;
|
u32 val;
|
||||||
u16 addr;
|
u16 addr;
|
||||||
u8 len;
|
u8 len;
|
||||||
int i, j;
|
int i, j;
|
||||||
|
|
||||||
tx_ring = priv->tx;
|
|
||||||
mcp251xfd_for_each_rx_ring(priv, rx_ring, i) {
|
mcp251xfd_for_each_rx_ring(priv, rx_ring, i) {
|
||||||
rx_ring->head = 0;
|
rx_ring->head = 0;
|
||||||
rx_ring->tail = 0;
|
rx_ring->tail = 0;
|
||||||
|
rx_ring->base = *base;
|
||||||
rx_ring->nr = i;
|
rx_ring->nr = i;
|
||||||
rx_ring->fifo_nr = MCP251XFD_RX_FIFO(i);
|
rx_ring->fifo_nr = *fifo_nr;
|
||||||
|
|
||||||
if (!prev_rx_ring)
|
*base = mcp251xfd_get_rx_obj_addr(rx_ring, rx_ring->obj_num);
|
||||||
rx_ring->base =
|
*fifo_nr += 1;
|
||||||
mcp251xfd_get_tx_obj_addr(tx_ring,
|
|
||||||
tx_ring->obj_num);
|
|
||||||
else
|
|
||||||
rx_ring->base = prev_rx_ring->base +
|
|
||||||
prev_rx_ring->obj_size *
|
|
||||||
prev_rx_ring->obj_num;
|
|
||||||
|
|
||||||
prev_rx_ring = rx_ring;
|
|
||||||
|
|
||||||
/* FIFO increment RX tail pointer */
|
/* FIFO increment RX tail pointer */
|
||||||
addr = MCP251XFD_REG_FIFOCON(rx_ring->fifo_nr);
|
addr = MCP251XFD_REG_FIFOCON(rx_ring->fifo_nr);
|
||||||
@ -209,11 +209,14 @@ static void mcp251xfd_ring_init_rx(struct mcp251xfd_priv *priv)
|
|||||||
|
|
||||||
void mcp251xfd_ring_init(struct mcp251xfd_priv *priv)
|
void mcp251xfd_ring_init(struct mcp251xfd_priv *priv)
|
||||||
{
|
{
|
||||||
|
u16 base = 0;
|
||||||
|
u8 fifo_nr = 1;
|
||||||
|
|
||||||
netdev_reset_queue(priv->ndev);
|
netdev_reset_queue(priv->ndev);
|
||||||
|
|
||||||
mcp251xfd_ring_init_tef(priv);
|
mcp251xfd_ring_init_tef(priv, &base);
|
||||||
mcp251xfd_ring_init_tx(priv);
|
mcp251xfd_ring_init_tx(priv, &base, &fifo_nr);
|
||||||
mcp251xfd_ring_init_rx(priv);
|
mcp251xfd_ring_init_rx(priv, &base, &fifo_nr);
|
||||||
}
|
}
|
||||||
|
|
||||||
void mcp251xfd_ring_free(struct mcp251xfd_priv *priv)
|
void mcp251xfd_ring_free(struct mcp251xfd_priv *priv)
|
||||||
|
@ -383,8 +383,6 @@
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
#define MCP251XFD_NAPI_WEIGHT 32
|
#define MCP251XFD_NAPI_WEIGHT 32
|
||||||
#define MCP251XFD_TX_FIFO 1
|
|
||||||
#define MCP251XFD_RX_FIFO(x) (MCP251XFD_TX_FIFO + 1 + (x))
|
|
||||||
|
|
||||||
/* SPI commands */
|
/* SPI commands */
|
||||||
#define MCP251XFD_SPI_INSTRUCTION_RESET 0x0000
|
#define MCP251XFD_SPI_INSTRUCTION_RESET 0x0000
|
||||||
|
Loading…
x
Reference in New Issue
Block a user