From e74fd1eda53a3d707df99496f16ee769fb9f2080 Mon Sep 17 00:00:00 2001 From: yuanxia Date: Tue, 21 May 2024 16:43:56 +0800 Subject: [PATCH] arm64: can: phytium: Optimize the low performance of can communication This patch is designed to optimize the issue of low performance in CAN communication at 1Mbps,as there is a waste of time in the interaction between software and hardware. Signed-off-by: yuanxia Signed-off-by: Wu Jinyong Signed-off-by: Wang Yinfeng --- drivers/net/can/phytium/phytium_can.c | 192 ++++++++++++------ drivers/net/can/phytium/phytium_can.h | 18 +- drivers/net/can/phytium/phytium_can_pci.c | 14 +- .../net/can/phytium/phytium_can_platform.c | 5 - 4 files changed, 160 insertions(+), 69 deletions(-) diff --git a/drivers/net/can/phytium/phytium_can.c b/drivers/net/can/phytium/phytium_can.c index 21bf74a1d66f2..0be8dd3876edf 100644 --- a/drivers/net/can/phytium/phytium_can.c +++ b/drivers/net/can/phytium/phytium_can.c @@ -23,7 +23,7 @@ enum phytium_can_reg { CAN_ACC_ID2_MASK = 0x28, /* Acceptance identifier2 mask register */ CAN_ACC_ID3_MASK = 0x2c, /* Acceptance identifier3 mask register */ CAN_XFER_STS = 0x30, /* Transfer status register */ - CAN_ERROR_CNT = 0x34, /* Error counter register */ + CAN_PHYTIUM_ERR_CNT = 0x34, /* Error counter register */ CAN_FIFO_CNT = 0x38, /* FIFO counter register */ CAN_DMA_CTRL = 0x3c, /* DMA request control register */ CAN_XFER_EN = 0x40, /* Transfer enable register */ @@ -85,8 +85,10 @@ enum phytium_can_reg { #define INTR_STATUS_MASK (INTR_BOIS | INTR_PWIS | INTR_PEIS | INTR_RFIS | \ INTR_TFIS | INTR_REIS | INTR_TEIS | INTR_EIS) -#define INTR_EN_MASK (INTR_BOIE | INTR_PWIE | INTR_PEIE | INTR_RFIE | \ - INTR_REIE | INTR_TEIE | INTR_EIE) + +#define INTR_EN_MASK (INTR_BOIE | INTR_PWIE | INTR_PEIE | INTR_RFIE | \ + INTR_TFIE | INTR_REIE | INTR_TEIE | INTR_EIE) + #define INTR_CLEAR_MASK (INTR_BOIC | INTR_PWIC | INTR_PEIC | INTR_RFIC | \ INTR_TFIC | INTR_REIC | INTR_TEIC | INTR_EIC) @@ -281,8 +283,8 @@ static inline u32 phytium_can_read(const struct phytium_can_dev *cdev, enum phyt return readl(cdev->base + reg); } -static inline void phytium_can_write(const struct phytium_can_dev *cdev, enum phytium_can_reg reg, - u32 val) +static inline void +phytium_can_write(const struct phytium_can_dev *cdev, enum phytium_can_reg reg, u32 val) { writel(val, cdev->base + reg); } @@ -302,8 +304,8 @@ static int phytium_can_get_berr_counter(const struct net_device *dev, { struct phytium_can_dev *cdev = netdev_priv(dev); - bec->rxerr = phytium_can_read(cdev, CAN_ERROR_CNT) & ERR_CNT_REC; - bec->txerr = (phytium_can_read(cdev, CAN_ERROR_CNT) & ERR_CNT_TEC) >> 16; + bec->rxerr = phytium_can_read(cdev, CAN_PHYTIUM_ERR_CNT) & ERR_CNT_REC; + bec->txerr = (phytium_can_read(cdev, CAN_PHYTIUM_ERR_CNT) & ERR_CNT_TEC) >> 16; return 0; } @@ -451,6 +453,7 @@ static int phytium_can_poll(struct napi_struct *napi, int quota) struct net_device *dev = napi->dev; struct phytium_can_dev *cdev = netdev_priv(dev); int work_done; + unsigned long flags; netdev_dbg(dev, "The receive processing is going on !\n"); @@ -461,7 +464,10 @@ static int phytium_can_poll(struct napi_struct *napi, int quota) */ if (work_done >= 0 && work_done < quota) { napi_complete_done(napi, work_done); - phytium_can_enable_all_interrupts(cdev); + + spin_lock_irqsave(&cdev->lock, flags); + phytium_can_set_reg_bits(cdev, CAN_INTR, INTR_REIE); + spin_unlock_irqrestore(&cdev->lock, flags); } return work_done; @@ -471,9 +477,10 @@ static void phytium_can_write_frame(struct phytium_can_dev *cdev) { struct canfd_frame *cf = (struct canfd_frame *)cdev->tx_skb->data; struct net_device *dev = cdev->net; + struct net_device_stats *stats = &dev->stats; struct sk_buff *skb = cdev->tx_skb; u32 i, id, dlc = 0, frame_head[2] = {0, 0}; - u32 data_len; + u32 data_len, tmp_len; data_len = can_fd_len2dlc(cf->len); cdev->tx_skb = NULL; @@ -558,37 +565,43 @@ static void phytium_can_write_frame(struct phytium_can_dev *cdev) } } - can_put_echo_skb(skb, dev, cdev->tx_head % cdev->tx_max, 0); - cdev->tx_head++; + stats->tx_bytes += cf->len; + stats->tx_packets++; - netif_stop_queue(dev); - /* trigger transmission */ - phytium_can_clr_reg_bits(cdev, CAN_CTRL, CTRL_XFER); - phytium_can_set_reg_bits(cdev, CAN_CTRL, CTRL_TXREQ | CTRL_XFER); + cdev->is_tx_done = false; + cdev->is_need_stop_xmit = true; + mod_timer(&cdev->timer, jiffies + HZ / 10); netdev_dbg(dev, "Trigger send message!\n"); + can_put_echo_skb(skb, dev, 0, 0); + tmp_len = can_get_echo_skb(dev, 0, 0); } static netdev_tx_t phytium_can_tx_handler(struct phytium_can_dev *cdev) { struct net_device *dev = cdev->net; u32 tx_fifo_used; + unsigned long flags; + phytium_can_write_frame(cdev); /* Check if the TX buffer is full */ - tx_fifo_used = (phytium_can_read(cdev, CAN_FIFO_CNT) & FIFO_CNT_TFN) >> 16; - if (tx_fifo_used == cdev->tx_max) { - netif_stop_queue(dev); - netdev_err(dev, "BUG!, TX FIFO full when queue awake!\n"); - return NETDEV_TX_BUSY; - } - - if (cdev->tx_head == cdev->tx_tail) { - cdev->tx_head = 0; - cdev->tx_tail = 0; + tx_fifo_used = 4 * ((phytium_can_read(cdev, CAN_FIFO_CNT) & FIFO_CNT_TFN) >> 16); + if (cdev->can.ctrlmode & CAN_CTRLMODE_FD) { + if (CAN_FIFO_BYTE_LEN - tx_fifo_used <= KEEP_CANFD_FIFO_MIN_LEN) { + netif_stop_queue(dev); + spin_lock_irqsave(&cdev->lock, flags); + cdev->is_stop_queue_flag = STOP_QUEUE_TRUE; + spin_unlock_irqrestore(&cdev->lock, flags); + } + } else { + if (CAN_FIFO_BYTE_LEN - tx_fifo_used <= KEEP_CAN_FIFO_MIN_LEN) { + netif_stop_queue(dev); + spin_lock_irqsave(&cdev->lock, flags); + cdev->is_stop_queue_flag = STOP_QUEUE_TRUE; + spin_unlock_irqrestore(&cdev->lock, flags); + } } - phytium_can_write_frame(cdev); - return NETDEV_TX_OK; } @@ -601,21 +614,59 @@ static void phytium_can_tx_interrupt(struct net_device *ndev, u32 isr) { struct phytium_can_dev *cdev = netdev_priv(ndev); struct net_device_stats *stats = &ndev->stats; + u32 tx_fifo_used = 0; + + if (isr & INTR_TEIS) + phytium_can_set_reg_bits(cdev, CAN_INTR, INTR_TEIC); - while ((cdev->tx_head - cdev->tx_tail > 0) && (isr & INTR_TEIS)) { - phytium_can_set_reg_bits(cdev, CAN_INTR, INTR_TEIC | INTR_REIC); - stats->tx_bytes = can_get_echo_skb(ndev, cdev->tx_tail % cdev->tx_max, NULL); - cdev->tx_tail++; - stats->tx_packets++; - isr = (phytium_can_read(cdev, CAN_INTR) & INTR_STATUS_MASK); + /* Check if the TX buffer is full */ + if (cdev->is_stop_queue_flag) { + tx_fifo_used = 4 * ((phytium_can_read(cdev, CAN_FIFO_CNT) & FIFO_CNT_TFN) >> 16); + if (cdev->can.ctrlmode & CAN_CTRLMODE_FD) { + if (CAN_FIFO_BYTE_LEN - tx_fifo_used > KEEP_CANFD_FIFO_MIN_LEN) { + netif_wake_queue(ndev); + cdev->is_stop_queue_flag = STOP_QUEUE_FALSE; + } + } else { + if (CAN_FIFO_BYTE_LEN - tx_fifo_used > KEEP_CAN_FIFO_MIN_LEN) { + netif_wake_queue(ndev); + cdev->is_stop_queue_flag = STOP_QUEUE_FALSE; + } + } } - phytium_can_clr_reg_bits(cdev, CAN_CTRL, CTRL_XFER); - phytium_can_clr_reg_bits(cdev, CAN_CTRL, CTRL_TXREQ); - phytium_can_set_reg_bits(cdev, CAN_CTRL, CTRL_XFER); + cdev->is_tx_done = true; + cdev->is_need_stop_xmit = false; + del_timer(&cdev->timer); + netdev_dbg(ndev, "Finish transform packets %lu\n", stats->tx_packets); - netdev_dbg(ndev, "\n-------------------\n"); - netif_wake_queue(ndev); + + phytium_can_set_reg_bits(cdev, CAN_INTR, (INTR_BOIE | + INTR_PWIE | INTR_PEIE)); +} + +static void phytium_can_tx_done_timeout(struct timer_list *t) +{ + struct phytium_can_dev *priv = from_timer(priv, t, timer); + struct net_device *ndev = priv->net; + + if (!priv->is_tx_done) { + if (priv->is_need_stop_xmit) { + netdev_dbg(ndev, "%s stop xmit\n", __func__); + priv->is_need_stop_xmit = false; + phytium_can_clr_reg_bits(priv, CAN_CTRL, CTRL_XFER); + phytium_can_clr_reg_bits(priv, CAN_INTR, (INTR_BOIE | + INTR_PWIE | INTR_PEIE)); + /* stop xmit and restart after 500ms */ + mod_timer(&priv->timer, jiffies + HZ / 2); + } else { + netdev_dbg(ndev, "%s start xmit\n", __func__); + priv->is_need_stop_xmit = true; + phytium_can_set_reg_bits(priv, CAN_CTRL, CTRL_XFER); + /* start xmit and stop after 250ms */ + mod_timer(&priv->timer, jiffies + HZ / 4); + } + } } static void phytium_can_err_interrupt(struct net_device *ndev, u32 isr) @@ -628,11 +679,11 @@ static void phytium_can_err_interrupt(struct net_device *ndev, u32 isr) skb = alloc_can_err_skb(ndev, &cf); - rxerr = phytium_can_read(cdev, CAN_ERROR_CNT) & ERR_CNT_REC; - txerr = ((phytium_can_read(cdev, CAN_ERROR_CNT) & ERR_CNT_TEC) >> 16); + rxerr = phytium_can_read(cdev, CAN_PHYTIUM_ERR_CNT) & ERR_CNT_REC; + txerr = ((phytium_can_read(cdev, CAN_PHYTIUM_ERR_CNT) & ERR_CNT_TEC) >> 16); if (isr & INTR_BOIS) { - netdev_dbg(ndev, "%s: txerr :%u rxerr :%u\n", + netdev_dbg(ndev, "bus_off %s: txerr :%u rxerr :%u\n", __func__, txerr, rxerr); cdev->can.state = CAN_STATE_BUS_OFF; cdev->can.can_stats.bus_off++; @@ -642,10 +693,15 @@ static void phytium_can_err_interrupt(struct net_device *ndev, u32 isr) if (skb) cf->can_id |= CAN_ERR_BUSOFF; } else if ((isr & INTR_PEIS) == INTR_PEIS) { - netdev_dbg(ndev, "%s: txerr :%u rxerr :%u\n", + netdev_dbg(ndev, "error_passive %s: txerr :%u rxerr :%u\n", __func__, txerr, rxerr); cdev->can.state = CAN_STATE_ERROR_PASSIVE; cdev->can.can_stats.error_passive++; + /* Clear interrupt condition */ + phytium_can_set_reg_bits(cdev, CAN_INTR, INTR_PEIC); + phytium_can_set_reg_bits(cdev, CAN_INTR, INTR_PWIC); + phytium_can_set_reg_bits(cdev, CAN_INTR, INTR_TEIC); + phytium_can_set_reg_bits(cdev, CAN_INTR, INTR_EIC); if (skb) { cf->can_id |= CAN_ERR_CRTL; cf->data[1] = (rxerr > 127) ? @@ -655,10 +711,13 @@ static void phytium_can_err_interrupt(struct net_device *ndev, u32 isr) cf->data[7] = rxerr; } } else if (isr & INTR_PWIS) { - netdev_dbg(ndev, "%s: txerr :%u rxerr :%u\n", + netdev_dbg(ndev, "error_warning %s: txerr :%u rxerr :%u\n", __func__, txerr, rxerr); cdev->can.state = CAN_STATE_ERROR_WARNING; cdev->can.can_stats.error_warning++; + phytium_can_set_reg_bits(cdev, CAN_INTR, INTR_PWIC); + phytium_can_set_reg_bits(cdev, CAN_INTR, INTR_TEIC); + phytium_can_set_reg_bits(cdev, CAN_INTR, INTR_EIC); if (skb) { cf->can_id |= CAN_ERR_CRTL; cf->data[1] |= (txerr > rxerr) ? @@ -709,6 +768,7 @@ static irqreturn_t phytium_can_isr(int irq, void *dev_id) if (!isr) return IRQ_NONE; + spin_lock(&cdev->lock); /* Check for FIFO full interrupt and alarm */ if ((isr & INTR_RFIS)) { netdev_dbg(dev, "rx_fifo is full!.\n"); @@ -718,14 +778,22 @@ static irqreturn_t phytium_can_isr(int irq, void *dev_id) napi_schedule(&cdev->napi); } + /* Check for FIFO empty interrupt and alarm */ + if ((isr & INTR_TFIS)) { + netdev_dbg(dev, "tx_fifo is empty!.\n"); + isr &= (~INTR_TFIS); + phytium_can_clr_reg_bits(cdev, CAN_INTR, INTR_TFIE); + phytium_can_set_reg_bits(cdev, CAN_INTR, INTR_TFIC); + } + /* Check for the type of error interrupt and Processing it */ - if (isr & (INTR_EIS | INTR_RFIS | INTR_BOIS | INTR_PEIS)) { - phytium_can_clr_reg_bits(cdev, CAN_INTR, (INTR_EIE - | INTR_RFIE | INTR_BOIE | INTR_PEIE)); + if (isr & (INTR_EIS | INTR_RFIS | INTR_BOIS | INTR_PWIS | INTR_PEIS)) { + phytium_can_clr_reg_bits(cdev, CAN_INTR, (INTR_EIE | INTR_RFIE | + INTR_BOIE | INTR_PWIE | INTR_PEIE)); phytium_can_err_interrupt(dev, isr); - phytium_can_set_reg_bits(cdev, CAN_INTR, (INTR_EIC - | INTR_RFIC | INTR_BOIC | INTR_PEIC)); - phytium_can_set_reg_bits(cdev, CAN_INTR, INTR_EN_MASK); + phytium_can_set_reg_bits(cdev, CAN_INTR, (INTR_EIC | INTR_RFIC | + INTR_BOIC | INTR_PWIC | INTR_PEIC)); + spin_unlock(&cdev->lock); return IRQ_HANDLED; } @@ -742,7 +810,7 @@ static irqreturn_t phytium_can_isr(int irq, void *dev_id) phytium_can_set_reg_bits(cdev, CAN_INTR, INTR_REIC); napi_schedule(&cdev->napi); } - + spin_unlock(&cdev->lock); return IRQ_HANDLED; } @@ -876,6 +944,8 @@ static void phytium_can_stop(struct net_device *dev) ctrl &= ~(CTRL_XFER | CTRL_TXREQ); phytium_can_write(cdev, CAN_CTRL, ctrl); + del_timer(&cdev->timer); + /* Set the state as STOPPED */ cdev->can.state = CAN_STATE_STOPPED; } @@ -917,10 +987,12 @@ static int phytium_can_open(struct net_device *dev) struct phytium_can_dev *cdev = netdev_priv(dev); int ret; - /* Start clock */ - ret = pm_runtime_resume(cdev->dev); - if (ret) + ret = pm_runtime_get_sync(cdev->dev); + if (ret < 0) { + netdev_err(dev, "%s: pm_runtime_get failed(%d)\n", + __func__, ret); return ret; + } /* Open the CAN device */ ret = open_candev(dev); @@ -940,12 +1012,16 @@ static int phytium_can_open(struct net_device *dev) /* Start the controller */ phytium_can_start(dev); + netdev_dbg(dev, "%s is going on\n", __func__); + napi_enable(&cdev->napi); + cdev->is_stop_queue_flag = STOP_QUEUE_FALSE; netif_start_queue(dev); return 0; fail: + pm_runtime_put(cdev->dev); close_candev(dev); disable_clk: pm_runtime_put_sync(cdev->dev); @@ -1025,7 +1101,7 @@ static int phytium_can_dev_setup(struct phytium_can_dev *cdev) cdev->can.ctrlmode = CAN_CTRLMODE_FD; cdev->can.data_bittiming_const = cdev->bit_timing; } - + spin_lock_init(&cdev->lock); return 0; } @@ -1062,10 +1138,6 @@ int phytium_can_register(struct phytium_can_dev *cdev) { int ret; - ret = pm_runtime_resume(cdev->dev); - if (ret) - return ret; - ret = phytium_can_dev_setup(cdev); if (ret) goto fail; @@ -1077,6 +1149,10 @@ int phytium_can_register(struct phytium_can_dev *cdev) goto fail; } + cdev->is_tx_done = true; + cdev->is_need_stop_xmit = false; + timer_setup(&cdev->timer, phytium_can_tx_done_timeout, 0); + dev_info(cdev->dev, "%s device registered (irq=%d)\n", KBUILD_MODNAME, cdev->net->irq); diff --git a/drivers/net/can/phytium/phytium_can.h b/drivers/net/can/phytium/phytium_can.h index edb74df010ca3..802bb36a13116 100644 --- a/drivers/net/can/phytium/phytium_can.h +++ b/drivers/net/can/phytium/phytium_can.h @@ -23,6 +23,12 @@ #include #include +#define KEEP_CAN_FIFO_MIN_LEN 16 +#define KEEP_CANFD_FIFO_MIN_LEN 128 +#define CAN_FIFO_BYTE_LEN 256 +#define STOP_QUEUE_TRUE 1 +#define STOP_QUEUE_FALSE 0 + enum phytium_can_ip_type { PHYTIUM_CAN = 0, PHYTIUM_CANFD, @@ -35,9 +41,7 @@ struct phytium_can_devtype { struct phytium_can_dev { struct can_priv can; - unsigned int tx_head; - unsigned int tx_tail; - unsigned int tx_max; + struct napi_struct napi; struct net_device *net; struct device *dev; @@ -46,12 +50,16 @@ struct phytium_can_dev { struct sk_buff *tx_skb; const struct can_bittiming_const *bit_timing; - + spinlock_t lock; /*spinlock*/ int fdmode; u32 isr; u32 tx_fifo_depth; - + unsigned int is_stop_queue_flag; void __iomem *base; + + struct timer_list timer; /* xmit done timer */ + u32 is_tx_done; + u32 is_need_stop_xmit; }; struct phytium_can_dev *phytium_can_allocate_dev(struct device *dev, int sizeof_priv, diff --git a/drivers/net/can/phytium/phytium_can_pci.c b/drivers/net/can/phytium/phytium_can_pci.c index ac84b3b15c448..af7f606f432a0 100644 --- a/drivers/net/can/phytium/phytium_can_pci.c +++ b/drivers/net/can/phytium/phytium_can_pci.c @@ -81,12 +81,22 @@ static int phytium_can_pci_probe(struct pci_dev *pdev, const struct pci_device_i pci_set_drvdata(pdev, cdev->net); - pm_runtime_enable(cdev->dev); + if (!pm_runtime_enabled(cdev->dev)) + pm_runtime_enable(cdev->dev); + ret = pm_runtime_get_sync(cdev->dev); + if (ret < 0) { + netdev_err(cdev->net, "%s: pm_runtime_get failed(%d)\n", + __func__, ret); + goto err_pmdisable; + } ret = phytium_can_register(cdev); if (ret) goto err; return 0; + +err_pmdisable: + pm_runtime_disable(&pdev->dev); err: return ret; } @@ -96,6 +106,8 @@ static void phytium_can_pci_remove(struct pci_dev *pdev) struct net_device *dev = pci_get_drvdata(pdev); struct phytium_can_dev *cdev = netdev_priv(dev); + pm_runtime_disable(cdev->dev); + phytium_can_unregister(cdev); phytium_can_free_dev(cdev->net); } diff --git a/drivers/net/can/phytium/phytium_can_platform.c b/drivers/net/can/phytium/phytium_can_platform.c index 63f89d721784f..852da788f9f75 100644 --- a/drivers/net/can/phytium/phytium_can_platform.c +++ b/drivers/net/can/phytium/phytium_can_platform.c @@ -126,14 +126,9 @@ static int phytium_can_plat_probe(struct platform_device *pdev) "mode-select", &str); if (!(strcmp(str, "canfd"))) devtype = &phytium_canfd_data; - else - devtype = &phytium_can_data; } cdev->tx_fifo_depth = tx_fifo_depth; - cdev->tx_head = 0; - cdev->tx_tail = 0; - cdev->tx_max = tx_fifo_depth; if (devtype->cantype == PHYTIUM_CANFD) cdev->fdmode = 1;