Commit 6de19fa0 authored by Biju Das's avatar Biju Das Committed by David S. Miller

ravb: Add multi_irq to struct ravb_hw_info

R-Car Gen3 supports separate interrupts for E-MAC and DMA queues,
whereas R-Car Gen2 and RZ/G2L have a single interrupt instead.

Add a multi_irq hw feature bit to struct ravb_hw_info to enable
this only for R-Car Gen3.
Signed-off-by: default avatarBiju Das <biju.das.jz@bp.renesas.com>
Reviewed-by: default avatarLad Prabhakar <prabhakar.mahadev-lad.rj@bp.renesas.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent c81d8942
...@@ -997,6 +997,7 @@ struct ravb_hw_info { ...@@ -997,6 +997,7 @@ struct ravb_hw_info {
/* hardware features */ /* hardware features */
unsigned internal_delay:1; /* AVB-DMAC has internal delays */ unsigned internal_delay:1; /* AVB-DMAC has internal delays */
unsigned tx_counters:1; /* E-MAC has TX counters */ unsigned tx_counters:1; /* E-MAC has TX counters */
unsigned multi_irqs:1; /* AVB-DMAC and E-MAC has multiple irqs */
}; };
struct ravb_private { struct ravb_private {
......
...@@ -427,6 +427,7 @@ static void ravb_emac_init(struct net_device *ndev) ...@@ -427,6 +427,7 @@ static void ravb_emac_init(struct net_device *ndev)
static int ravb_dmac_init(struct net_device *ndev) static int ravb_dmac_init(struct net_device *ndev)
{ {
struct ravb_private *priv = netdev_priv(ndev); struct ravb_private *priv = netdev_priv(ndev);
const struct ravb_hw_info *info = priv->info;
int error; int error;
/* Set CONFIG mode */ /* Set CONFIG mode */
...@@ -458,7 +459,7 @@ static int ravb_dmac_init(struct net_device *ndev) ...@@ -458,7 +459,7 @@ static int ravb_dmac_init(struct net_device *ndev)
ravb_write(ndev, TCCR_TFEN, TCCR); ravb_write(ndev, TCCR_TFEN, TCCR);
/* Interrupt init: */ /* Interrupt init: */
if (priv->chip_id == RCAR_GEN3) { if (info->multi_irqs) {
/* Clear DIL.DPLx */ /* Clear DIL.DPLx */
ravb_write(ndev, 0, DIL); ravb_write(ndev, 0, DIL);
/* Set queue specific interrupt */ /* Set queue specific interrupt */
...@@ -758,6 +759,7 @@ static void ravb_error_interrupt(struct net_device *ndev) ...@@ -758,6 +759,7 @@ static void ravb_error_interrupt(struct net_device *ndev)
static bool ravb_queue_interrupt(struct net_device *ndev, int q) static bool ravb_queue_interrupt(struct net_device *ndev, int q)
{ {
struct ravb_private *priv = netdev_priv(ndev); struct ravb_private *priv = netdev_priv(ndev);
const struct ravb_hw_info *info = priv->info;
u32 ris0 = ravb_read(ndev, RIS0); u32 ris0 = ravb_read(ndev, RIS0);
u32 ric0 = ravb_read(ndev, RIC0); u32 ric0 = ravb_read(ndev, RIC0);
u32 tis = ravb_read(ndev, TIS); u32 tis = ravb_read(ndev, TIS);
...@@ -766,7 +768,7 @@ static bool ravb_queue_interrupt(struct net_device *ndev, int q) ...@@ -766,7 +768,7 @@ static bool ravb_queue_interrupt(struct net_device *ndev, int q)
if (((ris0 & ric0) & BIT(q)) || ((tis & tic) & BIT(q))) { if (((ris0 & ric0) & BIT(q)) || ((tis & tic) & BIT(q))) {
if (napi_schedule_prep(&priv->napi[q])) { if (napi_schedule_prep(&priv->napi[q])) {
/* Mask RX and TX interrupts */ /* Mask RX and TX interrupts */
if (priv->chip_id == RCAR_GEN2) { if (!info->multi_irqs) {
ravb_write(ndev, ric0 & ~BIT(q), RIC0); ravb_write(ndev, ric0 & ~BIT(q), RIC0);
ravb_write(ndev, tic & ~BIT(q), TIC); ravb_write(ndev, tic & ~BIT(q), TIC);
} else { } else {
...@@ -909,6 +911,7 @@ static int ravb_poll(struct napi_struct *napi, int budget) ...@@ -909,6 +911,7 @@ static int ravb_poll(struct napi_struct *napi, int budget)
{ {
struct net_device *ndev = napi->dev; struct net_device *ndev = napi->dev;
struct ravb_private *priv = netdev_priv(ndev); struct ravb_private *priv = netdev_priv(ndev);
const struct ravb_hw_info *info = priv->info;
unsigned long flags; unsigned long flags;
int q = napi - priv->napi; int q = napi - priv->napi;
int mask = BIT(q); int mask = BIT(q);
...@@ -932,7 +935,7 @@ static int ravb_poll(struct napi_struct *napi, int budget) ...@@ -932,7 +935,7 @@ static int ravb_poll(struct napi_struct *napi, int budget)
/* Re-enable RX/TX interrupts */ /* Re-enable RX/TX interrupts */
spin_lock_irqsave(&priv->lock, flags); spin_lock_irqsave(&priv->lock, flags);
if (priv->chip_id == RCAR_GEN2) { if (!info->multi_irqs) {
ravb_modify(ndev, RIC0, mask, mask); ravb_modify(ndev, RIC0, mask, mask);
ravb_modify(ndev, TIC, mask, mask); ravb_modify(ndev, TIC, mask, mask);
} else { } else {
...@@ -1338,6 +1341,7 @@ static inline int ravb_hook_irq(unsigned int irq, irq_handler_t handler, ...@@ -1338,6 +1341,7 @@ static inline int ravb_hook_irq(unsigned int irq, irq_handler_t handler,
static int ravb_open(struct net_device *ndev) static int ravb_open(struct net_device *ndev)
{ {
struct ravb_private *priv = netdev_priv(ndev); struct ravb_private *priv = netdev_priv(ndev);
const struct ravb_hw_info *info = priv->info;
struct platform_device *pdev = priv->pdev; struct platform_device *pdev = priv->pdev;
struct device *dev = &pdev->dev; struct device *dev = &pdev->dev;
int error; int error;
...@@ -1345,7 +1349,7 @@ static int ravb_open(struct net_device *ndev) ...@@ -1345,7 +1349,7 @@ static int ravb_open(struct net_device *ndev)
napi_enable(&priv->napi[RAVB_BE]); napi_enable(&priv->napi[RAVB_BE]);
napi_enable(&priv->napi[RAVB_NC]); napi_enable(&priv->napi[RAVB_NC]);
if (priv->chip_id == RCAR_GEN2) { if (!info->multi_irqs) {
error = request_irq(ndev->irq, ravb_interrupt, IRQF_SHARED, error = request_irq(ndev->irq, ravb_interrupt, IRQF_SHARED,
ndev->name, ndev); ndev->name, ndev);
if (error) { if (error) {
...@@ -1403,7 +1407,7 @@ static int ravb_open(struct net_device *ndev) ...@@ -1403,7 +1407,7 @@ static int ravb_open(struct net_device *ndev)
if (priv->chip_id == RCAR_GEN2) if (priv->chip_id == RCAR_GEN2)
ravb_ptp_stop(ndev); ravb_ptp_stop(ndev);
out_free_irq_nc_tx: out_free_irq_nc_tx:
if (priv->chip_id == RCAR_GEN2) if (!info->multi_irqs)
goto out_free_irq; goto out_free_irq;
free_irq(priv->tx_irqs[RAVB_NC], ndev); free_irq(priv->tx_irqs[RAVB_NC], ndev);
out_free_irq_nc_rx: out_free_irq_nc_rx:
...@@ -1680,6 +1684,7 @@ static int ravb_close(struct net_device *ndev) ...@@ -1680,6 +1684,7 @@ static int ravb_close(struct net_device *ndev)
{ {
struct device_node *np = ndev->dev.parent->of_node; struct device_node *np = ndev->dev.parent->of_node;
struct ravb_private *priv = netdev_priv(ndev); struct ravb_private *priv = netdev_priv(ndev);
const struct ravb_hw_info *info = priv->info;
struct ravb_tstamp_skb *ts_skb, *ts_skb2; struct ravb_tstamp_skb *ts_skb, *ts_skb2;
netif_tx_stop_all_queues(ndev); netif_tx_stop_all_queues(ndev);
...@@ -1713,7 +1718,7 @@ static int ravb_close(struct net_device *ndev) ...@@ -1713,7 +1718,7 @@ static int ravb_close(struct net_device *ndev)
of_phy_deregister_fixed_link(np); of_phy_deregister_fixed_link(np);
} }
if (priv->chip_id != RCAR_GEN2) { if (info->multi_irqs) {
free_irq(priv->tx_irqs[RAVB_NC], ndev); free_irq(priv->tx_irqs[RAVB_NC], ndev);
free_irq(priv->rx_irqs[RAVB_NC], ndev); free_irq(priv->rx_irqs[RAVB_NC], ndev);
free_irq(priv->tx_irqs[RAVB_BE], ndev); free_irq(priv->tx_irqs[RAVB_BE], ndev);
...@@ -1939,6 +1944,7 @@ static const struct ravb_hw_info ravb_gen3_hw_info = { ...@@ -1939,6 +1944,7 @@ static const struct ravb_hw_info ravb_gen3_hw_info = {
.max_rx_len = RX_BUF_SZ + RAVB_ALIGN - 1, .max_rx_len = RX_BUF_SZ + RAVB_ALIGN - 1,
.internal_delay = 1, .internal_delay = 1,
.tx_counters = 1, .tx_counters = 1,
.multi_irqs = 1,
}; };
static const struct ravb_hw_info ravb_gen2_hw_info = { static const struct ravb_hw_info ravb_gen2_hw_info = {
...@@ -2077,7 +2083,7 @@ static int ravb_probe(struct platform_device *pdev) ...@@ -2077,7 +2083,7 @@ static int ravb_probe(struct platform_device *pdev)
pm_runtime_enable(&pdev->dev); pm_runtime_enable(&pdev->dev);
pm_runtime_get_sync(&pdev->dev); pm_runtime_get_sync(&pdev->dev);
if (info->chip_id == RCAR_GEN3) if (info->multi_irqs)
irq = platform_get_irq_byname(pdev, "ch22"); irq = platform_get_irq_byname(pdev, "ch22");
else else
irq = platform_get_irq(pdev, 0); irq = platform_get_irq(pdev, 0);
...@@ -2117,7 +2123,7 @@ static int ravb_probe(struct platform_device *pdev) ...@@ -2117,7 +2123,7 @@ static int ravb_probe(struct platform_device *pdev)
priv->avb_link_active_low = priv->avb_link_active_low =
of_property_read_bool(np, "renesas,ether-link-active-low"); of_property_read_bool(np, "renesas,ether-link-active-low");
if (info->chip_id == RCAR_GEN3) { if (info->multi_irqs) {
irq = platform_get_irq_byname(pdev, "ch24"); irq = platform_get_irq_byname(pdev, "ch24");
if (irq < 0) { if (irq < 0) {
error = irq; error = irq;
......
...@@ -179,6 +179,7 @@ static int ravb_ptp_extts(struct ptp_clock_info *ptp, ...@@ -179,6 +179,7 @@ static int ravb_ptp_extts(struct ptp_clock_info *ptp,
{ {
struct ravb_private *priv = container_of(ptp, struct ravb_private, struct ravb_private *priv = container_of(ptp, struct ravb_private,
ptp.info); ptp.info);
const struct ravb_hw_info *info = priv->info;
struct net_device *ndev = priv->ndev; struct net_device *ndev = priv->ndev;
unsigned long flags; unsigned long flags;
...@@ -197,7 +198,7 @@ static int ravb_ptp_extts(struct ptp_clock_info *ptp, ...@@ -197,7 +198,7 @@ static int ravb_ptp_extts(struct ptp_clock_info *ptp,
priv->ptp.extts[req->index] = on; priv->ptp.extts[req->index] = on;
spin_lock_irqsave(&priv->lock, flags); spin_lock_irqsave(&priv->lock, flags);
if (priv->chip_id == RCAR_GEN2) if (!info->multi_irqs)
ravb_modify(ndev, GIC, GIC_PTCE, on ? GIC_PTCE : 0); ravb_modify(ndev, GIC, GIC_PTCE, on ? GIC_PTCE : 0);
else if (on) else if (on)
ravb_write(ndev, GIE_PTCS, GIE); ravb_write(ndev, GIE_PTCS, GIE);
...@@ -213,6 +214,7 @@ static int ravb_ptp_perout(struct ptp_clock_info *ptp, ...@@ -213,6 +214,7 @@ static int ravb_ptp_perout(struct ptp_clock_info *ptp,
{ {
struct ravb_private *priv = container_of(ptp, struct ravb_private, struct ravb_private *priv = container_of(ptp, struct ravb_private,
ptp.info); ptp.info);
const struct ravb_hw_info *info = priv->info;
struct net_device *ndev = priv->ndev; struct net_device *ndev = priv->ndev;
struct ravb_ptp_perout *perout; struct ravb_ptp_perout *perout;
unsigned long flags; unsigned long flags;
...@@ -252,7 +254,7 @@ static int ravb_ptp_perout(struct ptp_clock_info *ptp, ...@@ -252,7 +254,7 @@ static int ravb_ptp_perout(struct ptp_clock_info *ptp,
error = ravb_ptp_update_compare(priv, (u32)start_ns); error = ravb_ptp_update_compare(priv, (u32)start_ns);
if (!error) { if (!error) {
/* Unmask interrupt */ /* Unmask interrupt */
if (priv->chip_id == RCAR_GEN2) if (!info->multi_irqs)
ravb_modify(ndev, GIC, GIC_PTME, GIC_PTME); ravb_modify(ndev, GIC, GIC_PTME, GIC_PTME);
else else
ravb_write(ndev, GIE_PTMS0, GIE); ravb_write(ndev, GIE_PTMS0, GIE);
...@@ -264,7 +266,7 @@ static int ravb_ptp_perout(struct ptp_clock_info *ptp, ...@@ -264,7 +266,7 @@ static int ravb_ptp_perout(struct ptp_clock_info *ptp,
perout->period = 0; perout->period = 0;
/* Mask interrupt */ /* Mask interrupt */
if (priv->chip_id == RCAR_GEN2) if (!info->multi_irqs)
ravb_modify(ndev, GIC, GIC_PTME, 0); ravb_modify(ndev, GIC, GIC_PTME, 0);
else else
ravb_write(ndev, GID_PTMD0, GID); ravb_write(ndev, GID_PTMD0, GID);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment