mirror of
https://github.com/FEX-Emu/linux.git
synced 2025-01-10 19:43:29 +00:00
can: ti_hecc: add fifo overflow error reporting
When the rx FIFO overflows the ti_hecc would silently drop them since the overwrite protection is enabled for all mailboxes. So disable it for the lowest priority mailbox and return a proper error value when receive message lost is set. Drop the message itself in that case, since it might be partially updated. Signed-off-by: Jeroen Hofstee <jhofstee@victronenergy.com> Acked-by: Jeroen Hofstee <jhofstee@victronenergy.com> Signed-off-by: Marc Kleine-Budde <mkl@pengutronix.de>
This commit is contained in:
parent
99383749c2
commit
678d85ed85
@ -73,6 +73,7 @@ MODULE_VERSION(HECC_MODULE_VERSION);
|
||||
*/
|
||||
#define HECC_MAX_RX_MBOX (HECC_MAX_MAILBOXES - HECC_MAX_TX_MBOX)
|
||||
#define HECC_RX_FIRST_MBOX (HECC_MAX_MAILBOXES - 1)
|
||||
#define HECC_RX_LAST_MBOX (HECC_MAX_TX_MBOX)
|
||||
|
||||
/* TI HECC module registers */
|
||||
#define HECC_CANME 0x0 /* Mailbox enable */
|
||||
@ -82,7 +83,7 @@ MODULE_VERSION(HECC_MODULE_VERSION);
|
||||
#define HECC_CANTA 0x10 /* Transmission acknowledge */
|
||||
#define HECC_CANAA 0x14 /* Abort acknowledge */
|
||||
#define HECC_CANRMP 0x18 /* Receive message pending */
|
||||
#define HECC_CANRML 0x1C /* Remote message lost */
|
||||
#define HECC_CANRML 0x1C /* Receive message lost */
|
||||
#define HECC_CANRFP 0x20 /* Remote frame pending */
|
||||
#define HECC_CANGAM 0x24 /* SECC only:Global acceptance mask */
|
||||
#define HECC_CANMC 0x28 /* Master control */
|
||||
@ -385,8 +386,15 @@ static void ti_hecc_start(struct net_device *ndev)
|
||||
/* Enable tx interrupts */
|
||||
hecc_set_bit(priv, HECC_CANMIM, BIT(HECC_MAX_TX_MBOX) - 1);
|
||||
|
||||
/* Prevent message over-write & Enable interrupts */
|
||||
hecc_write(priv, HECC_CANOPC, HECC_SET_REG);
|
||||
/* Prevent message over-write to create a rx fifo, but not for
|
||||
* the lowest priority mailbox, since that allows detecting
|
||||
* overflows instead of the hardware silently dropping the
|
||||
* messages.
|
||||
*/
|
||||
mbx_mask = ~BIT(HECC_RX_LAST_MBOX);
|
||||
hecc_write(priv, HECC_CANOPC, mbx_mask);
|
||||
|
||||
/* Enable interrupts */
|
||||
if (priv->use_hecc1int) {
|
||||
hecc_write(priv, HECC_CANMIL, HECC_SET_REG);
|
||||
hecc_write(priv, HECC_CANGIM, HECC_CANGIM_DEF_MASK |
|
||||
@ -531,6 +539,7 @@ static unsigned int ti_hecc_mailbox_read(struct can_rx_offload *offload,
|
||||
{
|
||||
struct ti_hecc_priv *priv = rx_offload_to_priv(offload);
|
||||
u32 data, mbx_mask;
|
||||
int ret = 1;
|
||||
|
||||
mbx_mask = BIT(mbxno);
|
||||
data = hecc_read_mbx(priv, mbxno, HECC_CANMID);
|
||||
@ -552,9 +561,26 @@ static unsigned int ti_hecc_mailbox_read(struct can_rx_offload *offload,
|
||||
}
|
||||
|
||||
*timestamp = hecc_read_stamp(priv, mbxno);
|
||||
|
||||
/* Check for FIFO overrun.
|
||||
*
|
||||
* All but the last RX mailbox have activated overwrite
|
||||
* protection. So skip check for overrun, if we're not
|
||||
* handling the last RX mailbox.
|
||||
*
|
||||
* As the overwrite protection for the last RX mailbox is
|
||||
* disabled, the CAN core might update while we're reading
|
||||
* it. This means the skb might be inconsistent.
|
||||
*
|
||||
* Return an error to let rx-offload discard this CAN frame.
|
||||
*/
|
||||
if (unlikely(mbxno == HECC_RX_LAST_MBOX &&
|
||||
hecc_read(priv, HECC_CANRML) & mbx_mask))
|
||||
ret = -ENOBUFS;
|
||||
|
||||
hecc_write(priv, HECC_CANRMP, mbx_mask);
|
||||
|
||||
return 1;
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int ti_hecc_error(struct net_device *ndev, int int_status,
|
||||
@ -884,7 +910,7 @@ static int ti_hecc_probe(struct platform_device *pdev)
|
||||
|
||||
priv->offload.mailbox_read = ti_hecc_mailbox_read;
|
||||
priv->offload.mb_first = HECC_RX_FIRST_MBOX;
|
||||
priv->offload.mb_last = HECC_MAX_TX_MBOX;
|
||||
priv->offload.mb_last = HECC_RX_LAST_MBOX;
|
||||
err = can_rx_offload_add_timestamp(ndev, &priv->offload);
|
||||
if (err) {
|
||||
dev_err(&pdev->dev, "can_rx_offload_add_timestamp() failed\n");
|
||||
|
Loading…
x
Reference in New Issue
Block a user