Re: [net-next PATCH 2/6] octeontx2-af: CN20k basic mbox operations and structures

From: Kalesh Anakkur Purayil
Date: Sun Oct 20 2024 - 23:31:20 EST


On Sat, Oct 19, 2024 at 2:02 AM Sai Krishna <saikrishnag@xxxxxxxxxxx> wrote:
>
> This patch adds basic mbox operation APIs and structures to add support
> for mbox module on CN20k silicon. There are few CSR offsets, interrupts
> changed between CN20k and prior Octeon series of devices.
>
> Signed-off-by: Sunil Kovvuri Goutham <sgoutham@xxxxxxxxxxx>
> Signed-off-by: Sai Krishna <saikrishnag@xxxxxxxxxxx>
> ---
> .../ethernet/marvell/octeontx2/af/Makefile | 3 +-
> .../ethernet/marvell/octeontx2/af/cn20k/api.h | 22 +++++++
> .../marvell/octeontx2/af/cn20k/mbox_init.c | 52 +++++++++++++++
> .../ethernet/marvell/octeontx2/af/cn20k/reg.h | 27 ++++++++
> .../net/ethernet/marvell/octeontx2/af/mbox.c | 3 +
> .../net/ethernet/marvell/octeontx2/af/mbox.h | 7 ++
> .../net/ethernet/marvell/octeontx2/af/rvu.c | 65 +++++++++++++++----
> .../net/ethernet/marvell/octeontx2/af/rvu.h | 22 +++++++
> .../marvell/octeontx2/af/rvu_struct.h | 6 +-
> 9 files changed, 192 insertions(+), 15 deletions(-)
> create mode 100644 drivers/net/ethernet/marvell/octeontx2/af/cn20k/api.h
> create mode 100644 drivers/net/ethernet/marvell/octeontx2/af/cn20k/mbox_init.c
> create mode 100644 drivers/net/ethernet/marvell/octeontx2/af/cn20k/reg.h
>
> diff --git a/drivers/net/ethernet/marvell/octeontx2/af/Makefile b/drivers/net/ethernet/marvell/octeontx2/af/Makefile
> index 3cf4c8285c90..38d8599dc6eb 100644
> --- a/drivers/net/ethernet/marvell/octeontx2/af/Makefile
> +++ b/drivers/net/ethernet/marvell/octeontx2/af/Makefile
> @@ -11,4 +11,5 @@ rvu_mbox-y := mbox.o rvu_trace.o
> rvu_af-y := cgx.o rvu.o rvu_cgx.o rvu_npa.o rvu_nix.o \
> rvu_reg.o rvu_npc.o rvu_debugfs.o ptp.o rvu_npc_fs.o \
> rvu_cpt.o rvu_devlink.o rpm.o rvu_cn10k.o rvu_switch.o \
> - rvu_sdp.o rvu_npc_hash.o mcs.o mcs_rvu_if.o mcs_cnf10kb.o
> + rvu_sdp.o rvu_npc_hash.o mcs.o mcs_rvu_if.o mcs_cnf10kb.o \
> + cn20k/mbox_init.o
> diff --git a/drivers/net/ethernet/marvell/octeontx2/af/cn20k/api.h b/drivers/net/ethernet/marvell/octeontx2/af/cn20k/api.h
> new file mode 100644
> index 000000000000..b57bd38181aa
> --- /dev/null
> +++ b/drivers/net/ethernet/marvell/octeontx2/af/cn20k/api.h
> @@ -0,0 +1,22 @@
> +/* SPDX-License-Identifier: GPL-2.0 */
> +/* Marvell RVU Admin Function driver
> + *
> + * Copyright (C) 2024 Marvell.
> + *
> + */
> +
> +#ifndef CN20K_API_H
> +#define CN20K_API_H
> +
> +#include "../rvu.h"
> +
> +struct ng_rvu {
> + struct mbox_ops *rvu_mbox_ops;
> + struct qmem *pf_mbox_addr;
> +};
> +
> +/* Mbox related APIs */
> +int cn20k_rvu_mbox_init(struct rvu *rvu, int type, int num);
> +int cn20k_rvu_get_mbox_regions(struct rvu *rvu, void **mbox_addr,
> + int num, int type, unsigned long *pf_bmap);
> +#endif /* CN20K_API_H */
> diff --git a/drivers/net/ethernet/marvell/octeontx2/af/cn20k/mbox_init.c b/drivers/net/ethernet/marvell/octeontx2/af/cn20k/mbox_init.c
> new file mode 100644
> index 000000000000..0d7ad31e5dfb
> --- /dev/null
> +++ b/drivers/net/ethernet/marvell/octeontx2/af/cn20k/mbox_init.c
> @@ -0,0 +1,52 @@
> +// SPDX-License-Identifier: GPL-2.0
> +/* Marvell RVU Admin Function driver
> + *
> + * Copyright (C) 2024 Marvell.
> + *
> + */
> +
> +#include <linux/interrupt.h>
> +#include <linux/irq.h>
> +
> +#include "rvu_trace.h"
> +#include "mbox.h"
> +#include "reg.h"
> +#include "api.h"
> +
> +int cn20k_rvu_get_mbox_regions(struct rvu *rvu, void **mbox_addr,
> + int num, int type, unsigned long *pf_bmap)
> +{
> + int region;
> + u64 bar;
> +
> + for (region = 0; region < num; region++) {
> + if (!test_bit(region, pf_bmap))
> + continue;
> +
> + bar = (u64)phys_to_virt((u64)rvu->ng_rvu->pf_mbox_addr->base);
> + bar += region * MBOX_SIZE;
> +
> + mbox_addr[region] = (void *)bar;
> +
> + if (!mbox_addr[region])
> + goto error;
[Kalesh] Maybe you can return directly from here as there is no
cleanup action performed under the label.
> + }
> + return 0;
> +
> +error:
> + return -ENOMEM;
> +}
> +
> +int cn20k_rvu_mbox_init(struct rvu *rvu, int type, int ndevs)
> +{
> + int dev;
> +
> + if (!is_cn20k(rvu->pdev))
> + return 0;
> +
> + for (dev = 0; dev < ndevs; dev++)
> + rvu_write64(rvu, BLKADDR_RVUM,
> + RVU_MBOX_AF_PFX_CFG(dev), ilog2(MBOX_SIZE));
> +
> + return 0;
> +}
> diff --git a/drivers/net/ethernet/marvell/octeontx2/af/cn20k/reg.h b/drivers/net/ethernet/marvell/octeontx2/af/cn20k/reg.h
> new file mode 100644
> index 000000000000..58152a4024ec
> --- /dev/null
> +++ b/drivers/net/ethernet/marvell/octeontx2/af/cn20k/reg.h
> @@ -0,0 +1,27 @@
> +/* SPDX-License-Identifier: GPL-2.0 */
> +/* Marvell RVU Admin Function driver
> + *
> + * Copyright (C) 2024 Marvell.
> + *
> + */
> +
> +#ifndef RVU_MBOX_REG_H
> +#define RVU_MBOX_REG_H
> +#include "../rvu.h"
> +#include "../rvu_reg.h"
> +
> +/* RVUM block registers */
> +#define RVU_PF_DISC (0x0)
> +#define RVU_PRIV_PFX_DISC(a) (0x8000208 | (a) << 16)
> +#define RVU_PRIV_HWVFX_DISC(a) (0xD000000 | (a) << 12)
> +
> +/* Mbox Registers */
> +/* RVU AF BAR0 Mbox registers for AF => PFx */
> +#define RVU_MBOX_AF_PFX_ADDR(a) (0x5000 | (a) << 4)
> +#define RVU_MBOX_AF_PFX_CFG(a) (0x6000 | (a) << 4)
> +#define RVU_AF_BAR2_SEL (0x9000000)
> +#define RVU_AF_BAR2_PFID (0x16400)
> +#define NIX_CINTX_INT_W1S(a) (0xd30 | (a) << 12)
> +#define NIX_QINTX_CNT(a) (0xc00 | (a) << 12)
> +
> +#endif /* RVU_MBOX_REG_H */
> diff --git a/drivers/net/ethernet/marvell/octeontx2/af/mbox.c b/drivers/net/ethernet/marvell/octeontx2/af/mbox.c
> index 791c468a10c5..1e3e72107a9d 100644
> --- a/drivers/net/ethernet/marvell/octeontx2/af/mbox.c
> +++ b/drivers/net/ethernet/marvell/octeontx2/af/mbox.c
> @@ -10,8 +10,11 @@
> #include <linux/pci.h>
>
> #include "rvu_reg.h"
> +#include "cn20k/reg.h"
> +#include "cn20k/api.h"
> #include "mbox.h"
> #include "rvu_trace.h"
> +#include "rvu.h"
>
> /* Default values of PF and VF bit encodings in PCIFUNC for
> * CN9XXX and CN10K series silicons.
> diff --git a/drivers/net/ethernet/marvell/octeontx2/af/mbox.h b/drivers/net/ethernet/marvell/octeontx2/af/mbox.h
> index 38a0badcdb68..df64a18fe1d6 100644
> --- a/drivers/net/ethernet/marvell/octeontx2/af/mbox.h
> +++ b/drivers/net/ethernet/marvell/octeontx2/af/mbox.h
> @@ -55,6 +55,11 @@ extern u16 rvu_pcifunc_pf_mask;
> extern u16 rvu_pcifunc_func_shift;
> extern u16 rvu_pcifunc_func_mask;
>
> +enum {
> + TYPE_AFVF,
> + TYPE_AFPF,
> +};
> +
> struct otx2_mbox_dev {
> void *mbase; /* This dev's mbox region */
> void *hwbase;
> @@ -83,6 +88,8 @@ struct otx2_mbox {
> struct mbox_hdr {
> u64 msg_size; /* Total msgs size embedded */
> u16 num_msgs; /* No of msgs embedded */
> + u16 opt_msg;
> + u8 sig;
> };
>
> /* Header which precedes every msg and is also part of it */
> diff --git a/drivers/net/ethernet/marvell/octeontx2/af/rvu.c b/drivers/net/ethernet/marvell/octeontx2/af/rvu.c
> index dcfc27a60b43..a5ebd7cd3a5c 100644
> --- a/drivers/net/ethernet/marvell/octeontx2/af/rvu.c
> +++ b/drivers/net/ethernet/marvell/octeontx2/af/rvu.c
> @@ -20,6 +20,8 @@
>
> #include "rvu_trace.h"
> #include "rvu_npc_hash.h"
> +#include "cn20k/reg.h"
> +#include "cn20k/api.h"
>
> #define DRV_NAME "rvu_af"
> #define DRV_STRING "Marvell OcteonTX2 RVU Admin Function Driver"
> @@ -34,10 +36,8 @@ static int rvu_mbox_init(struct rvu *rvu, struct mbox_wq_info *mw,
> int type, int num,
> void (mbox_handler)(struct work_struct *),
> void (mbox_up_handler)(struct work_struct *));
> -enum {
> - TYPE_AFVF,
> - TYPE_AFPF,
> -};
> +static irqreturn_t rvu_mbox_pf_intr_handler(int irq, void *rvu_irq);
> +static irqreturn_t rvu_mbox_intr_handler(int irq, void *rvu_irq);
>
> /* Supported devices */
> static const struct pci_device_id rvu_id_table[] = {
> @@ -2212,6 +2212,22 @@ static void __rvu_mbox_handler(struct rvu_work *mwork, int type, bool poll)
>
> offset = mbox->rx_start + ALIGN(sizeof(*req_hdr), MBOX_MSG_ALIGN);
>
> + if (req_hdr->sig) {
> + req_hdr->opt_msg = mw->mbox_wrk[devid].num_msgs;
> + rvu_write64(rvu, BLKADDR_NIX0, RVU_AF_BAR2_SEL,
> + RVU_AF_BAR2_PFID);
> + if (type == TYPE_AFPF)
> + rvu_write64(rvu, BLKADDR_NIX0,
> + AF_BAR2_ALIASX(0, NIX_CINTX_INT_W1S(devid)),
> + 0x1);
> + else
> + rvu_write64(rvu, BLKADDR_NIX0,
> + AF_BAR2_ALIASX(0, NIX_QINTX_CNT(devid)),
> + 0x1);
> + usleep_range(1000, 2000);
> + goto done;
> + }
> +
> for (id = 0; id < mw->mbox_wrk[devid].num_msgs; id++) {
> msg = mdev->mbase + offset;
>
> @@ -2245,9 +2261,10 @@ static void __rvu_mbox_handler(struct rvu_work *mwork, int type, bool poll)
> err, otx2_mbox_id2name(msg->id),
> msg->id, devid);
> }
> +done:
> mw->mbox_wrk[devid].num_msgs = 0;
>
> - if (poll)
> + if (!is_cn20k(mbox->pdev) && poll)
> otx2_mbox_wait_for_zero(mbox, devid);
>
> /* Send mbox responses to VF/PF */
> @@ -2360,6 +2377,10 @@ static int rvu_get_mbox_regions(struct rvu *rvu, void **mbox_addr,
> int region;
> u64 bar4;
>
> + if (is_cn20k(rvu->pdev))
> + return cn20k_rvu_get_mbox_regions(rvu, mbox_addr,
> + num, type, pf_bmap);
> +
> /* For cn10k platform VF mailbox regions of a PF follows after the
> * PF <-> AF mailbox region. Whereas for Octeontx2 it is read from
> * RVU_PF_VF_BAR4_ADDR register.
> @@ -2413,12 +2434,17 @@ static int rvu_get_mbox_regions(struct rvu *rvu, void **mbox_addr,
> return -ENOMEM;
> }
>
> +static struct mbox_ops rvu_mbox_ops = {
> + .pf_intr_handler = rvu_mbox_pf_intr_handler,
> +};
> +
> static int rvu_mbox_init(struct rvu *rvu, struct mbox_wq_info *mw,
> int type, int num,
> void (mbox_handler)(struct work_struct *),
> void (mbox_up_handler)(struct work_struct *))
> {
> int err = -EINVAL, i, dir, dir_up;
> + struct ng_rvu *ng_rvu_mbox;
> void __iomem *reg_base;
> struct rvu_work *mwork;
> unsigned long *pf_bmap;
> @@ -2443,6 +2469,18 @@ static int rvu_mbox_init(struct rvu *rvu, struct mbox_wq_info *mw,
> }
> }
>
> + ng_rvu_mbox = kzalloc(sizeof(*ng_rvu_mbox), GFP_KERNEL);
> + if (!ng_rvu_mbox) {
> + err = -ENOMEM;
> + goto free_bitmap;
> + }
> +
> + rvu->ng_rvu = ng_rvu_mbox;
> +
> + rvu->ng_rvu->rvu_mbox_ops = &rvu_mbox_ops;
> +
> + cn20k_rvu_mbox_init(rvu, type, num);
> +
> mutex_init(&rvu->mbox_lock);
>
> mbox_regions = kcalloc(num, sizeof(void *), GFP_KERNEL);
> @@ -2475,7 +2513,7 @@ static int rvu_mbox_init(struct rvu *rvu, struct mbox_wq_info *mw,
> }
>
> mw->mbox_wq = alloc_workqueue("%s",
> - WQ_UNBOUND | WQ_HIGHPRI | WQ_MEM_RECLAIM,
> + WQ_HIGHPRI | WQ_MEM_RECLAIM,
> num, name);
> if (!mw->mbox_wq) {
> err = -ENOMEM;
> @@ -2553,8 +2591,8 @@ static void rvu_mbox_destroy(struct mbox_wq_info *mw)
> otx2_mbox_destroy(&mw->mbox_up);
> }
>
> -static void rvu_queue_work(struct mbox_wq_info *mw, int first,
> - int mdevs, u64 intr)
> +void rvu_queue_work(struct mbox_wq_info *mw, int first,
> + int mdevs, u64 intr)
> {
> struct otx2_mbox_dev *mdev;
> struct otx2_mbox *mbox;
> @@ -2965,12 +3003,14 @@ static int rvu_register_interrupts(struct rvu *rvu)
>
> /* Register mailbox interrupt handler */
> sprintf(&rvu->irq_name[RVU_AF_INT_VEC_MBOX * NAME_SIZE], "RVUAF Mbox");
> - ret = request_irq(pci_irq_vector(rvu->pdev, RVU_AF_INT_VEC_MBOX),
> - rvu_mbox_pf_intr_handler, 0,
> - &rvu->irq_name[RVU_AF_INT_VEC_MBOX * NAME_SIZE], rvu);
> + ret = request_irq(pci_irq_vector
> + (rvu->pdev, RVU_AF_INT_VEC_MBOX),
> + rvu->ng_rvu->rvu_mbox_ops->pf_intr_handler, 0,
> + &rvu->irq_name[RVU_AF_INT_VEC_MBOX *
> + NAME_SIZE], rvu);
> if (ret) {
> dev_err(rvu->dev,
> - "RVUAF: IRQ registration failed for mbox irq\n");
> + "RVUAF: IRQ registration failed for mbox\n");
> goto fail;
> }
>
> @@ -3478,6 +3518,7 @@ static void rvu_remove(struct pci_dev *pdev)
> pci_set_drvdata(pdev, NULL);
>
> devm_kfree(&pdev->dev, rvu->hw);
> + kfree(rvu->ng_rvu);
> devm_kfree(&pdev->dev, rvu);
> }
>
> diff --git a/drivers/net/ethernet/marvell/octeontx2/af/rvu.h b/drivers/net/ethernet/marvell/octeontx2/af/rvu.h
> index 938a911cbf1c..9fd7aea8c481 100644
> --- a/drivers/net/ethernet/marvell/octeontx2/af/rvu.h
> +++ b/drivers/net/ethernet/marvell/octeontx2/af/rvu.h
> @@ -444,6 +444,10 @@ struct mbox_wq_info {
> struct workqueue_struct *mbox_wq;
> };
>
> +struct mbox_ops {
> + irqreturn_t (*pf_intr_handler)(int irq, void *rvu_irq);
> +};
> +
> struct channel_fwdata {
> struct sdp_node_info info;
> u8 valid;
> @@ -594,6 +598,7 @@ struct rvu {
> spinlock_t cpt_intr_lock;
>
> struct mutex mbox_lock; /* Serialize mbox up and down msgs */
> + struct ng_rvu *ng_rvu;
> };
>
> static inline void rvu_write64(struct rvu *rvu, u64 block, u64 offset, u64 val)
> @@ -875,11 +880,28 @@ static inline bool is_cgx_vf(struct rvu *rvu, u16 pcifunc)
> is_pf_cgxmapped(rvu, rvu_get_pf(pcifunc)));
> }
>
> +#define CN20K_CHIPID 0x20
> +
> +/*
> + * Silicon check for CN20K family
> + */
> +static inline bool is_cn20k(struct pci_dev *pdev)
> +{
> + if ((pdev->subsystem_device & 0xFF) == CN20K_CHIPID)
> + return true;
> +
> + return false;
[Kalesh] You can simplify this as:
return (pdev->subsystem_device & 0xFF) == CN20K_CHIPID;
> +}
> +
> #define M(_name, _id, fn_name, req, rsp) \
> int rvu_mbox_handler_ ## fn_name(struct rvu *, struct req *, struct rsp *);
> MBOX_MESSAGES
> #undef M
>
> +/* Mbox APIs */
> +void rvu_queue_work(struct mbox_wq_info *mw, int first,
> + int mdevs, u64 intr);
> +
> int rvu_cgx_init(struct rvu *rvu);
> int rvu_cgx_exit(struct rvu *rvu);
> void *rvu_cgx_pdata(u8 cgx_id, struct rvu *rvu);
> diff --git a/drivers/net/ethernet/marvell/octeontx2/af/rvu_struct.h b/drivers/net/ethernet/marvell/octeontx2/af/rvu_struct.h
> index fc8da2090657..90cb063d00f0 100644
> --- a/drivers/net/ethernet/marvell/octeontx2/af/rvu_struct.h
> +++ b/drivers/net/ethernet/marvell/octeontx2/af/rvu_struct.h
> @@ -33,7 +33,8 @@ enum rvu_block_addr_e {
> BLKADDR_NDC_NIX1_RX = 0x10ULL,
> BLKADDR_NDC_NIX1_TX = 0x11ULL,
> BLKADDR_APR = 0x16ULL,
> - BLK_COUNT = 0x17ULL,
> + BLKADDR_MBOX = 0x1bULL,
> + BLK_COUNT = 0x1cULL,
> };
>
> /* RVU Block Type Enumeration */
> @@ -49,7 +50,8 @@ enum rvu_block_type_e {
> BLKTYPE_TIM = 0x8,
> BLKTYPE_CPT = 0x9,
> BLKTYPE_NDC = 0xa,
> - BLKTYPE_MAX = 0xa,
> + BLKTYPE_MBOX = 0x13,
> + BLKTYPE_MAX = 0x13,
> };
>
> /* RVU Admin function Interrupt Vector Enumeration */
> --
> 2.25.1
>
>


--
Regards,
Kalesh A P

Attachment: smime.p7s
Description: S/MIME Cryptographic Signature