mirror of
https://mirrors.bfsu.edu.cn/git/linux.git
synced 2025-01-21 13:24:15 +08:00
can: mcp251xfd: prepare for multiple RX-FIFOs
This patch prepares the driver to use more than one RX-FIFO. Having a bigger RX buffer is beneficial in high load situations, where the system temporarily cannot keep up reading CAN frames from the chip. Using a bigger RX buffer also allows to implement RX IRQ coalescing, which will be added in a later patch series. If using more than 1 RX-FIFO the driver has to figure out, which FIFOs have RX'ed CAN frames pending. This is indicated by a set bit in the RXIF register, which is positioned directly after the interrupt status register INT. If more than 1 RX-FIFO is used, the driver reads both registers in 1 transfer. The mcp251xfd_handle_rxif() function iterates over all RX rings and reads out the RX'ed CAN frames for for all pending FIFOs. To keep the logic for the 1 RX-FIFO only case in mcp251xfd_handle_rxif() simple, the driver marks that FIFO pending in mcp251xfd_ring_init(). The chip has a dedicated RX interrupt line to signal pending RX'ed frames. If connected to an input GPIO and the driver will skip the initial read of the interrupt status register (INT) and directly read the pending RX'ed frames if the line is active. The driver assumes the 1st RX-FIFO pending (a read of the RXIF register would re-introduce the skipped initial read of the INT register). Any other pending RX-FIFO will be served in the main interrupt handler. Link: https://lore.kernel.org/all/20220217103826.2299157-8-mkl@pengutronix.de Signed-off-by: Marc Kleine-Budde <mkl@pengutronix.de>
This commit is contained in:
parent
83daa863f1
commit
887e359d6c
@ -1384,6 +1384,20 @@ static int mcp251xfd_handle_spicrcif(struct mcp251xfd_priv *priv)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int mcp251xfd_read_regs_status(struct mcp251xfd_priv *priv)
|
||||||
|
{
|
||||||
|
const int val_bytes = regmap_get_val_bytes(priv->map_reg);
|
||||||
|
size_t len;
|
||||||
|
|
||||||
|
if (priv->rx_ring_num == 1)
|
||||||
|
len = sizeof(priv->regs_status.intf);
|
||||||
|
else
|
||||||
|
len = sizeof(priv->regs_status);
|
||||||
|
|
||||||
|
return regmap_bulk_read(priv->map_reg, MCP251XFD_REG_INT,
|
||||||
|
&priv->regs_status, len / val_bytes);
|
||||||
|
}
|
||||||
|
|
||||||
#define mcp251xfd_handle(priv, irq, ...) \
|
#define mcp251xfd_handle(priv, irq, ...) \
|
||||||
({ \
|
({ \
|
||||||
struct mcp251xfd_priv *_priv = (priv); \
|
struct mcp251xfd_priv *_priv = (priv); \
|
||||||
@ -1400,7 +1414,6 @@ static int mcp251xfd_handle_spicrcif(struct mcp251xfd_priv *priv)
|
|||||||
static irqreturn_t mcp251xfd_irq(int irq, void *dev_id)
|
static irqreturn_t mcp251xfd_irq(int irq, void *dev_id)
|
||||||
{
|
{
|
||||||
struct mcp251xfd_priv *priv = dev_id;
|
struct mcp251xfd_priv *priv = dev_id;
|
||||||
const int val_bytes = regmap_get_val_bytes(priv->map_reg);
|
|
||||||
irqreturn_t handled = IRQ_NONE;
|
irqreturn_t handled = IRQ_NONE;
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
@ -1412,21 +1425,28 @@ static irqreturn_t mcp251xfd_irq(int irq, void *dev_id)
|
|||||||
if (!rx_pending)
|
if (!rx_pending)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
/* Assume 1st RX-FIFO pending, if other FIFOs
|
||||||
|
* are pending the main IRQ handler will take
|
||||||
|
* care.
|
||||||
|
*/
|
||||||
|
priv->regs_status.rxif = BIT(priv->rx[0]->fifo_nr);
|
||||||
err = mcp251xfd_handle(priv, rxif);
|
err = mcp251xfd_handle(priv, rxif);
|
||||||
if (err)
|
if (err)
|
||||||
goto out_fail;
|
goto out_fail;
|
||||||
|
|
||||||
handled = IRQ_HANDLED;
|
handled = IRQ_HANDLED;
|
||||||
} while (1);
|
|
||||||
|
/* We don't know which RX-FIFO is pending, but only
|
||||||
|
* handle the 1st RX-FIFO. Leave loop here if we have
|
||||||
|
* more than 1 RX-FIFO to avoid starvation.
|
||||||
|
*/
|
||||||
|
} while (priv->rx_ring_num == 1);
|
||||||
|
|
||||||
do {
|
do {
|
||||||
u32 intf_pending, intf_pending_clearable;
|
u32 intf_pending, intf_pending_clearable;
|
||||||
bool set_normal_mode = false;
|
bool set_normal_mode = false;
|
||||||
|
|
||||||
err = regmap_bulk_read(priv->map_reg, MCP251XFD_REG_INT,
|
err = mcp251xfd_read_regs_status(priv);
|
||||||
&priv->regs_status,
|
|
||||||
sizeof(priv->regs_status) /
|
|
||||||
val_bytes);
|
|
||||||
if (err)
|
if (err)
|
||||||
goto out_fail;
|
goto out_fail;
|
||||||
|
|
||||||
|
@ -220,6 +220,24 @@ int mcp251xfd_ring_init(struct mcp251xfd_priv *priv)
|
|||||||
mcp251xfd_ring_init_rx(priv, &base, &fifo_nr);
|
mcp251xfd_ring_init_rx(priv, &base, &fifo_nr);
|
||||||
mcp251xfd_ring_init_tx(priv, &base, &fifo_nr);
|
mcp251xfd_ring_init_tx(priv, &base, &fifo_nr);
|
||||||
|
|
||||||
|
/* mcp251xfd_handle_rxif() will iterate over all RX rings.
|
||||||
|
* Rings with their corresponding bit set in
|
||||||
|
* priv->regs_status.rxif are read out.
|
||||||
|
*
|
||||||
|
* If the chip is configured for only 1 RX-FIFO, and if there
|
||||||
|
* is an RX interrupt pending (RXIF in INT register is set),
|
||||||
|
* it must be the 1st RX-FIFO.
|
||||||
|
*
|
||||||
|
* We mark the RXIF of the 1st FIFO as pending here, so that
|
||||||
|
* we can skip the read of the RXIF register in
|
||||||
|
* mcp251xfd_read_regs_status() for the 1 RX-FIFO only case.
|
||||||
|
*
|
||||||
|
* If we use more than 1 RX-FIFO, this value gets overwritten
|
||||||
|
* in mcp251xfd_read_regs_status(), so set it unconditionally
|
||||||
|
* here.
|
||||||
|
*/
|
||||||
|
priv->regs_status.rxif = BIT(priv->rx[0]->fifo_nr);
|
||||||
|
|
||||||
netdev_dbg(priv->ndev,
|
netdev_dbg(priv->ndev,
|
||||||
"FIFO setup: TEF: 0x%03x: %2d*%zu bytes = %4zu bytes\n",
|
"FIFO setup: TEF: 0x%03x: %2d*%zu bytes = %4zu bytes\n",
|
||||||
mcp251xfd_get_tef_obj_addr(0),
|
mcp251xfd_get_tef_obj_addr(0),
|
||||||
|
@ -19,7 +19,7 @@
|
|||||||
static inline int
|
static inline int
|
||||||
mcp251xfd_rx_head_get_from_chip(const struct mcp251xfd_priv *priv,
|
mcp251xfd_rx_head_get_from_chip(const struct mcp251xfd_priv *priv,
|
||||||
const struct mcp251xfd_rx_ring *ring,
|
const struct mcp251xfd_rx_ring *ring,
|
||||||
u8 *rx_head)
|
u8 *rx_head, bool *fifo_empty)
|
||||||
{
|
{
|
||||||
u32 fifo_sta;
|
u32 fifo_sta;
|
||||||
int err;
|
int err;
|
||||||
@ -30,6 +30,7 @@ mcp251xfd_rx_head_get_from_chip(const struct mcp251xfd_priv *priv,
|
|||||||
return err;
|
return err;
|
||||||
|
|
||||||
*rx_head = FIELD_GET(MCP251XFD_REG_FIFOSTA_FIFOCI_MASK, fifo_sta);
|
*rx_head = FIELD_GET(MCP251XFD_REG_FIFOSTA_FIFOCI_MASK, fifo_sta);
|
||||||
|
*fifo_empty = !(fifo_sta & MCP251XFD_REG_FIFOSTA_TFNRFNIF);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@ -84,10 +85,12 @@ mcp251xfd_rx_ring_update(const struct mcp251xfd_priv *priv,
|
|||||||
{
|
{
|
||||||
u32 new_head;
|
u32 new_head;
|
||||||
u8 chip_rx_head;
|
u8 chip_rx_head;
|
||||||
|
bool fifo_empty;
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
err = mcp251xfd_rx_head_get_from_chip(priv, ring, &chip_rx_head);
|
err = mcp251xfd_rx_head_get_from_chip(priv, ring, &chip_rx_head,
|
||||||
if (err)
|
&fifo_empty);
|
||||||
|
if (err || fifo_empty)
|
||||||
return err;
|
return err;
|
||||||
|
|
||||||
/* chip_rx_head, is the next RX-Object filled by the HW.
|
/* chip_rx_head, is the next RX-Object filled by the HW.
|
||||||
@ -251,6 +254,9 @@ int mcp251xfd_handle_rxif(struct mcp251xfd_priv *priv)
|
|||||||
int err, n;
|
int err, n;
|
||||||
|
|
||||||
mcp251xfd_for_each_rx_ring(priv, ring, n) {
|
mcp251xfd_for_each_rx_ring(priv, ring, n) {
|
||||||
|
if (!(priv->regs_status.rxif & BIT(ring->fifo_nr)))
|
||||||
|
continue;
|
||||||
|
|
||||||
err = mcp251xfd_handle_rxif_ring(priv, ring);
|
err = mcp251xfd_handle_rxif_ring(priv, ring);
|
||||||
if (err)
|
if (err)
|
||||||
return err;
|
return err;
|
||||||
|
@ -561,6 +561,7 @@ struct mcp251xfd_ecc {
|
|||||||
|
|
||||||
struct mcp251xfd_regs_status {
|
struct mcp251xfd_regs_status {
|
||||||
u32 intf;
|
u32 intf;
|
||||||
|
u32 rxif;
|
||||||
};
|
};
|
||||||
|
|
||||||
enum mcp251xfd_model {
|
enum mcp251xfd_model {
|
||||||
|
Loading…
Reference in New Issue
Block a user