f-stack/dpdk/drivers/common/cnxk/roc_cpt.c

1083 lines
24 KiB
C

/* SPDX-License-Identifier: BSD-3-Clause
* Copyright(C) 2021 Marvell.
*/
#include "roc_api.h"
#include "roc_priv.h"
#define CPT_IQ_FC_LEN 128
#define CPT_IQ_GRP_LEN 16
#define CPT_IQ_NB_DESC_MULTIPLIER 40
/* The effective queue size to software is (CPT_LF_Q_SIZE[SIZE_DIV40] - 1 - 8).
*
* CPT requires 320 free entries (+8). And 40 entries are required for
* allowing CPT to discard packet when the queues are full (+1).
*/
#define CPT_IQ_NB_DESC_SIZE_DIV40(nb_desc) \
(PLT_DIV_CEIL(nb_desc, CPT_IQ_NB_DESC_MULTIPLIER) + 1 + 8)
#define CPT_IQ_GRP_SIZE(nb_desc) \
(CPT_IQ_NB_DESC_SIZE_DIV40(nb_desc) * CPT_IQ_GRP_LEN)
#define CPT_LF_MAX_NB_DESC 128000
#define CPT_LF_DEFAULT_NB_DESC 1024
#define CPT_LF_FC_MIN_THRESHOLD 32
static void
cpt_lf_misc_intr_enb_dis(struct roc_cpt_lf *lf, bool enb)
{
/* Enable all cpt lf error irqs except RQ_DISABLED and CQ_DISABLED */
if (enb)
plt_write64((BIT_ULL(6) | BIT_ULL(5) | BIT_ULL(3) | BIT_ULL(2) |
BIT_ULL(1)),
lf->rbase + CPT_LF_MISC_INT_ENA_W1S);
else
plt_write64((BIT_ULL(6) | BIT_ULL(5) | BIT_ULL(3) | BIT_ULL(2) |
BIT_ULL(1)),
lf->rbase + CPT_LF_MISC_INT_ENA_W1C);
}
static void
cpt_lf_misc_irq(void *param)
{
struct roc_cpt_lf *lf = (struct roc_cpt_lf *)param;
struct dev *dev = lf->dev;
uint64_t intr;
intr = plt_read64(lf->rbase + CPT_LF_MISC_INT);
if (intr == 0)
return;
plt_err("Err_irq=0x%" PRIx64 " pf=%d, vf=%d", intr, dev->pf, dev->vf);
/* Dump lf registers */
cpt_lf_print(lf);
/* Clear interrupt */
plt_write64(intr, lf->rbase + CPT_LF_MISC_INT);
}
static int
cpt_lf_register_misc_irq(struct roc_cpt_lf *lf)
{
struct plt_pci_device *pci_dev = lf->pci_dev;
struct plt_intr_handle *handle;
int rc, vec;
handle = pci_dev->intr_handle;
vec = lf->msixoff + CPT_LF_INT_VEC_MISC;
/* Clear err interrupt */
cpt_lf_misc_intr_enb_dis(lf, false);
/* Set used interrupt vectors */
rc = dev_irq_register(handle, cpt_lf_misc_irq, lf, vec);
/* Enable all dev interrupt except for RQ_DISABLED */
cpt_lf_misc_intr_enb_dis(lf, true);
return rc;
}
static void
cpt_lf_unregister_misc_irq(struct roc_cpt_lf *lf)
{
struct plt_pci_device *pci_dev = lf->pci_dev;
struct plt_intr_handle *handle;
int vec;
handle = pci_dev->intr_handle;
vec = lf->msixoff + CPT_LF_INT_VEC_MISC;
/* Clear err interrupt */
cpt_lf_misc_intr_enb_dis(lf, false);
dev_irq_unregister(handle, cpt_lf_misc_irq, lf, vec);
}
static void
cpt_lf_done_intr_enb_dis(struct roc_cpt_lf *lf, bool enb)
{
if (enb)
plt_write64(0x1, lf->rbase + CPT_LF_DONE_INT_ENA_W1S);
else
plt_write64(0x1, lf->rbase + CPT_LF_DONE_INT_ENA_W1C);
}
static void
cpt_lf_done_irq(void *param)
{
struct roc_cpt_lf *lf = param;
uint64_t done_wait;
uint64_t intr;
/* Read the number of completed requests */
intr = plt_read64(lf->rbase + CPT_LF_DONE);
if (intr == 0)
return;
done_wait = plt_read64(lf->rbase + CPT_LF_DONE_WAIT);
/* Acknowledge the number of completed requests */
plt_write64(intr, lf->rbase + CPT_LF_DONE_ACK);
plt_write64(done_wait, lf->rbase + CPT_LF_DONE_WAIT);
}
static int
cpt_lf_register_done_irq(struct roc_cpt_lf *lf)
{
struct plt_pci_device *pci_dev = lf->pci_dev;
struct plt_intr_handle *handle;
int rc, vec;
handle = pci_dev->intr_handle;
vec = lf->msixoff + CPT_LF_INT_VEC_DONE;
/* Clear done interrupt */
cpt_lf_done_intr_enb_dis(lf, false);
/* Set used interrupt vectors */
rc = dev_irq_register(handle, cpt_lf_done_irq, lf, vec);
/* Enable done interrupt */
cpt_lf_done_intr_enb_dis(lf, true);
return rc;
}
static void
cpt_lf_unregister_done_irq(struct roc_cpt_lf *lf)
{
struct plt_pci_device *pci_dev = lf->pci_dev;
struct plt_intr_handle *handle;
int vec;
handle = pci_dev->intr_handle;
vec = lf->msixoff + CPT_LF_INT_VEC_DONE;
/* Clear done interrupt */
cpt_lf_done_intr_enb_dis(lf, false);
dev_irq_unregister(handle, cpt_lf_done_irq, lf, vec);
}
static int
cpt_lf_register_irqs(struct roc_cpt_lf *lf)
{
int rc;
if (lf->msixoff == MSIX_VECTOR_INVALID) {
plt_err("Invalid CPTLF MSIX vector offset vector: 0x%x",
lf->msixoff);
return -EINVAL;
}
/* Register lf err interrupt */
rc = cpt_lf_register_misc_irq(lf);
if (rc)
plt_err("Error registering IRQs");
rc = cpt_lf_register_done_irq(lf);
if (rc)
plt_err("Error registering IRQs");
return rc;
}
static void
cpt_lf_unregister_irqs(struct roc_cpt_lf *lf)
{
cpt_lf_unregister_misc_irq(lf);
cpt_lf_unregister_done_irq(lf);
}
static void
cpt_lf_dump(struct roc_cpt_lf *lf)
{
plt_cpt_dbg("CPT LF");
plt_cpt_dbg("RBASE: 0x%016" PRIx64, lf->rbase);
plt_cpt_dbg("LMT_BASE: 0x%016" PRIx64, lf->lmt_base);
plt_cpt_dbg("MSIXOFF: 0x%x", lf->msixoff);
plt_cpt_dbg("LF_ID: 0x%x", lf->lf_id);
plt_cpt_dbg("NB DESC: %d", lf->nb_desc);
plt_cpt_dbg("FC_ADDR: 0x%016" PRIx64, (uintptr_t)lf->fc_addr);
plt_cpt_dbg("CQ.VADDR: 0x%016" PRIx64, (uintptr_t)lf->iq_vaddr);
plt_cpt_dbg("CPT LF REG:");
plt_cpt_dbg("LF_CTL[0x%016llx]: 0x%016" PRIx64, CPT_LF_CTL,
plt_read64(lf->rbase + CPT_LF_CTL));
plt_cpt_dbg("LF_INPROG[0x%016llx]: 0x%016" PRIx64, CPT_LF_INPROG,
plt_read64(lf->rbase + CPT_LF_INPROG));
plt_cpt_dbg("Q_BASE[0x%016llx]: 0x%016" PRIx64, CPT_LF_Q_BASE,
plt_read64(lf->rbase + CPT_LF_Q_BASE));
plt_cpt_dbg("Q_SIZE[0x%016llx]: 0x%016" PRIx64, CPT_LF_Q_SIZE,
plt_read64(lf->rbase + CPT_LF_Q_SIZE));
plt_cpt_dbg("Q_INST_PTR[0x%016llx]: 0x%016" PRIx64, CPT_LF_Q_INST_PTR,
plt_read64(lf->rbase + CPT_LF_Q_INST_PTR));
plt_cpt_dbg("Q_GRP_PTR[0x%016llx]: 0x%016" PRIx64, CPT_LF_Q_GRP_PTR,
plt_read64(lf->rbase + CPT_LF_Q_GRP_PTR));
}
int
cpt_lf_outb_cfg(struct dev *dev, uint16_t sso_pf_func, uint16_t nix_pf_func,
uint8_t lf_id, bool ena)
{
struct cpt_inline_ipsec_cfg_msg *req;
struct mbox *mbox = dev->mbox;
req = mbox_alloc_msg_cpt_inline_ipsec_cfg(mbox);
if (req == NULL)
return -ENOSPC;
req->dir = CPT_INLINE_OUTBOUND;
req->slot = lf_id;
if (ena) {
req->enable = 1;
req->sso_pf_func = sso_pf_func;
req->nix_pf_func = nix_pf_func;
} else {
req->enable = 0;
}
return mbox_process(mbox);
}
int
roc_cpt_inline_ipsec_cfg(struct dev *cpt_dev, uint8_t lf_id,
struct roc_nix *roc_nix)
{
bool ena = roc_nix ? true : false;
uint16_t nix_pf_func = 0;
uint16_t sso_pf_func = 0;
if (ena) {
nix_pf_func = roc_nix_get_pf_func(roc_nix);
sso_pf_func = idev_sso_pffunc_get();
}
return cpt_lf_outb_cfg(cpt_dev, sso_pf_func, nix_pf_func, lf_id, ena);
}
int
roc_cpt_inline_ipsec_inb_cfg_read(struct roc_cpt *roc_cpt,
struct nix_inline_ipsec_cfg *inb_cfg)
{
struct cpt *cpt = roc_cpt_to_cpt_priv(roc_cpt);
struct dev *dev = &cpt->dev;
struct msg_req *req;
req = mbox_alloc_msg_nix_read_inline_ipsec_cfg(dev->mbox);
if (req == NULL)
return -EIO;
return mbox_process_msg(dev->mbox, (void *)&inb_cfg);
}
int
roc_cpt_inline_ipsec_inb_cfg(struct roc_cpt *roc_cpt, uint16_t param1,
uint16_t param2, uint16_t opcode)
{
struct cpt *cpt = roc_cpt_to_cpt_priv(roc_cpt);
struct cpt_rx_inline_lf_cfg_msg *req;
struct mbox *mbox;
mbox = cpt->dev.mbox;
req = mbox_alloc_msg_cpt_rx_inline_lf_cfg(mbox);
if (req == NULL)
return -ENOSPC;
req->sso_pf_func = idev_sso_pffunc_get();
req->param1 = param1;
req->param2 = param2;
req->opcode = opcode;
return mbox_process(mbox);
}
int
roc_cpt_rxc_time_cfg(struct roc_cpt *roc_cpt, struct roc_cpt_rxc_time_cfg *cfg)
{
struct cpt *cpt = roc_cpt_to_cpt_priv(roc_cpt);
struct cpt_rxc_time_cfg_req *req;
struct dev *dev = &cpt->dev;
req = mbox_alloc_msg_cpt_rxc_time_cfg(dev->mbox);
if (req == NULL)
return -ENOSPC;
req->blkaddr = 0;
/* The step value is in microseconds. */
req->step = cfg->step;
/* The timeout will be: limit * step microseconds */
req->zombie_limit = cfg->zombie_limit;
req->zombie_thres = cfg->zombie_thres;
/* The timeout will be: limit * step microseconds */
req->active_limit = cfg->active_limit;
req->active_thres = cfg->active_thres;
return mbox_process(dev->mbox);
}
int
cpt_get_msix_offset(struct dev *dev, struct msix_offset_rsp **msix_rsp)
{
struct mbox *mbox = dev->mbox;
int rc;
/* Get MSIX vector offsets */
mbox_alloc_msg_msix_offset(mbox);
rc = mbox_process_msg(mbox, (void *)msix_rsp);
return rc;
}
int
cpt_lfs_attach(struct dev *dev, uint8_t blkaddr, bool modify, uint16_t nb_lf)
{
struct mbox *mbox = dev->mbox;
struct rsrc_attach_req *req;
if (blkaddr != RVU_BLOCK_ADDR_CPT0 && blkaddr != RVU_BLOCK_ADDR_CPT1)
return -EINVAL;
/* Attach CPT(lf) */
req = mbox_alloc_msg_attach_resources(mbox);
if (req == NULL)
return -ENOSPC;
req->cptlfs = nb_lf;
req->modify = modify;
req->cpt_blkaddr = blkaddr;
return mbox_process(mbox);
}
int
cpt_lfs_detach(struct dev *dev)
{
struct mbox *mbox = dev->mbox;
struct rsrc_detach_req *req;
req = mbox_alloc_msg_detach_resources(mbox);
if (req == NULL)
return -ENOSPC;
req->cptlfs = 1;
req->partial = 1;
return mbox_process(mbox);
}
static int
cpt_available_lfs_get(struct dev *dev, uint16_t *nb_lf)
{
struct mbox *mbox = dev->mbox;
struct free_rsrcs_rsp *rsp;
int rc;
mbox_alloc_msg_free_rsrc_cnt(mbox);
rc = mbox_process_msg(mbox, (void *)&rsp);
if (rc)
return -EIO;
*nb_lf = PLT_MAX((uint16_t)rsp->cpt, (uint16_t)rsp->cpt1);
return 0;
}
int
cpt_lfs_alloc(struct dev *dev, uint8_t eng_grpmsk, uint8_t blkaddr,
bool inl_dev_sso)
{
struct cpt_lf_alloc_req_msg *req;
struct mbox *mbox = dev->mbox;
if (blkaddr != RVU_BLOCK_ADDR_CPT0 && blkaddr != RVU_BLOCK_ADDR_CPT1)
return -EINVAL;
req = mbox_alloc_msg_cpt_lf_alloc(mbox);
if (!req)
return -ENOSPC;
req->nix_pf_func = 0;
if (inl_dev_sso && nix_inl_dev_pffunc_get())
req->sso_pf_func = nix_inl_dev_pffunc_get();
else
req->sso_pf_func = idev_sso_pffunc_get();
req->eng_grpmsk = eng_grpmsk;
req->blkaddr = blkaddr;
return mbox_process(mbox);
}
int
cpt_lfs_free(struct dev *dev)
{
mbox_alloc_msg_cpt_lf_free(dev->mbox);
return mbox_process(dev->mbox);
}
static int
cpt_hardware_caps_get(struct dev *dev, struct roc_cpt *roc_cpt)
{
struct cpt_caps_rsp_msg *rsp;
int ret;
mbox_alloc_msg_cpt_caps_get(dev->mbox);
ret = mbox_process_msg(dev->mbox, (void *)&rsp);
if (ret)
return -EIO;
roc_cpt->cpt_revision = rsp->cpt_revision;
mbox_memcpy(roc_cpt->hw_caps, rsp->eng_caps,
sizeof(union cpt_eng_caps) * CPT_MAX_ENG_TYPES);
return 0;
}
static uint32_t
cpt_lf_iq_mem_calc(uint32_t nb_desc)
{
uint32_t len;
/* Space for instruction group memory */
len = CPT_IQ_GRP_SIZE(nb_desc);
/* Align to 128B */
len = PLT_ALIGN(len, ROC_ALIGN);
/* Space for FC */
len += CPT_IQ_FC_LEN;
/* For instruction queues */
len += PLT_ALIGN(CPT_IQ_NB_DESC_SIZE_DIV40(nb_desc) *
CPT_IQ_NB_DESC_MULTIPLIER *
sizeof(struct cpt_inst_s),
ROC_ALIGN);
return len;
}
static inline void
cpt_iq_init(struct roc_cpt_lf *lf)
{
union cpt_lf_q_size lf_q_size = {.u = 0x0};
union cpt_lf_q_base lf_q_base = {.u = 0x0};
uintptr_t addr;
lf->io_addr = lf->rbase + CPT_LF_NQX(0);
/* Disable command queue */
roc_cpt_iq_disable(lf);
/* Set command queue base address */
addr = (uintptr_t)lf->iq_vaddr +
PLT_ALIGN(CPT_IQ_GRP_SIZE(lf->nb_desc), ROC_ALIGN);
lf_q_base.u = addr;
plt_write64(lf_q_base.u, lf->rbase + CPT_LF_Q_BASE);
/* Set command queue size */
lf_q_size.s.size_div40 = CPT_IQ_NB_DESC_SIZE_DIV40(lf->nb_desc);
plt_write64(lf_q_size.u, lf->rbase + CPT_LF_Q_SIZE);
lf->fc_addr = (uint64_t *)addr;
}
int
roc_cpt_dev_configure(struct roc_cpt *roc_cpt, int nb_lf)
{
struct cpt *cpt = roc_cpt_to_cpt_priv(roc_cpt);
uint8_t blkaddr[ROC_CPT_MAX_BLKS];
struct msix_offset_rsp *rsp;
uint8_t eng_grpmsk;
int blknum = 0;
int rc, i;
blkaddr[0] = RVU_BLOCK_ADDR_CPT0;
blkaddr[1] = RVU_BLOCK_ADDR_CPT1;
if ((roc_cpt->cpt_revision == ROC_CPT_REVISION_ID_98XX) &&
(cpt->dev.pf_func & 0x1))
blknum = (blknum + 1) % ROC_CPT_MAX_BLKS;
/* Request LF resources */
rc = cpt_lfs_attach(&cpt->dev, blkaddr[blknum], true, nb_lf);
/* Request LFs from another block if current block has less LFs */
if (roc_cpt->cpt_revision == ROC_CPT_REVISION_ID_98XX && rc == ENOSPC) {
blknum = (blknum + 1) % ROC_CPT_MAX_BLKS;
rc = cpt_lfs_attach(&cpt->dev, blkaddr[blknum], true, nb_lf);
}
if (rc) {
plt_err("Could not attach LFs");
return rc;
}
for (i = 0; i < nb_lf; i++)
cpt->lf_blkaddr[i] = blkaddr[blknum];
eng_grpmsk = (1 << roc_cpt->eng_grp[CPT_ENG_TYPE_AE]) |
(1 << roc_cpt->eng_grp[CPT_ENG_TYPE_SE]) |
(1 << roc_cpt->eng_grp[CPT_ENG_TYPE_IE]);
rc = cpt_lfs_alloc(&cpt->dev, eng_grpmsk, blkaddr[blknum], false);
if (rc)
goto lfs_detach;
rc = cpt_get_msix_offset(&cpt->dev, &rsp);
if (rc)
goto lfs_free;
for (i = 0; i < nb_lf; i++)
cpt->lf_msix_off[i] =
(cpt->lf_blkaddr[i] == RVU_BLOCK_ADDR_CPT1) ?
rsp->cpt1_lf_msixoff[i] :
rsp->cptlf_msixoff[i];
roc_cpt->nb_lf = nb_lf;
return 0;
lfs_free:
cpt_lfs_free(&cpt->dev);
lfs_detach:
cpt_lfs_detach(&cpt->dev);
return rc;
}
uint64_t
cpt_get_blkaddr(struct dev *dev)
{
uint64_t reg;
uint64_t off;
/* Reading the discovery register to know which CPT is the LF
* attached to. Assume CPT LF's of only one block are attached
* to a pffunc.
*/
if (dev_is_vf(dev))
off = RVU_VF_BLOCK_ADDRX_DISC(RVU_BLOCK_ADDR_CPT1);
else
off = RVU_PF_BLOCK_ADDRX_DISC(RVU_BLOCK_ADDR_CPT1);
reg = plt_read64(dev->bar2 + off);
return reg & 0x1FFULL ? RVU_BLOCK_ADDR_CPT1 : RVU_BLOCK_ADDR_CPT0;
}
int
cpt_lf_init(struct roc_cpt_lf *lf)
{
struct dev *dev = lf->dev;
uint64_t blkaddr;
void *iq_mem;
int rc;
if (lf->nb_desc == 0 || lf->nb_desc > CPT_LF_MAX_NB_DESC)
lf->nb_desc = CPT_LF_DEFAULT_NB_DESC;
/* Allocate memory for instruction queue for CPT LF. */
iq_mem = plt_zmalloc(cpt_lf_iq_mem_calc(lf->nb_desc), ROC_ALIGN);
if (iq_mem == NULL)
return -ENOMEM;
plt_atomic_thread_fence(__ATOMIC_ACQ_REL);
blkaddr = cpt_get_blkaddr(dev);
lf->rbase = dev->bar2 + ((blkaddr << 20) | (lf->lf_id << 12));
lf->iq_vaddr = iq_mem;
lf->lmt_base = dev->lmt_base;
lf->pf_func = dev->pf_func;
/* Initialize instruction queue */
cpt_iq_init(lf);
rc = cpt_lf_register_irqs(lf);
if (rc)
goto disable_iq;
return 0;
disable_iq:
roc_cpt_iq_disable(lf);
plt_free(iq_mem);
return rc;
}
int
roc_cpt_lf_init(struct roc_cpt *roc_cpt, struct roc_cpt_lf *lf)
{
struct cpt *cpt = roc_cpt_to_cpt_priv(roc_cpt);
int rc;
lf->dev = &cpt->dev;
lf->roc_cpt = roc_cpt;
lf->msixoff = cpt->lf_msix_off[lf->lf_id];
lf->pci_dev = cpt->pci_dev;
rc = cpt_lf_init(lf);
if (rc)
return rc;
/* LF init successful */
roc_cpt->lf[lf->lf_id] = lf;
return rc;
}
int
roc_cpt_dev_init(struct roc_cpt *roc_cpt)
{
struct plt_pci_device *pci_dev;
uint16_t nb_lf_avail;
struct dev *dev;
struct cpt *cpt;
int rc;
if (roc_cpt == NULL || roc_cpt->pci_dev == NULL)
return -EINVAL;
PLT_STATIC_ASSERT(sizeof(struct cpt) <= ROC_CPT_MEM_SZ);
cpt = roc_cpt_to_cpt_priv(roc_cpt);
memset(cpt, 0, sizeof(*cpt));
pci_dev = roc_cpt->pci_dev;
dev = &cpt->dev;
/* Initialize device */
rc = dev_init(dev, pci_dev);
if (rc) {
plt_err("Failed to init roc device");
return rc;
}
cpt->pci_dev = pci_dev;
roc_cpt->lmt_base = dev->lmt_base;
rc = cpt_hardware_caps_get(dev, roc_cpt);
if (rc) {
plt_err("Could not determine hardware capabilities");
goto fail;
}
rc = cpt_available_lfs_get(&cpt->dev, &nb_lf_avail);
if (rc) {
plt_err("Could not get available lfs");
goto fail;
}
/* Reserve 1 CPT LF for inline inbound */
nb_lf_avail = PLT_MIN(nb_lf_avail, (uint16_t)(ROC_CPT_MAX_LFS - 1));
roc_cpt->nb_lf_avail = nb_lf_avail;
dev->roc_cpt = roc_cpt;
/* Set it to idev if not already present */
if (!roc_idev_cpt_get())
roc_idev_cpt_set(roc_cpt);
return 0;
fail:
dev_fini(dev, pci_dev);
return rc;
}
int
roc_cpt_lf_ctx_flush(struct roc_cpt_lf *lf, void *cptr, bool inval)
{
union cpt_lf_ctx_flush reg;
if (lf == NULL) {
plt_err("Could not trigger CTX flush");
return -ENOTSUP;
}
reg.u = 0;
reg.s.inval = inval;
reg.s.cptr = (uintptr_t)cptr >> 7;
plt_write64(reg.u, lf->rbase + CPT_LF_CTX_FLUSH);
return 0;
}
int
roc_cpt_lf_ctx_reload(struct roc_cpt_lf *lf, void *cptr)
{
union cpt_lf_ctx_reload reg;
if (lf == NULL) {
plt_err("Could not trigger CTX reload");
return -ENOTSUP;
}
reg.u = 0;
reg.s.cptr = (uintptr_t)cptr >> 7;
plt_write64(reg.u, lf->rbase + CPT_LF_CTX_RELOAD);
return 0;
}
static int
cpt_lf_reset(struct roc_cpt_lf *lf)
{
struct cpt_lf_rst_req *req;
struct dev *dev = lf->dev;
req = mbox_alloc_msg_cpt_lf_reset(dev->mbox);
if (req == NULL)
return -EIO;
req->slot = lf->lf_id;
return mbox_process(dev->mbox);
}
static void
cpt_9k_lf_rst_lmtst(struct roc_cpt_lf *lf, uint8_t egrp)
{
struct cpt_inst_s inst;
uint64_t lmt_status;
memset(&inst, 0, sizeof(struct cpt_inst_s));
inst.w7.s.egrp = egrp;
plt_io_wmb();
do {
/* Copy CPT command to LMTLINE */
roc_lmt_mov64((void *)lf->lmt_base, &inst);
lmt_status = roc_lmt_submit_ldeor(lf->io_addr);
} while (lmt_status == 0);
}
static void
cpt_10k_lf_rst_lmtst(struct roc_cpt_lf *lf, uint8_t egrp)
{
uint64_t lmt_base, lmt_arg, io_addr;
struct cpt_inst_s *inst;
uint16_t lmt_id;
lmt_base = lf->lmt_base;
io_addr = lf->io_addr;
io_addr |= ROC_CN10K_CPT_INST_DW_M1 << 4;
ROC_LMT_BASE_ID_GET(lmt_base, lmt_id);
inst = (struct cpt_inst_s *)lmt_base;
memset(inst, 0, sizeof(struct cpt_inst_s));
inst->w7.s.egrp = egrp;
lmt_arg = ROC_CN10K_CPT_LMT_ARG | (uint64_t)lmt_id;
roc_lmt_submit_steorl(lmt_arg, io_addr);
}
static void
roc_cpt_iq_reset(struct roc_cpt_lf *lf)
{
union cpt_lf_inprog lf_inprog = {.u = 0x0};
union cpt_lf_ctl lf_ctl = {.u = 0x0};
lf_inprog.u = plt_read64(lf->rbase + CPT_LF_INPROG);
if (((lf_inprog.s.gwb_cnt & 0x1) == 0x1) &&
(lf_inprog.s.grb_partial == 0x0)) {
lf_inprog.s.grp_drp = 1;
plt_write64(lf_inprog.u, lf->rbase + CPT_LF_INPROG);
lf_ctl.u = plt_read64(lf->rbase + CPT_LF_CTL);
lf_ctl.s.ena = 1;
plt_write64(lf_ctl.u, lf->rbase + CPT_LF_CTL);
if (roc_model_is_cn10k())
cpt_10k_lf_rst_lmtst(lf, ROC_CPT_DFLT_ENG_GRP_SE);
else
cpt_9k_lf_rst_lmtst(lf, ROC_CPT_DFLT_ENG_GRP_SE);
plt_read64(lf->rbase + CPT_LF_INPROG);
plt_delay_us(2);
}
if (cpt_lf_reset(lf))
plt_err("Invalid CPT LF to reset");
}
void
cpt_lf_fini(struct roc_cpt_lf *lf)
{
/* Unregister IRQ's */
cpt_lf_unregister_irqs(lf);
/* Disable IQ */
roc_cpt_iq_disable(lf);
roc_cpt_iq_reset(lf);
/* Free memory */
plt_free(lf->iq_vaddr);
lf->iq_vaddr = NULL;
}
void
roc_cpt_lf_fini(struct roc_cpt_lf *lf)
{
if (lf == NULL)
return;
lf->roc_cpt->lf[lf->lf_id] = NULL;
cpt_lf_fini(lf);
}
int
roc_cpt_dev_fini(struct roc_cpt *roc_cpt)
{
struct cpt *cpt = roc_cpt_to_cpt_priv(roc_cpt);
if (cpt == NULL)
return -EINVAL;
/* Remove idev references */
if (roc_idev_cpt_get() == roc_cpt)
roc_idev_cpt_set(NULL);
roc_cpt->nb_lf_avail = 0;
roc_cpt->lmt_base = 0;
return dev_fini(&cpt->dev, cpt->pci_dev);
}
void
roc_cpt_dev_clear(struct roc_cpt *roc_cpt)
{
struct cpt *cpt = roc_cpt_to_cpt_priv(roc_cpt);
int i;
if (cpt == NULL)
return;
for (i = 0; i < roc_cpt->nb_lf; i++)
cpt->lf_msix_off[i] = 0;
roc_cpt->nb_lf = 0;
cpt_lfs_free(&cpt->dev);
cpt_lfs_detach(&cpt->dev);
}
int
roc_cpt_eng_grp_add(struct roc_cpt *roc_cpt, enum cpt_eng_type eng_type)
{
struct cpt *cpt = roc_cpt_to_cpt_priv(roc_cpt);
struct dev *dev = &cpt->dev;
struct cpt_eng_grp_req *req;
struct cpt_eng_grp_rsp *rsp;
int ret;
req = mbox_alloc_msg_cpt_eng_grp_get(dev->mbox);
if (req == NULL)
return -EIO;
switch (eng_type) {
case CPT_ENG_TYPE_AE:
case CPT_ENG_TYPE_SE:
case CPT_ENG_TYPE_IE:
break;
default:
return -EINVAL;
}
req->eng_type = eng_type;
ret = mbox_process_msg(dev->mbox, (void *)&rsp);
if (ret)
return -EIO;
if (rsp->eng_grp_num > 8) {
plt_err("Invalid CPT engine group");
return -ENOTSUP;
}
roc_cpt->eng_grp[eng_type] = rsp->eng_grp_num;
return rsp->eng_grp_num;
}
void
roc_cpt_iq_disable(struct roc_cpt_lf *lf)
{
volatile union cpt_lf_q_grp_ptr grp_ptr = {.u = 0x0};
volatile union cpt_lf_inprog lf_inprog = {.u = 0x0};
union cpt_lf_ctl lf_ctl = {.u = 0x0};
int timeout = 20;
int cnt;
/* Disable instructions enqueuing */
plt_write64(lf_ctl.u, lf->rbase + CPT_LF_CTL);
/* Wait for instruction queue to become empty */
do {
lf_inprog.u = plt_read64(lf->rbase + CPT_LF_INPROG);
if (!lf_inprog.s.inflight)
break;
plt_delay_ms(20);
if (timeout-- < 0) {
plt_err("CPT LF %d is still busy", lf->lf_id);
break;
}
} while (1);
/* Disable executions in the LF's queue.
* The queue should be empty at this point
*/
lf_inprog.s.eena = 0x0;
plt_write64(lf_inprog.u, lf->rbase + CPT_LF_INPROG);
/* Wait for instruction queue to become empty */
cnt = 0;
do {
lf_inprog.u = plt_read64(lf->rbase + CPT_LF_INPROG);
if (lf_inprog.s.grb_partial)
cnt = 0;
else
cnt++;
grp_ptr.u = plt_read64(lf->rbase + CPT_LF_Q_GRP_PTR);
} while ((cnt < 10) && (grp_ptr.s.nq_ptr != grp_ptr.s.dq_ptr));
cnt = 0;
do {
lf_inprog.u = plt_read64(lf->rbase + CPT_LF_INPROG);
if ((lf_inprog.s.inflight == 0) && (lf_inprog.s.gwb_cnt < 40) &&
((lf_inprog.s.grb_cnt == 0) || (lf_inprog.s.grb_cnt == 40)))
cnt++;
else
cnt = 0;
} while (cnt < 10);
}
void
roc_cpt_iq_enable(struct roc_cpt_lf *lf)
{
union cpt_lf_inprog lf_inprog;
union cpt_lf_ctl lf_ctl;
/* Disable command queue */
roc_cpt_iq_disable(lf);
/* Enable instruction queue enqueuing */
lf_ctl.u = plt_read64(lf->rbase + CPT_LF_CTL);
lf_ctl.s.ena = 1;
lf_ctl.s.fc_ena = 1;
lf_ctl.s.fc_up_crossing = 0;
lf_ctl.s.fc_hyst_bits = plt_log2_u32(CPT_LF_FC_MIN_THRESHOLD);
plt_write64(lf_ctl.u, lf->rbase + CPT_LF_CTL);
/* Enable command queue execution */
lf_inprog.u = plt_read64(lf->rbase + CPT_LF_INPROG);
lf_inprog.s.eena = 1;
plt_write64(lf_inprog.u, lf->rbase + CPT_LF_INPROG);
cpt_lf_dump(lf);
}
int
roc_cpt_lmtline_init(struct roc_cpt *roc_cpt, struct roc_cpt_lmtline *lmtline,
int lf_id)
{
struct roc_cpt_lf *lf;
lf = roc_cpt->lf[lf_id];
if (lf == NULL)
return -ENOTSUP;
lmtline->io_addr = lf->io_addr;
if (roc_model_is_cn10k())
lmtline->io_addr |= ROC_CN10K_CPT_INST_DW_M1 << 4;
lmtline->fc_addr = lf->fc_addr;
lmtline->lmt_base = lf->lmt_base;
lmtline->fc_thresh = lf->nb_desc - CPT_LF_FC_MIN_THRESHOLD;
return 0;
}
int
roc_cpt_ctx_write(struct roc_cpt_lf *lf, void *sa_dptr, void *sa_cptr,
uint16_t sa_len)
{
uintptr_t lmt_base = lf->lmt_base;
union cpt_res_s res, *hw_res;
uint64_t lmt_arg, io_addr;
struct cpt_inst_s *inst;
uint16_t lmt_id;
uint64_t *dptr;
int i;
ROC_LMT_CPT_BASE_ID_GET(lmt_base, lmt_id);
inst = (struct cpt_inst_s *)lmt_base;
memset(inst, 0, sizeof(struct cpt_inst_s));
hw_res = plt_zmalloc(sizeof(*hw_res), ROC_CPT_RES_ALIGN);
if (hw_res == NULL) {
plt_err("Couldn't allocate memory for result address");
return -ENOMEM;
}
dptr = plt_zmalloc(sa_len, 8);
if (dptr == NULL) {
plt_err("Couldn't allocate memory for SA dptr");
plt_free(hw_res);
return -ENOMEM;
}
for (i = 0; i < (sa_len / 8); i++)
dptr[i] = plt_cpu_to_be_64(((uint64_t *)sa_dptr)[i]);
/* Fill CPT_INST_S for WRITE_SA microcode op */
hw_res->cn10k.compcode = CPT_COMP_NOT_DONE;
inst->res_addr = (uint64_t)hw_res;
inst->dptr = (uint64_t)dptr;
inst->w4.s.param2 = sa_len >> 3;
inst->w4.s.dlen = sa_len;
inst->w4.s.opcode_major = ROC_IE_OT_MAJOR_OP_WRITE_SA;
inst->w4.s.opcode_minor = ROC_IE_OT_MINOR_OP_WRITE_SA;
inst->w7.s.cptr = (uint64_t)sa_cptr;
inst->w7.s.ctx_val = 1;
inst->w7.s.egrp = ROC_CPT_DFLT_ENG_GRP_SE_IE;
lmt_arg = ROC_CN10K_CPT_LMT_ARG | (uint64_t)lmt_id;
io_addr = lf->io_addr | ROC_CN10K_CPT_INST_DW_M1 << 4;
roc_lmt_submit_steorl(lmt_arg, io_addr);
plt_io_wmb();
/* Use 1 min timeout for the poll */
const uint64_t timeout = plt_tsc_cycles() + 60 * plt_tsc_hz();
/* Wait until CPT instruction completes */
do {
res.u64[0] = __atomic_load_n(&hw_res->u64[0], __ATOMIC_RELAXED);
if (unlikely(plt_tsc_cycles() > timeout))
break;
} while (res.cn10k.compcode == CPT_COMP_NOT_DONE);
plt_free(dptr);
plt_free(hw_res);
if (res.cn10k.compcode != CPT_COMP_WARN) {
plt_err("Write SA operation timed out");
return -ETIMEDOUT;
}
return 0;
}