1cf718eaaSSrikanth, Jampala // SPDX-License-Identifier: GPL-2.0 2cf718eaaSSrikanth, Jampala #include <linux/workqueue.h> 3cf718eaaSSrikanth, Jampala 4cf718eaaSSrikanth, Jampala #include "nitrox_csr.h" 5cf718eaaSSrikanth, Jampala #include "nitrox_hal.h" 6cf718eaaSSrikanth, Jampala #include "nitrox_dev.h" 7cd078cb6SHerbert Xu #include "nitrox_mbx.h" 8cf718eaaSSrikanth, Jampala 9cf718eaaSSrikanth, Jampala #define RING_TO_VFNO(_x, _y) ((_x) / (_y)) 10cf718eaaSSrikanth, Jampala 11*c4d7d318SLee Jones /* 12cf718eaaSSrikanth, Jampala * mbx_msg_type - Mailbox message types 13cf718eaaSSrikanth, Jampala */ 14cf718eaaSSrikanth, Jampala enum mbx_msg_type { 15cf718eaaSSrikanth, Jampala MBX_MSG_TYPE_NOP, 16cf718eaaSSrikanth, Jampala MBX_MSG_TYPE_REQ, 17cf718eaaSSrikanth, Jampala MBX_MSG_TYPE_ACK, 18cf718eaaSSrikanth, Jampala MBX_MSG_TYPE_NACK, 19cf718eaaSSrikanth, Jampala }; 20cf718eaaSSrikanth, Jampala 21*c4d7d318SLee Jones /* 22cf718eaaSSrikanth, Jampala * mbx_msg_opcode - Mailbox message opcodes 23cf718eaaSSrikanth, Jampala */ 24cf718eaaSSrikanth, Jampala enum mbx_msg_opcode { 25cf718eaaSSrikanth, Jampala MSG_OP_VF_MODE = 1, 26cf718eaaSSrikanth, Jampala MSG_OP_VF_UP, 27cf718eaaSSrikanth, Jampala MSG_OP_VF_DOWN, 28cf718eaaSSrikanth, Jampala MSG_OP_CHIPID_VFID, 299e5de3e0SNagadheeraj Rottela MSG_OP_MCODE_INFO = 11, 30cf718eaaSSrikanth, Jampala }; 31cf718eaaSSrikanth, Jampala 32cf718eaaSSrikanth, Jampala struct pf2vf_work { 33cf718eaaSSrikanth, Jampala struct nitrox_vfdev *vfdev; 34cf718eaaSSrikanth, Jampala struct nitrox_device *ndev; 35cf718eaaSSrikanth, Jampala struct work_struct pf2vf_resp; 36cf718eaaSSrikanth, Jampala }; 37cf718eaaSSrikanth, Jampala 38cf718eaaSSrikanth, Jampala static inline u64 pf2vf_read_mbox(struct nitrox_device *ndev, int ring) 39cf718eaaSSrikanth, Jampala { 40cf718eaaSSrikanth, Jampala u64 reg_addr; 41cf718eaaSSrikanth, Jampala 42cf718eaaSSrikanth, Jampala reg_addr = NPS_PKT_MBOX_VF_PF_PFDATAX(ring); 43cf718eaaSSrikanth, Jampala return nitrox_read_csr(ndev, reg_addr); 44cf718eaaSSrikanth, Jampala } 45cf718eaaSSrikanth, Jampala 46cf718eaaSSrikanth, Jampala static inline void pf2vf_write_mbox(struct nitrox_device *ndev, u64 value, 47cf718eaaSSrikanth, Jampala int ring) 48cf718eaaSSrikanth, Jampala { 49cf718eaaSSrikanth, Jampala u64 reg_addr; 50cf718eaaSSrikanth, Jampala 51cf718eaaSSrikanth, Jampala reg_addr = NPS_PKT_MBOX_PF_VF_PFDATAX(ring); 52cf718eaaSSrikanth, Jampala nitrox_write_csr(ndev, reg_addr, value); 53cf718eaaSSrikanth, Jampala } 54cf718eaaSSrikanth, Jampala 55cf718eaaSSrikanth, Jampala static void pf2vf_send_response(struct nitrox_device *ndev, 56cf718eaaSSrikanth, Jampala struct nitrox_vfdev *vfdev) 57cf718eaaSSrikanth, Jampala { 58cf718eaaSSrikanth, Jampala union mbox_msg msg; 59cf718eaaSSrikanth, Jampala 60cf718eaaSSrikanth, Jampala msg.value = vfdev->msg.value; 61cf718eaaSSrikanth, Jampala 62cf718eaaSSrikanth, Jampala switch (vfdev->msg.opcode) { 63cf718eaaSSrikanth, Jampala case MSG_OP_VF_MODE: 64cf718eaaSSrikanth, Jampala msg.data = ndev->mode; 65cf718eaaSSrikanth, Jampala break; 66cf718eaaSSrikanth, Jampala case MSG_OP_VF_UP: 67cf718eaaSSrikanth, Jampala vfdev->nr_queues = vfdev->msg.data; 68cf718eaaSSrikanth, Jampala atomic_set(&vfdev->state, __NDEV_READY); 69cf718eaaSSrikanth, Jampala break; 70cf718eaaSSrikanth, Jampala case MSG_OP_CHIPID_VFID: 71cf718eaaSSrikanth, Jampala msg.id.chipid = ndev->idx; 72cf718eaaSSrikanth, Jampala msg.id.vfid = vfdev->vfno; 73cf718eaaSSrikanth, Jampala break; 74cf718eaaSSrikanth, Jampala case MSG_OP_VF_DOWN: 75cf718eaaSSrikanth, Jampala vfdev->nr_queues = 0; 76cf718eaaSSrikanth, Jampala atomic_set(&vfdev->state, __NDEV_NOT_READY); 77cf718eaaSSrikanth, Jampala break; 789e5de3e0SNagadheeraj Rottela case MSG_OP_MCODE_INFO: 799e5de3e0SNagadheeraj Rottela msg.data = 0; 809e5de3e0SNagadheeraj Rottela msg.mcode_info.count = 2; 819e5de3e0SNagadheeraj Rottela msg.mcode_info.info = MCODE_TYPE_SE_SSL | (MCODE_TYPE_AE << 5); 829e5de3e0SNagadheeraj Rottela msg.mcode_info.next_se_grp = 1; 839e5de3e0SNagadheeraj Rottela msg.mcode_info.next_ae_grp = 1; 849e5de3e0SNagadheeraj Rottela break; 85cf718eaaSSrikanth, Jampala default: 86cf718eaaSSrikanth, Jampala msg.type = MBX_MSG_TYPE_NOP; 87cf718eaaSSrikanth, Jampala break; 88cf718eaaSSrikanth, Jampala } 89cf718eaaSSrikanth, Jampala 90cf718eaaSSrikanth, Jampala if (msg.type == MBX_MSG_TYPE_NOP) 91cf718eaaSSrikanth, Jampala return; 92cf718eaaSSrikanth, Jampala 93cf718eaaSSrikanth, Jampala /* send ACK to VF */ 94cf718eaaSSrikanth, Jampala msg.type = MBX_MSG_TYPE_ACK; 95cf718eaaSSrikanth, Jampala pf2vf_write_mbox(ndev, msg.value, vfdev->ring); 96cf718eaaSSrikanth, Jampala 97cf718eaaSSrikanth, Jampala vfdev->msg.value = 0; 98cf718eaaSSrikanth, Jampala atomic64_inc(&vfdev->mbx_resp); 99cf718eaaSSrikanth, Jampala } 100cf718eaaSSrikanth, Jampala 101cf718eaaSSrikanth, Jampala static void pf2vf_resp_handler(struct work_struct *work) 102cf718eaaSSrikanth, Jampala { 103cf718eaaSSrikanth, Jampala struct pf2vf_work *pf2vf_resp = container_of(work, struct pf2vf_work, 104cf718eaaSSrikanth, Jampala pf2vf_resp); 105cf718eaaSSrikanth, Jampala struct nitrox_vfdev *vfdev = pf2vf_resp->vfdev; 106cf718eaaSSrikanth, Jampala struct nitrox_device *ndev = pf2vf_resp->ndev; 107cf718eaaSSrikanth, Jampala 108cf718eaaSSrikanth, Jampala switch (vfdev->msg.type) { 109cf718eaaSSrikanth, Jampala case MBX_MSG_TYPE_REQ: 110cf718eaaSSrikanth, Jampala /* process the request from VF */ 111cf718eaaSSrikanth, Jampala pf2vf_send_response(ndev, vfdev); 112cf718eaaSSrikanth, Jampala break; 113cf718eaaSSrikanth, Jampala case MBX_MSG_TYPE_ACK: 114cf718eaaSSrikanth, Jampala case MBX_MSG_TYPE_NACK: 115cf718eaaSSrikanth, Jampala break; 11676bb6a7bSTom Rix } 117cf718eaaSSrikanth, Jampala 118cf718eaaSSrikanth, Jampala kfree(pf2vf_resp); 119cf718eaaSSrikanth, Jampala } 120cf718eaaSSrikanth, Jampala 121cf718eaaSSrikanth, Jampala void nitrox_pf2vf_mbox_handler(struct nitrox_device *ndev) 122cf718eaaSSrikanth, Jampala { 123cf718eaaSSrikanth, Jampala struct nitrox_vfdev *vfdev; 124cf718eaaSSrikanth, Jampala struct pf2vf_work *pfwork; 125cf718eaaSSrikanth, Jampala u64 value, reg_addr; 126cf718eaaSSrikanth, Jampala u32 i; 127cf718eaaSSrikanth, Jampala int vfno; 128cf718eaaSSrikanth, Jampala 129cf718eaaSSrikanth, Jampala /* loop for VF(0..63) */ 130cf718eaaSSrikanth, Jampala reg_addr = NPS_PKT_MBOX_INT_LO; 131cf718eaaSSrikanth, Jampala value = nitrox_read_csr(ndev, reg_addr); 132cf718eaaSSrikanth, Jampala for_each_set_bit(i, (const unsigned long *)&value, BITS_PER_LONG) { 133cf718eaaSSrikanth, Jampala /* get the vfno from ring */ 134cf718eaaSSrikanth, Jampala vfno = RING_TO_VFNO(i, ndev->iov.max_vf_queues); 135cf718eaaSSrikanth, Jampala vfdev = ndev->iov.vfdev + vfno; 136cf718eaaSSrikanth, Jampala vfdev->ring = i; 137cf718eaaSSrikanth, Jampala /* fill the vf mailbox data */ 138cf718eaaSSrikanth, Jampala vfdev->msg.value = pf2vf_read_mbox(ndev, vfdev->ring); 139cf718eaaSSrikanth, Jampala pfwork = kzalloc(sizeof(*pfwork), GFP_ATOMIC); 140cf718eaaSSrikanth, Jampala if (!pfwork) 141cf718eaaSSrikanth, Jampala continue; 142cf718eaaSSrikanth, Jampala 143cf718eaaSSrikanth, Jampala pfwork->vfdev = vfdev; 144cf718eaaSSrikanth, Jampala pfwork->ndev = ndev; 145cf718eaaSSrikanth, Jampala INIT_WORK(&pfwork->pf2vf_resp, pf2vf_resp_handler); 146cf718eaaSSrikanth, Jampala queue_work(ndev->iov.pf2vf_wq, &pfwork->pf2vf_resp); 147cf718eaaSSrikanth, Jampala /* clear the corresponding vf bit */ 148cf718eaaSSrikanth, Jampala nitrox_write_csr(ndev, reg_addr, BIT_ULL(i)); 149cf718eaaSSrikanth, Jampala } 150cf718eaaSSrikanth, Jampala 151cf718eaaSSrikanth, Jampala /* loop for VF(64..127) */ 152cf718eaaSSrikanth, Jampala reg_addr = NPS_PKT_MBOX_INT_HI; 153cf718eaaSSrikanth, Jampala value = nitrox_read_csr(ndev, reg_addr); 154cf718eaaSSrikanth, Jampala for_each_set_bit(i, (const unsigned long *)&value, BITS_PER_LONG) { 155cf718eaaSSrikanth, Jampala /* get the vfno from ring */ 156cf718eaaSSrikanth, Jampala vfno = RING_TO_VFNO(i + 64, ndev->iov.max_vf_queues); 157cf718eaaSSrikanth, Jampala vfdev = ndev->iov.vfdev + vfno; 158cf718eaaSSrikanth, Jampala vfdev->ring = (i + 64); 159cf718eaaSSrikanth, Jampala /* fill the vf mailbox data */ 160cf718eaaSSrikanth, Jampala vfdev->msg.value = pf2vf_read_mbox(ndev, vfdev->ring); 161cf718eaaSSrikanth, Jampala 162cf718eaaSSrikanth, Jampala pfwork = kzalloc(sizeof(*pfwork), GFP_ATOMIC); 163cf718eaaSSrikanth, Jampala if (!pfwork) 164cf718eaaSSrikanth, Jampala continue; 165cf718eaaSSrikanth, Jampala 166cf718eaaSSrikanth, Jampala pfwork->vfdev = vfdev; 167cf718eaaSSrikanth, Jampala pfwork->ndev = ndev; 168cf718eaaSSrikanth, Jampala INIT_WORK(&pfwork->pf2vf_resp, pf2vf_resp_handler); 169cf718eaaSSrikanth, Jampala queue_work(ndev->iov.pf2vf_wq, &pfwork->pf2vf_resp); 170cf718eaaSSrikanth, Jampala /* clear the corresponding vf bit */ 171cf718eaaSSrikanth, Jampala nitrox_write_csr(ndev, reg_addr, BIT_ULL(i)); 172cf718eaaSSrikanth, Jampala } 173cf718eaaSSrikanth, Jampala } 174cf718eaaSSrikanth, Jampala 175cf718eaaSSrikanth, Jampala int nitrox_mbox_init(struct nitrox_device *ndev) 176cf718eaaSSrikanth, Jampala { 177cf718eaaSSrikanth, Jampala struct nitrox_vfdev *vfdev; 178cf718eaaSSrikanth, Jampala int i; 179cf718eaaSSrikanth, Jampala 180cf718eaaSSrikanth, Jampala ndev->iov.vfdev = kcalloc(ndev->iov.num_vfs, 181cf718eaaSSrikanth, Jampala sizeof(struct nitrox_vfdev), GFP_KERNEL); 182cf718eaaSSrikanth, Jampala if (!ndev->iov.vfdev) 183cf718eaaSSrikanth, Jampala return -ENOMEM; 184cf718eaaSSrikanth, Jampala 185cf718eaaSSrikanth, Jampala for (i = 0; i < ndev->iov.num_vfs; i++) { 186cf718eaaSSrikanth, Jampala vfdev = ndev->iov.vfdev + i; 187cf718eaaSSrikanth, Jampala vfdev->vfno = i; 188cf718eaaSSrikanth, Jampala } 189cf718eaaSSrikanth, Jampala 190cf718eaaSSrikanth, Jampala /* allocate pf2vf response workqueue */ 191cf718eaaSSrikanth, Jampala ndev->iov.pf2vf_wq = alloc_workqueue("nitrox_pf2vf", 0, 0); 192cf718eaaSSrikanth, Jampala if (!ndev->iov.pf2vf_wq) { 193cf718eaaSSrikanth, Jampala kfree(ndev->iov.vfdev); 194cf718eaaSSrikanth, Jampala return -ENOMEM; 195cf718eaaSSrikanth, Jampala } 196cf718eaaSSrikanth, Jampala /* enable pf2vf mailbox interrupts */ 197cf718eaaSSrikanth, Jampala enable_pf2vf_mbox_interrupts(ndev); 198cf718eaaSSrikanth, Jampala 199cf718eaaSSrikanth, Jampala return 0; 200cf718eaaSSrikanth, Jampala } 201cf718eaaSSrikanth, Jampala 202cf718eaaSSrikanth, Jampala void nitrox_mbox_cleanup(struct nitrox_device *ndev) 203cf718eaaSSrikanth, Jampala { 204cf718eaaSSrikanth, Jampala /* disable pf2vf mailbox interrupts */ 205cf718eaaSSrikanth, Jampala disable_pf2vf_mbox_interrupts(ndev); 206cf718eaaSSrikanth, Jampala /* destroy workqueue */ 207cf718eaaSSrikanth, Jampala if (ndev->iov.pf2vf_wq) 208cf718eaaSSrikanth, Jampala destroy_workqueue(ndev->iov.pf2vf_wq); 209cf718eaaSSrikanth, Jampala 210cf718eaaSSrikanth, Jampala kfree(ndev->iov.vfdev); 211cf718eaaSSrikanth, Jampala ndev->iov.pf2vf_wq = NULL; 212cf718eaaSSrikanth, Jampala ndev->iov.vfdev = NULL; 213cf718eaaSSrikanth, Jampala } 214