.sqe_flush = otx2_sqe_flush,
        .aura_freeptr = otx2_aura_freeptr,
        .refill_pool_ptrs = otx2_refill_pool_ptrs,
+       .pfaf_mbox_intr_handler = otx2_pfaf_mbox_intr_handler,
 };
 
 static struct dev_hw_ops cn10k_hw_ops = {
        .sqe_flush = cn10k_sqe_flush,
        .aura_freeptr = cn10k_aura_freeptr,
        .refill_pool_ptrs = cn10k_refill_pool_ptrs,
+       .pfaf_mbox_intr_handler = otx2_pfaf_mbox_intr_handler,
 };
 
+void otx2_init_hw_ops(struct otx2_nic *pfvf)
+{
+       if (!test_bit(CN10K_LMTST, &pfvf->hw.cap_flag)) {
+               pfvf->hw_ops = &otx2_hw_ops;
+               return;
+       }
+
+       pfvf->hw_ops = &cn10k_hw_ops;
+}
+EXPORT_SYMBOL(otx2_init_hw_ops);
+
 int cn10k_lmtst_init(struct otx2_nic *pfvf)
 {
 
        struct otx2_lmt_info *lmt_info;
        int err, cpu;
 
-       if (!test_bit(CN10K_LMTST, &pfvf->hw.cap_flag)) {
-               pfvf->hw_ops = &otx2_hw_ops;
+       if (!test_bit(CN10K_LMTST, &pfvf->hw.cap_flag))
                return 0;
-       }
 
-       pfvf->hw_ops = &cn10k_hw_ops;
        /* Total LMTLINES = num_online_cpus() * 32 (For Burst flush).*/
        pfvf->tot_lmt_lines = (num_online_cpus() * LMT_BURST_SIZE);
        pfvf->hw.lmt_info = alloc_percpu(struct otx2_lmt_info);
 
--- /dev/null
+// SPDX-License-Identifier: GPL-2.0
+/* Marvell RVU Ethernet driver
+ *
+ * Copyright (C) 2024 Marvell.
+ *
+ */
+
+#include "otx2_common.h"
+#include "otx2_reg.h"
+#include "otx2_struct.h"
+#include "cn10k.h"
+
+static struct dev_hw_ops cn20k_hw_ops = {
+       .pfaf_mbox_intr_handler = cn20k_pfaf_mbox_intr_handler,
+};
+
+void cn20k_init(struct otx2_nic *pfvf)
+{
+       pfvf->hw_ops = &cn20k_hw_ops;
+}
+EXPORT_SYMBOL(cn20k_init);
+/* CN20K mbox AF => PFx irq handler */
+irqreturn_t cn20k_pfaf_mbox_intr_handler(int irq, void *pf_irq)
+{
+       struct otx2_nic *pf = pf_irq;
+       struct mbox *mw = &pf->mbox;
+       struct otx2_mbox_dev *mdev;
+       struct otx2_mbox *mbox;
+       struct mbox_hdr *hdr;
+       u64 pf_trig_val;
+
+       pf_trig_val = otx2_read64(pf, RVU_PF_INT) & 0x3ULL;
+
+       /* Clear the IRQ */
+       otx2_write64(pf, RVU_PF_INT, pf_trig_val);
+
+       if (pf_trig_val & BIT_ULL(0)) {
+               mbox = &mw->mbox_up;
+               mdev = &mbox->dev[0];
+               otx2_sync_mbox_bbuf(mbox, 0);
+
+               hdr = (struct mbox_hdr *)(mdev->mbase + mbox->rx_start);
+               if (hdr->num_msgs)
+                       queue_work(pf->mbox_wq, &mw->mbox_up_wrk);
+
+               trace_otx2_msg_interrupt(pf->pdev, "UP message from AF to PF",
+                                        BIT_ULL(0));
+       }
+
+       if (pf_trig_val & BIT_ULL(1)) {
+               mbox = &mw->mbox;
+               mdev = &mbox->dev[0];
+               otx2_sync_mbox_bbuf(mbox, 0);
+
+               hdr = (struct mbox_hdr *)(mdev->mbase + mbox->rx_start);
+               if (hdr->num_msgs)
+                       queue_work(pf->mbox_wq, &mw->mbox_wrk);
+               trace_otx2_msg_interrupt(pf->pdev, "DOWN reply from AF to PF",
+                                        BIT_ULL(1));
+       }
+
+       return IRQ_HANDLED;
+}
 
        otx2_mbox_msg_send(mbox, 0);
 }
 
-static irqreturn_t otx2_pfaf_mbox_intr_handler(int irq, void *pf_irq)
+irqreturn_t otx2_pfaf_mbox_intr_handler(int irq, void *pf_irq)
 {
        struct otx2_nic *pf = (struct otx2_nic *)pf_irq;
        struct mbox *mw = &pf->mbox;
 
 void otx2_disable_mbox_intr(struct otx2_nic *pf)
 {
-       int vector = pci_irq_vector(pf->pdev, RVU_PF_INT_VEC_AFPF_MBOX);
+       int vector;
 
        /* Disable AF => PF mailbox IRQ */
-       otx2_write64(pf, RVU_PF_INT_ENA_W1C, BIT_ULL(0));
+       if (!is_cn20k(pf->pdev)) {
+               vector = pci_irq_vector(pf->pdev, RVU_PF_INT_VEC_AFPF_MBOX);
+               otx2_write64(pf, RVU_PF_INT_ENA_W1C, BIT_ULL(0));
+       } else {
+               vector = pci_irq_vector(pf->pdev,
+                                       RVU_MBOX_PF_INT_VEC_AFPF_MBOX);
+               otx2_write64(pf, RVU_PF_INT_ENA_W1C,
+                            BIT_ULL(0) | BIT_ULL(1));
+       }
        free_irq(vector, pf);
 }
 EXPORT_SYMBOL(otx2_disable_mbox_intr);
        int err;
 
        /* Register mailbox interrupt handler */
-       irq_name = &hw->irq_name[RVU_PF_INT_VEC_AFPF_MBOX * NAME_SIZE];
-       snprintf(irq_name, NAME_SIZE, "RVUPFAF Mbox");
-       err = request_irq(pci_irq_vector(pf->pdev, RVU_PF_INT_VEC_AFPF_MBOX),
-                         otx2_pfaf_mbox_intr_handler, 0, irq_name, pf);
+       if (!is_cn20k(pf->pdev)) {
+               irq_name = &hw->irq_name[RVU_PF_INT_VEC_AFPF_MBOX * NAME_SIZE];
+               snprintf(irq_name, NAME_SIZE, "RVUPF%d AFPF Mbox",
+                        rvu_get_pf(pf->pdev, pf->pcifunc));
+               err = request_irq(pci_irq_vector
+                                 (pf->pdev, RVU_PF_INT_VEC_AFPF_MBOX),
+                                 pf->hw_ops->pfaf_mbox_intr_handler,
+                                 0, irq_name, pf);
+       } else {
+               irq_name = &hw->irq_name[RVU_MBOX_PF_INT_VEC_AFPF_MBOX *
+                                               NAME_SIZE];
+               snprintf(irq_name, NAME_SIZE, "RVUPF%d AFPF Mbox",
+                        rvu_get_pf(pf->pdev, pf->pcifunc));
+               err = request_irq(pci_irq_vector
+                                 (pf->pdev, RVU_MBOX_PF_INT_VEC_AFPF_MBOX),
+                                 pf->hw_ops->pfaf_mbox_intr_handler,
+                                 0, irq_name, pf);
+       }
        if (err) {
                dev_err(pf->dev,
                        "RVUPF: IRQ registration failed for PFAF mbox irq\n");
        /* Enable mailbox interrupt for msgs coming from AF.
         * First clear to avoid spurious interrupts, if any.
         */
-       otx2_write64(pf, RVU_PF_INT, BIT_ULL(0));
-       otx2_write64(pf, RVU_PF_INT_ENA_W1S, BIT_ULL(0));
+       if (!is_cn20k(pf->pdev)) {
+               otx2_write64(pf, RVU_PF_INT, BIT_ULL(0));
+               otx2_write64(pf, RVU_PF_INT_ENA_W1S, BIT_ULL(0));
+       } else {
+               otx2_write64(pf, RVU_PF_INT, BIT_ULL(0) | BIT_ULL(1));
+               otx2_write64(pf, RVU_PF_INT_ENA_W1S, BIT_ULL(0) |
+                            BIT_ULL(1));
+       }
 
        if (!probe_af)
                return 0;
                pf->mbox_wq = NULL;
        }
 
-       if (mbox->mbox.hwbase)
+       if (mbox->mbox.hwbase && !is_cn20k(pf->pdev))
                iounmap((void __iomem *)mbox->mbox.hwbase);
 
        otx2_mbox_destroy(&mbox->mbox);
        if (!pf->mbox_wq)
                return -ENOMEM;
 
-       /* Mailbox is a reserved memory (in RAM) region shared between
-        * admin function (i.e AF) and this PF, shouldn't be mapped as
-        * device memory to allow unaligned accesses.
+       /* For CN20K, AF allocates mbox memory in DRAM and writes PF
+        * regions/offsets in RVU_MBOX_AF_PFX_ADDR, the RVU_PFX_FUNC_PFAF_MBOX
+        * gives the aliased address to access AF/PF mailbox regions.
         */
-       hwbase = ioremap_wc(pci_resource_start(pf->pdev, PCI_MBOX_BAR_NUM),
-                           MBOX_SIZE);
+       if (is_cn20k(pf->pdev))
+               hwbase = pf->reg_base + RVU_PFX_FUNC_PFAF_MBOX +
+                       ((u64)BLKADDR_MBOX << RVU_FUNC_BLKADDR_SHIFT);
+       else
+               /* Mailbox is a reserved memory (in RAM) region shared between
+                * admin function (i.e AF) and this PF, shouldn't be mapped as
+                * device memory to allow unaligned accesses.
+                */
+               hwbase = ioremap_wc(pci_resource_start
+                                   (pf->pdev, PCI_MBOX_BAR_NUM), MBOX_SIZE);
        if (!hwbase) {
                dev_err(pf->dev, "Unable to map PFAF mailbox region\n");
                err = -ENOMEM;
        if (err)
                return err;
 
-       err = pci_alloc_irq_vectors(hw->pdev, RVU_PF_INT_VEC_CNT,
-                                   RVU_PF_INT_VEC_CNT, PCI_IRQ_MSIX);
+       if (!is_cn20k(pf->pdev))
+               err = pci_alloc_irq_vectors(hw->pdev, RVU_PF_INT_VEC_CNT,
+                                           RVU_PF_INT_VEC_CNT, PCI_IRQ_MSIX);
+       else
+               err = pci_alloc_irq_vectors(hw->pdev, RVU_MBOX_PF_INT_VEC_CNT,
+                                           RVU_MBOX_PF_INT_VEC_CNT,
+                                           PCI_IRQ_MSIX);
        if (err < 0) {
                dev_err(dev, "%s: Failed to alloc %d IRQ vectors\n",
                        __func__, num_vec);
 
        otx2_setup_dev_hw_settings(pf);
 
+       if (is_cn20k(pf->pdev))
+               cn20k_init(pf);
+       else
+               otx2_init_hw_ops(pf);
+
        /* Init PF <=> AF mailbox stuff */
        err = otx2_pfaf_mbox_init(pf);
        if (err)