ravb: Support separate Line0 (Desc), Line1 (Err) and Line2 (Mgmt) irqs
authorPhil Edworthy <phil.edworthy@renesas.com>
Thu, 12 May 2022 11:47:20 +0000 (12:47 +0100)
committerDavid S. Miller <davem@davemloft.net>
Mon, 16 May 2022 09:14:27 +0000 (10:14 +0100)
R-Car has a combined interrupt line, ch22 = Line0_DiA | Line1_A | Line2_A.
RZ/V2M has separate interrupt lines for each of these, so add a feature
that allows the driver to get these interrupts and call the common handler.

Signed-off-by: Phil Edworthy <phil.edworthy@renesas.com>
Reviewed-by: Biju Das <biju.das.jz@bp.renesas.com>
Reviewed-by: Sergey Shtylyov <s.shtylyov@omp.ru>
Signed-off-by: David S. Miller <davem@davemloft.net>
drivers/net/ethernet/renesas/ravb.h
drivers/net/ethernet/renesas/ravb_main.c

index bb82efd222c74cf0ede09a77356e99633436f8f3..e505e8088445841c2709b76a99f1ad6ed03eab37 100644 (file)
@@ -1028,6 +1028,7 @@ struct ravb_hw_info {
        unsigned carrier_counters:1;    /* E-MAC has carrier counters */
        unsigned multi_irqs:1;          /* AVB-DMAC and E-MAC has multiple irqs */
        unsigned irq_en_dis:1;          /* Has separate irq enable and disable regs */
+       unsigned err_mgmt_irqs:1;       /* Line1 (Err) and Line2 (Mgmt) irqs are separate */
        unsigned gptp:1;                /* AVB-DMAC has gPTP support */
        unsigned ccc_gac:1;             /* AVB-DMAC has gPTP support active in config mode */
        unsigned nc_queues:1;           /* AVB-DMAC has RX and TX NC queues */
@@ -1078,6 +1079,8 @@ struct ravb_private {
        int msg_enable;
        int speed;
        int emac_irq;
+       int erra_irq;
+       int mgmta_irq;
        int rx_irqs[NUM_RX_QUEUE];
        int tx_irqs[NUM_TX_QUEUE];
 
index e22c0e6ed0f3959db0a7e5570067b5fbb807328c..8ccc817b8b5d83b7d54a4a26fd5c741a67005f00 100644 (file)
@@ -1798,12 +1798,23 @@ static int ravb_open(struct net_device *ndev)
                                      ndev, dev, "ch19:tx_nc");
                if (error)
                        goto out_free_irq_nc_rx;
+
+               if (info->err_mgmt_irqs) {
+                       error = ravb_hook_irq(priv->erra_irq, ravb_multi_interrupt,
+                                             ndev, dev, "err_a");
+                       if (error)
+                               goto out_free_irq_nc_tx;
+                       error = ravb_hook_irq(priv->mgmta_irq, ravb_multi_interrupt,
+                                             ndev, dev, "mgmt_a");
+                       if (error)
+                               goto out_free_irq_erra;
+               }
        }
 
        /* Device init */
        error = ravb_dmac_init(ndev);
        if (error)
-               goto out_free_irq_nc_tx;
+               goto out_free_irq_mgmta;
        ravb_emac_init(ndev);
 
        /* Initialise PTP Clock driver */
@@ -1823,9 +1834,15 @@ out_ptp_stop:
        /* Stop PTP Clock driver */
        if (info->gptp)
                ravb_ptp_stop(ndev);
-out_free_irq_nc_tx:
+out_free_irq_mgmta:
        if (!info->multi_irqs)
                goto out_free_irq;
+       if (info->err_mgmt_irqs)
+               free_irq(priv->mgmta_irq, ndev);
+out_free_irq_erra:
+       if (info->err_mgmt_irqs)
+               free_irq(priv->erra_irq, ndev);
+out_free_irq_nc_tx:
        free_irq(priv->tx_irqs[RAVB_NC], ndev);
 out_free_irq_nc_rx:
        free_irq(priv->rx_irqs[RAVB_NC], ndev);
@@ -2166,6 +2183,10 @@ static int ravb_close(struct net_device *ndev)
                free_irq(priv->tx_irqs[RAVB_BE], ndev);
                free_irq(priv->rx_irqs[RAVB_BE], ndev);
                free_irq(priv->emac_irq, ndev);
+               if (info->err_mgmt_irqs) {
+                       free_irq(priv->erra_irq, ndev);
+                       free_irq(priv->mgmta_irq, ndev);
+               }
        }
        free_irq(ndev->irq, ndev);
 
@@ -2595,10 +2616,14 @@ static int ravb_probe(struct platform_device *pdev)
        pm_runtime_enable(&pdev->dev);
        pm_runtime_get_sync(&pdev->dev);
 
-       if (info->multi_irqs)
-               irq = platform_get_irq_byname(pdev, "ch22");
-       else
+       if (info->multi_irqs) {
+               if (info->err_mgmt_irqs)
+                       irq = platform_get_irq_byname(pdev, "dia");
+               else
+                       irq = platform_get_irq_byname(pdev, "ch22");
+       } else {
                irq = platform_get_irq(pdev, 0);
+       }
        if (irq < 0) {
                error = irq;
                goto out_release;
@@ -2640,7 +2665,10 @@ static int ravb_probe(struct platform_device *pdev)
                of_property_read_bool(np, "renesas,ether-link-active-low");
 
        if (info->multi_irqs) {
-               irq = platform_get_irq_byname(pdev, "ch24");
+               if (info->err_mgmt_irqs)
+                       irq = platform_get_irq_byname(pdev, "line3");
+               else
+                       irq = platform_get_irq_byname(pdev, "ch24");
                if (irq < 0) {
                        error = irq;
                        goto out_release;
@@ -2662,6 +2690,22 @@ static int ravb_probe(struct platform_device *pdev)
                        }
                        priv->tx_irqs[i] = irq;
                }
+
+               if (info->err_mgmt_irqs) {
+                       irq = platform_get_irq_byname(pdev, "err_a");
+                       if (irq < 0) {
+                               error = irq;
+                               goto out_release;
+                       }
+                       priv->erra_irq = irq;
+
+                       irq = platform_get_irq_byname(pdev, "mgmt_a");
+                       if (irq < 0) {
+                               error = irq;
+                               goto out_release;
+                       }
+                       priv->mgmta_irq = irq;
+               }
        }
 
        priv->clk = devm_clk_get(&pdev->dev, NULL);