numam-dpdk/drivers/mempool/octeontx2/otx2_mempool_irq.c
Harman Kalra 9f741506f8 drivers/octeontx2: fix recursive interrupts
In case of QINT interrupt occurrence, SW fails to clear the QINT
line resulting in recursive interrupts because currently interrupt
handler gets the cause of the interrupt by reading
NIX_LF_RQ[SQ/CQ/AURA/POOL]_OP_INT but does not write 1 to clear
RQ[SQ/CQ/ERR]_INT field in respective NIX_LF_RQ[SQ/CQ/AURA/POOL]_OP_INT
registers.

Fixes: dc47ba15f645 ("net/octeontx2: handle queue specific error interrupts")
Fixes: 50b95c3ea7af ("mempool/octeontx2: add NPA IRQ handler")

Signed-off-by: Harman Kalra <hkalra@marvell.com>
Acked-by: Jerin Jacob <jerinj@marvell.com>
2019-07-29 22:18:01 +02:00

304 lines
6.9 KiB
C

/* SPDX-License-Identifier: BSD-3-Clause
* Copyright(C) 2019 Marvell International Ltd.
*/
#include <inttypes.h>
#include <rte_common.h>
#include <rte_bus_pci.h>
#include "otx2_common.h"
#include "otx2_irq.h"
#include "otx2_mempool.h"
static void
npa_lf_err_irq(void *param)
{
struct otx2_npa_lf *lf = (struct otx2_npa_lf *)param;
uint64_t intr;
intr = otx2_read64(lf->base + NPA_LF_ERR_INT);
if (intr == 0)
return;
otx2_err("Err_intr=0x%" PRIx64 "", intr);
/* Clear interrupt */
otx2_write64(intr, lf->base + NPA_LF_ERR_INT);
}
static int
npa_lf_register_err_irq(struct otx2_npa_lf *lf)
{
struct rte_intr_handle *handle = lf->intr_handle;
int rc, vec;
vec = lf->npa_msixoff + NPA_LF_INT_VEC_ERR_INT;
/* Clear err interrupt */
otx2_write64(~0ull, lf->base + NPA_LF_ERR_INT_ENA_W1C);
/* Register err interrupt vector */
rc = otx2_register_irq(handle, npa_lf_err_irq, lf, vec);
/* Enable hw interrupt */
otx2_write64(~0ull, lf->base + NPA_LF_ERR_INT_ENA_W1S);
return rc;
}
static void
npa_lf_unregister_err_irq(struct otx2_npa_lf *lf)
{
struct rte_intr_handle *handle = lf->intr_handle;
int vec;
vec = lf->npa_msixoff + NPA_LF_INT_VEC_ERR_INT;
/* Clear err interrupt */
otx2_write64(~0ull, lf->base + NPA_LF_ERR_INT_ENA_W1C);
otx2_unregister_irq(handle, npa_lf_err_irq, lf, vec);
}
static void
npa_lf_ras_irq(void *param)
{
struct otx2_npa_lf *lf = (struct otx2_npa_lf *)param;
uint64_t intr;
intr = otx2_read64(lf->base + NPA_LF_RAS);
if (intr == 0)
return;
otx2_err("Ras_intr=0x%" PRIx64 "", intr);
/* Clear interrupt */
otx2_write64(intr, lf->base + NPA_LF_RAS);
}
static int
npa_lf_register_ras_irq(struct otx2_npa_lf *lf)
{
struct rte_intr_handle *handle = lf->intr_handle;
int rc, vec;
vec = lf->npa_msixoff + NPA_LF_INT_VEC_POISON;
/* Clear err interrupt */
otx2_write64(~0ull, lf->base + NPA_LF_RAS_ENA_W1C);
/* Set used interrupt vectors */
rc = otx2_register_irq(handle, npa_lf_ras_irq, lf, vec);
/* Enable hw interrupt */
otx2_write64(~0ull, lf->base + NPA_LF_RAS_ENA_W1S);
return rc;
}
static void
npa_lf_unregister_ras_irq(struct otx2_npa_lf *lf)
{
int vec;
struct rte_intr_handle *handle = lf->intr_handle;
vec = lf->npa_msixoff + NPA_LF_INT_VEC_POISON;
/* Clear err interrupt */
otx2_write64(~0ull, lf->base + NPA_LF_RAS_ENA_W1C);
otx2_unregister_irq(handle, npa_lf_ras_irq, lf, vec);
}
static inline uint8_t
npa_lf_q_irq_get_and_clear(struct otx2_npa_lf *lf, uint32_t q,
uint32_t off, uint64_t mask)
{
uint64_t reg, wdata;
uint8_t qint;
wdata = (uint64_t)q << 44;
reg = otx2_atomic64_add_nosync(wdata, (int64_t *)(lf->base + off));
if (reg & BIT_ULL(42) /* OP_ERR */) {
otx2_err("Failed execute irq get off=0x%x", off);
return 0;
}
qint = reg & 0xff;
wdata &= mask;
otx2_write64(wdata | qint, lf->base + off);
return qint;
}
static inline uint8_t
npa_lf_pool_irq_get_and_clear(struct otx2_npa_lf *lf, uint32_t p)
{
return npa_lf_q_irq_get_and_clear(lf, p, NPA_LF_POOL_OP_INT, ~0xff00);
}
static inline uint8_t
npa_lf_aura_irq_get_and_clear(struct otx2_npa_lf *lf, uint32_t a)
{
return npa_lf_q_irq_get_and_clear(lf, a, NPA_LF_AURA_OP_INT, ~0xff00);
}
static void
npa_lf_q_irq(void *param)
{
struct otx2_npa_qint *qint = (struct otx2_npa_qint *)param;
struct otx2_npa_lf *lf = qint->lf;
uint8_t irq, qintx = qint->qintx;
uint32_t q, pool, aura;
uint64_t intr;
intr = otx2_read64(lf->base + NPA_LF_QINTX_INT(qintx));
if (intr == 0)
return;
otx2_err("queue_intr=0x%" PRIx64 " qintx=%d", intr, qintx);
/* Handle pool queue interrupts */
for (q = 0; q < lf->nr_pools; q++) {
/* Skip disabled POOL */
if (rte_bitmap_get(lf->npa_bmp, q))
continue;
pool = q % lf->qints;
irq = npa_lf_pool_irq_get_and_clear(lf, pool);
if (irq & BIT_ULL(NPA_POOL_ERR_INT_OVFLS))
otx2_err("Pool=%d NPA_POOL_ERR_INT_OVFLS", pool);
if (irq & BIT_ULL(NPA_POOL_ERR_INT_RANGE))
otx2_err("Pool=%d NPA_POOL_ERR_INT_RANGE", pool);
if (irq & BIT_ULL(NPA_POOL_ERR_INT_PERR))
otx2_err("Pool=%d NPA_POOL_ERR_INT_PERR", pool);
}
/* Handle aura queue interrupts */
for (q = 0; q < lf->nr_pools; q++) {
/* Skip disabled AURA */
if (rte_bitmap_get(lf->npa_bmp, q))
continue;
aura = q % lf->qints;
irq = npa_lf_aura_irq_get_and_clear(lf, aura);
if (irq & BIT_ULL(NPA_AURA_ERR_INT_AURA_ADD_OVER))
otx2_err("Aura=%d NPA_AURA_ERR_INT_ADD_OVER", aura);
if (irq & BIT_ULL(NPA_AURA_ERR_INT_AURA_ADD_UNDER))
otx2_err("Aura=%d NPA_AURA_ERR_INT_ADD_UNDER", aura);
if (irq & BIT_ULL(NPA_AURA_ERR_INT_AURA_FREE_UNDER))
otx2_err("Aura=%d NPA_AURA_ERR_INT_FREE_UNDER", aura);
if (irq & BIT_ULL(NPA_AURA_ERR_INT_POOL_DIS))
otx2_err("Aura=%d NPA_AURA_ERR_POOL_DIS", aura);
}
/* Clear interrupt */
otx2_write64(intr, lf->base + NPA_LF_QINTX_INT(qintx));
otx2_mempool_ctx_dump(lf);
}
static int
npa_lf_register_queue_irqs(struct otx2_npa_lf *lf)
{
struct rte_intr_handle *handle = lf->intr_handle;
int vec, q, qs, rc = 0;
/* Figure out max qintx required */
qs = RTE_MIN(lf->qints, lf->nr_pools);
for (q = 0; q < qs; q++) {
vec = lf->npa_msixoff + NPA_LF_INT_VEC_QINT_START + q;
/* Clear QINT CNT */
otx2_write64(0, lf->base + NPA_LF_QINTX_CNT(q));
/* Clear interrupt */
otx2_write64(~0ull, lf->base + NPA_LF_QINTX_ENA_W1C(q));
struct otx2_npa_qint *qintmem = lf->npa_qint_mem;
qintmem += q;
qintmem->lf = lf;
qintmem->qintx = q;
/* Sync qints_mem update */
rte_smp_wmb();
/* Register queue irq vector */
rc = otx2_register_irq(handle, npa_lf_q_irq, qintmem, vec);
if (rc)
break;
otx2_write64(0, lf->base + NPA_LF_QINTX_CNT(q));
otx2_write64(0, lf->base + NPA_LF_QINTX_INT(q));
/* Enable QINT interrupt */
otx2_write64(~0ull, lf->base + NPA_LF_QINTX_ENA_W1S(q));
}
return rc;
}
static void
npa_lf_unregister_queue_irqs(struct otx2_npa_lf *lf)
{
struct rte_intr_handle *handle = lf->intr_handle;
int vec, q, qs;
/* Figure out max qintx required */
qs = RTE_MIN(lf->qints, lf->nr_pools);
for (q = 0; q < qs; q++) {
vec = lf->npa_msixoff + NPA_LF_INT_VEC_QINT_START + q;
/* Clear QINT CNT */
otx2_write64(0, lf->base + NPA_LF_QINTX_CNT(q));
otx2_write64(0, lf->base + NPA_LF_QINTX_INT(q));
/* Clear interrupt */
otx2_write64(~0ull, lf->base + NPA_LF_QINTX_ENA_W1C(q));
struct otx2_npa_qint *qintmem = lf->npa_qint_mem;
qintmem += q;
/* Unregister queue irq vector */
otx2_unregister_irq(handle, npa_lf_q_irq, qintmem, vec);
qintmem->lf = NULL;
qintmem->qintx = 0;
}
}
int
otx2_npa_register_irqs(struct otx2_npa_lf *lf)
{
int rc;
if (lf->npa_msixoff == MSIX_VECTOR_INVALID) {
otx2_err("Invalid NPALF MSIX vector offset vector: 0x%x",
lf->npa_msixoff);
return -EINVAL;
}
/* Register lf err interrupt */
rc = npa_lf_register_err_irq(lf);
/* Register RAS interrupt */
rc |= npa_lf_register_ras_irq(lf);
/* Register queue interrupts */
rc |= npa_lf_register_queue_irqs(lf);
return rc;
}
void
otx2_npa_unregister_irqs(struct otx2_npa_lf *lf)
{
npa_lf_unregister_err_irq(lf);
npa_lf_unregister_ras_irq(lf);
npa_lf_unregister_queue_irqs(lf);
}