/* 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; }