|
@@ -32,6 +32,7 @@
|
|
#include "if_pci.h"
|
|
#include "if_pci.h"
|
|
#include "ahb_api.h"
|
|
#include "ahb_api.h"
|
|
#include "pci_api.h"
|
|
#include "pci_api.h"
|
|
|
|
+#include "hif_napi.h"
|
|
|
|
|
|
#if LINUX_VERSION_CODE >= KERNEL_VERSION(4, 1, 0)
|
|
#if LINUX_VERSION_CODE >= KERNEL_VERSION(4, 1, 0)
|
|
#define IRQF_DISABLED 0x00000020
|
|
#define IRQF_DISABLED 0x00000020
|
|
@@ -40,6 +41,7 @@
|
|
#define HIF_IC_CE0_IRQ_OFFSET 4
|
|
#define HIF_IC_CE0_IRQ_OFFSET 4
|
|
#define HIF_IC_MAX_IRQ 54
|
|
#define HIF_IC_MAX_IRQ 54
|
|
|
|
|
|
|
|
+static uint8_t ic_irqnum[HIF_IC_MAX_IRQ];
|
|
/* integrated chip irq names */
|
|
/* integrated chip irq names */
|
|
const char *ic_irqname[HIF_IC_MAX_IRQ] = {
|
|
const char *ic_irqname[HIF_IC_MAX_IRQ] = {
|
|
"misc_pulse1",
|
|
"misc_pulse1",
|
|
@@ -98,7 +100,6 @@ const char *ic_irqname[HIF_IC_MAX_IRQ] = {
|
|
"tcl2host_status_ring",
|
|
"tcl2host_status_ring",
|
|
};
|
|
};
|
|
|
|
|
|
-irqreturn_t hif_ahb_interrupt_handler(int irq, void *context);
|
|
|
|
/**
|
|
/**
|
|
* hif_disable_isr() - disable isr
|
|
* hif_disable_isr() - disable isr
|
|
*
|
|
*
|
|
@@ -114,8 +115,10 @@ void hif_ahb_disable_isr(struct hif_softc *scn)
|
|
|
|
|
|
hif_nointrs(scn);
|
|
hif_nointrs(scn);
|
|
ce_tasklet_kill(scn);
|
|
ce_tasklet_kill(scn);
|
|
|
|
+ hif_grp_tasklet_kill(scn);
|
|
tasklet_kill(&sc->intr_tq);
|
|
tasklet_kill(&sc->intr_tq);
|
|
qdf_atomic_set(&scn->active_tasklet_cnt, 0);
|
|
qdf_atomic_set(&scn->active_tasklet_cnt, 0);
|
|
|
|
+ qdf_atomic_set(&scn->active_grp_tasklet_cnt, 0);
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
/**
|
|
@@ -246,17 +249,19 @@ int hif_ahb_configure_irq(struct hif_pci_softc *sc)
|
|
int ret = 0;
|
|
int ret = 0;
|
|
struct hif_softc *scn = HIF_GET_SOFTC(sc);
|
|
struct hif_softc *scn = HIF_GET_SOFTC(sc);
|
|
struct platform_device *pdev = (struct platform_device *)sc->pdev;
|
|
struct platform_device *pdev = (struct platform_device *)sc->pdev;
|
|
- struct HIF_CE_state *hif_ce_state = HIF_GET_CE_STATE(scn);
|
|
|
|
|
|
+ struct HIF_CE_state *hif_state = HIF_GET_CE_STATE(scn);
|
|
|
|
+ struct hif_ext_group_entry *hif_ext_group;
|
|
int irq = 0;
|
|
int irq = 0;
|
|
- int i;
|
|
|
|
|
|
+ int i, j;
|
|
|
|
|
|
/* configure per CE interrupts */
|
|
/* configure per CE interrupts */
|
|
for (i = 0; i < scn->ce_count; i++) {
|
|
for (i = 0; i < scn->ce_count; i++) {
|
|
irq = platform_get_irq_byname(pdev, ic_irqname[HIF_IC_CE0_IRQ_OFFSET + i]);
|
|
irq = platform_get_irq_byname(pdev, ic_irqname[HIF_IC_CE0_IRQ_OFFSET + i]);
|
|
|
|
+ ic_irqnum[HIF_IC_CE0_IRQ_OFFSET + i] = irq;
|
|
ret = request_irq(irq ,
|
|
ret = request_irq(irq ,
|
|
hif_ahb_interrupt_handler,
|
|
hif_ahb_interrupt_handler,
|
|
IRQF_TRIGGER_RISING, ic_irqname[HIF_IC_CE0_IRQ_OFFSET + i],
|
|
IRQF_TRIGGER_RISING, ic_irqname[HIF_IC_CE0_IRQ_OFFSET + i],
|
|
- &hif_ce_state->tasklets[i]);
|
|
|
|
|
|
+ &hif_state->tasklets[i]);
|
|
if (ret) {
|
|
if (ret) {
|
|
dev_err(&pdev->dev, "ath_request_irq failed\n");
|
|
dev_err(&pdev->dev, "ath_request_irq failed\n");
|
|
ret = -1;
|
|
ret = -1;
|
|
@@ -265,6 +270,36 @@ int hif_ahb_configure_irq(struct hif_pci_softc *sc)
|
|
hif_ahb_irq_enable(scn, i);
|
|
hif_ahb_irq_enable(scn, i);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+ /* configure external interrupts */
|
|
|
|
+ for (i = 0; i < hif_state->hif_num_extgroup; i++) {
|
|
|
|
+
|
|
|
|
+ hif_ext_group = &hif_state->hif_ext_group[i];
|
|
|
|
+ if (hif_ext_group->configured) {
|
|
|
|
+
|
|
|
|
+ tasklet_init(&hif_ext_group->intr_tq,
|
|
|
|
+ hif_ext_grp_tasklet,
|
|
|
|
+ (unsigned long)hif_ext_group);
|
|
|
|
+ hif_ext_group->inited = true;
|
|
|
|
+
|
|
|
|
+ for (j = 0; j < hif_ext_group->numirq; j++) {
|
|
|
|
+ irq = platform_get_irq_byname(pdev,
|
|
|
|
+ ic_irqname[hif_ext_group->irq[j]]);
|
|
|
|
+
|
|
|
|
+ ic_irqnum[hif_ext_group->irq[j]] = irq;
|
|
|
|
+ ret = request_irq(irq,
|
|
|
|
+ hif_ext_group_ahb_interrupt_handler,
|
|
|
|
+ IRQF_TRIGGER_RISING, "wlan_ahb",
|
|
|
|
+ hif_ext_group);
|
|
|
|
+ if (ret) {
|
|
|
|
+ dev_err(&pdev->dev,
|
|
|
|
+ "ath_request_irq failed\n");
|
|
|
|
+ ret = -1;
|
|
|
|
+ goto end;
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+
|
|
end:
|
|
end:
|
|
return ret;
|
|
return ret;
|
|
}
|
|
}
|
|
@@ -275,6 +310,26 @@ irqreturn_t hif_ahb_interrupt_handler(int irq, void *context)
|
|
return ce_dispatch_interrupt(tasklet_entry->ce_id, tasklet_entry);
|
|
return ce_dispatch_interrupt(tasklet_entry->ce_id, tasklet_entry);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+irqreturn_t hif_ext_group_ahb_interrupt_handler(int irq, void *context)
|
|
|
|
+{
|
|
|
|
+ struct hif_ext_group_entry *hif_ext_group = context;
|
|
|
|
+ struct HIF_CE_state *hif_state = hif_ext_group->hif_state;
|
|
|
|
+ struct hif_softc *scn = HIF_GET_SOFTC(hif_state);
|
|
|
|
+ struct hif_opaque_softc *hif_hdl = GET_HIF_OPAQUE_HDL(scn);
|
|
|
|
+ uint32_t grp_id = hif_ext_group->grp_id;
|
|
|
|
+
|
|
|
|
+ hif_grp_irq_disable(scn, grp_id);
|
|
|
|
+
|
|
|
|
+ qdf_atomic_inc(&scn->active_grp_tasklet_cnt);
|
|
|
|
+
|
|
|
|
+ if (hif_ext_napi_enabled(hif_hdl, grp_id)) {
|
|
|
|
+ hif_napi_schedule_grp(hif_hdl, grp_id);
|
|
|
|
+ } else {
|
|
|
|
+ tasklet_schedule(&hif_ext_group->intr_tq);
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ return IRQ_HANDLED;
|
|
|
|
+}
|
|
|
|
|
|
/**
|
|
/**
|
|
* hif_target_sync() : ensure the target is ready
|
|
* hif_target_sync() : ensure the target is ready
|
|
@@ -493,9 +548,7 @@ void hif_ahb_reset_soc(struct hif_softc *hif_ctx)
|
|
void hif_ahb_nointrs(struct hif_softc *scn)
|
|
void hif_ahb_nointrs(struct hif_softc *scn)
|
|
{
|
|
{
|
|
int i;
|
|
int i;
|
|
- int irq;
|
|
|
|
struct hif_pci_softc *sc = HIF_GET_PCI_SOFTC(scn);
|
|
struct hif_pci_softc *sc = HIF_GET_PCI_SOFTC(scn);
|
|
- struct platform_device *pdev = (struct platform_device *)sc->pdev;
|
|
|
|
struct HIF_CE_state *hif_state = HIF_GET_CE_STATE(scn);
|
|
struct HIF_CE_state *hif_state = HIF_GET_CE_STATE(scn);
|
|
|
|
|
|
if (scn->request_irq_done == false)
|
|
if (scn->request_irq_done == false)
|
|
@@ -512,8 +565,8 @@ void hif_ahb_nointrs(struct hif_softc *scn)
|
|
free_irq(sc->irq, sc);
|
|
free_irq(sc->irq, sc);
|
|
} else {
|
|
} else {
|
|
for (i = 0; i < scn->ce_count; i++) {
|
|
for (i = 0; i < scn->ce_count; i++) {
|
|
- irq = platform_get_irq_byname(pdev, ic_irqname[HIF_IC_CE0_IRQ_OFFSET + i]);
|
|
|
|
- free_irq(irq, sc);
|
|
|
|
|
|
+ free_irq(ic_irqnum[HIF_IC_CE0_IRQ_OFFSET + i],
|
|
|
|
+ sc);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
@@ -592,3 +645,29 @@ void hif_ahb_irq_disable(struct hif_softc *scn, int ce_id)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
+
|
|
|
|
+void hif_ahb_grp_irq_disable(struct hif_softc *scn, uint32_t grp_id)
|
|
|
|
+{
|
|
|
|
+ struct HIF_CE_state *hif_state = HIF_GET_CE_STATE(scn);
|
|
|
|
+ struct hif_ext_group_entry *hif_ext_group;
|
|
|
|
+ uint32_t i;
|
|
|
|
+
|
|
|
|
+ hif_ext_group = &hif_state->hif_ext_group[grp_id];
|
|
|
|
+
|
|
|
|
+ for (i = 0; i < hif_ext_group->numirq; i++) {
|
|
|
|
+ disable_irq(ic_irqnum[hif_ext_group->irq[i]]);
|
|
|
|
+ }
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+void hif_ahb_grp_irq_enable(struct hif_softc *scn, uint32_t grp_id)
|
|
|
|
+{
|
|
|
|
+ struct HIF_CE_state *hif_state = HIF_GET_CE_STATE(scn);
|
|
|
|
+ struct hif_ext_group_entry *hif_ext_group;
|
|
|
|
+ uint32_t i;
|
|
|
|
+
|
|
|
|
+ hif_ext_group = &hif_state->hif_ext_group[grp_id];
|
|
|
|
+
|
|
|
|
+ for (i = 0; i < hif_ext_group->numirq; i++) {
|
|
|
|
+ enable_irq(ic_irqnum[hif_ext_group->irq[i]]);
|
|
|
|
+ }
|
|
|
|
+}
|