1f3bdbd42SAbhijit Gangurde // SPDX-License-Identifier: GPL-2.0 2f3bdbd42SAbhijit Gangurde /* Copyright (C) 2018-2025, Advanced Micro Devices, Inc. */ 3f3bdbd42SAbhijit Gangurde 4f3bdbd42SAbhijit Gangurde #include <linux/mman.h> 5f3bdbd42SAbhijit Gangurde #include <linux/dma-mapping.h> 6f3bdbd42SAbhijit Gangurde 7f3bdbd42SAbhijit Gangurde #include "ionic_fw.h" 8f3bdbd42SAbhijit Gangurde #include "ionic_ibdev.h" 9f3bdbd42SAbhijit Gangurde 10e8521822SAbhijit Gangurde __le64 ionic_pgtbl_dma(struct ionic_tbl_buf *buf, u64 va) 11e8521822SAbhijit Gangurde { 12e8521822SAbhijit Gangurde u64 pg_mask = BIT_ULL(buf->page_size_log2) - 1; 13e8521822SAbhijit Gangurde u64 dma; 14e8521822SAbhijit Gangurde 15e8521822SAbhijit Gangurde if (!buf->tbl_pages) 16e8521822SAbhijit Gangurde return cpu_to_le64(0); 17e8521822SAbhijit Gangurde 18e8521822SAbhijit Gangurde if (buf->tbl_pages > 1) 19e8521822SAbhijit Gangurde return cpu_to_le64(buf->tbl_dma); 20e8521822SAbhijit Gangurde 21e8521822SAbhijit Gangurde if (buf->tbl_buf) 22e8521822SAbhijit Gangurde dma = le64_to_cpu(buf->tbl_buf[0]); 23e8521822SAbhijit Gangurde else 24e8521822SAbhijit Gangurde dma = buf->tbl_dma; 25e8521822SAbhijit Gangurde 26e8521822SAbhijit Gangurde return cpu_to_le64(dma + (va & pg_mask)); 27e8521822SAbhijit Gangurde } 28e8521822SAbhijit Gangurde 29*b83c6205SAbhijit Gangurde __be64 ionic_pgtbl_off(struct ionic_tbl_buf *buf, u64 va) 30*b83c6205SAbhijit Gangurde { 31*b83c6205SAbhijit Gangurde if (buf->tbl_pages > 1) { 32*b83c6205SAbhijit Gangurde u64 pg_mask = BIT_ULL(buf->page_size_log2) - 1; 33*b83c6205SAbhijit Gangurde 34*b83c6205SAbhijit Gangurde return cpu_to_be64(va & pg_mask); 35*b83c6205SAbhijit Gangurde } 36*b83c6205SAbhijit Gangurde 37*b83c6205SAbhijit Gangurde return 0; 38*b83c6205SAbhijit Gangurde } 39*b83c6205SAbhijit Gangurde 40f3bdbd42SAbhijit Gangurde int ionic_pgtbl_page(struct ionic_tbl_buf *buf, u64 dma) 41f3bdbd42SAbhijit Gangurde { 42f3bdbd42SAbhijit Gangurde if (unlikely(buf->tbl_pages == buf->tbl_limit)) 43f3bdbd42SAbhijit Gangurde return -ENOMEM; 44f3bdbd42SAbhijit Gangurde 45f3bdbd42SAbhijit Gangurde if (buf->tbl_buf) 46f3bdbd42SAbhijit Gangurde buf->tbl_buf[buf->tbl_pages] = cpu_to_le64(dma); 47f3bdbd42SAbhijit Gangurde else 48f3bdbd42SAbhijit Gangurde buf->tbl_dma = dma; 49f3bdbd42SAbhijit Gangurde 50f3bdbd42SAbhijit Gangurde ++buf->tbl_pages; 51f3bdbd42SAbhijit Gangurde 52f3bdbd42SAbhijit Gangurde return 0; 53f3bdbd42SAbhijit Gangurde } 54f3bdbd42SAbhijit Gangurde 55f3bdbd42SAbhijit Gangurde static int ionic_tbl_buf_alloc(struct ionic_ibdev *dev, 56f3bdbd42SAbhijit Gangurde struct ionic_tbl_buf *buf) 57f3bdbd42SAbhijit Gangurde { 58f3bdbd42SAbhijit Gangurde int rc; 59f3bdbd42SAbhijit Gangurde 60f3bdbd42SAbhijit Gangurde buf->tbl_size = buf->tbl_limit * sizeof(*buf->tbl_buf); 61f3bdbd42SAbhijit Gangurde buf->tbl_buf = kmalloc(buf->tbl_size, GFP_KERNEL); 62f3bdbd42SAbhijit Gangurde if (!buf->tbl_buf) 63f3bdbd42SAbhijit Gangurde return -ENOMEM; 64f3bdbd42SAbhijit Gangurde 65f3bdbd42SAbhijit Gangurde buf->tbl_dma = dma_map_single(dev->lif_cfg.hwdev, buf->tbl_buf, 66f3bdbd42SAbhijit Gangurde buf->tbl_size, DMA_TO_DEVICE); 67f3bdbd42SAbhijit Gangurde rc = dma_mapping_error(dev->lif_cfg.hwdev, buf->tbl_dma); 68f3bdbd42SAbhijit Gangurde if (rc) { 69f3bdbd42SAbhijit Gangurde kfree(buf->tbl_buf); 70f3bdbd42SAbhijit Gangurde return rc; 71f3bdbd42SAbhijit Gangurde } 72f3bdbd42SAbhijit Gangurde 73f3bdbd42SAbhijit Gangurde return 0; 74f3bdbd42SAbhijit Gangurde } 75f3bdbd42SAbhijit Gangurde 76f3bdbd42SAbhijit Gangurde static int ionic_pgtbl_umem(struct ionic_tbl_buf *buf, struct ib_umem *umem) 77f3bdbd42SAbhijit Gangurde { 78f3bdbd42SAbhijit Gangurde struct ib_block_iter biter; 79f3bdbd42SAbhijit Gangurde u64 page_dma; 80f3bdbd42SAbhijit Gangurde int rc; 81f3bdbd42SAbhijit Gangurde 82f3bdbd42SAbhijit Gangurde rdma_umem_for_each_dma_block(umem, &biter, BIT_ULL(buf->page_size_log2)) { 83f3bdbd42SAbhijit Gangurde page_dma = rdma_block_iter_dma_address(&biter); 84f3bdbd42SAbhijit Gangurde rc = ionic_pgtbl_page(buf, page_dma); 85f3bdbd42SAbhijit Gangurde if (rc) 86f3bdbd42SAbhijit Gangurde return rc; 87f3bdbd42SAbhijit Gangurde } 88f3bdbd42SAbhijit Gangurde 89f3bdbd42SAbhijit Gangurde return 0; 90f3bdbd42SAbhijit Gangurde } 91f3bdbd42SAbhijit Gangurde 92f3bdbd42SAbhijit Gangurde void ionic_pgtbl_unbuf(struct ionic_ibdev *dev, struct ionic_tbl_buf *buf) 93f3bdbd42SAbhijit Gangurde { 94f3bdbd42SAbhijit Gangurde if (buf->tbl_buf) 95f3bdbd42SAbhijit Gangurde dma_unmap_single(dev->lif_cfg.hwdev, buf->tbl_dma, 96f3bdbd42SAbhijit Gangurde buf->tbl_size, DMA_TO_DEVICE); 97f3bdbd42SAbhijit Gangurde 98f3bdbd42SAbhijit Gangurde kfree(buf->tbl_buf); 99f3bdbd42SAbhijit Gangurde memset(buf, 0, sizeof(*buf)); 100f3bdbd42SAbhijit Gangurde } 101f3bdbd42SAbhijit Gangurde 102f3bdbd42SAbhijit Gangurde int ionic_pgtbl_init(struct ionic_ibdev *dev, 103f3bdbd42SAbhijit Gangurde struct ionic_tbl_buf *buf, 104f3bdbd42SAbhijit Gangurde struct ib_umem *umem, 105f3bdbd42SAbhijit Gangurde dma_addr_t dma, 106f3bdbd42SAbhijit Gangurde int limit, 107f3bdbd42SAbhijit Gangurde u64 page_size) 108f3bdbd42SAbhijit Gangurde { 109f3bdbd42SAbhijit Gangurde int rc; 110f3bdbd42SAbhijit Gangurde 111f3bdbd42SAbhijit Gangurde memset(buf, 0, sizeof(*buf)); 112f3bdbd42SAbhijit Gangurde 113f3bdbd42SAbhijit Gangurde if (umem) { 114f3bdbd42SAbhijit Gangurde limit = ib_umem_num_dma_blocks(umem, page_size); 115f3bdbd42SAbhijit Gangurde buf->page_size_log2 = order_base_2(page_size); 116f3bdbd42SAbhijit Gangurde } 117f3bdbd42SAbhijit Gangurde 118f3bdbd42SAbhijit Gangurde if (limit < 1) 119f3bdbd42SAbhijit Gangurde return -EINVAL; 120f3bdbd42SAbhijit Gangurde 121f3bdbd42SAbhijit Gangurde buf->tbl_limit = limit; 122f3bdbd42SAbhijit Gangurde 123f3bdbd42SAbhijit Gangurde /* skip pgtbl if contiguous / direct translation */ 124f3bdbd42SAbhijit Gangurde if (limit > 1) { 125f3bdbd42SAbhijit Gangurde rc = ionic_tbl_buf_alloc(dev, buf); 126f3bdbd42SAbhijit Gangurde if (rc) 127f3bdbd42SAbhijit Gangurde return rc; 128f3bdbd42SAbhijit Gangurde } 129f3bdbd42SAbhijit Gangurde 130f3bdbd42SAbhijit Gangurde if (umem) 131f3bdbd42SAbhijit Gangurde rc = ionic_pgtbl_umem(buf, umem); 132f3bdbd42SAbhijit Gangurde else 133f3bdbd42SAbhijit Gangurde rc = ionic_pgtbl_page(buf, dma); 134f3bdbd42SAbhijit Gangurde 135f3bdbd42SAbhijit Gangurde if (rc) 136f3bdbd42SAbhijit Gangurde goto err_unbuf; 137f3bdbd42SAbhijit Gangurde 138f3bdbd42SAbhijit Gangurde return 0; 139f3bdbd42SAbhijit Gangurde 140f3bdbd42SAbhijit Gangurde err_unbuf: 141f3bdbd42SAbhijit Gangurde ionic_pgtbl_unbuf(dev, buf); 142f3bdbd42SAbhijit Gangurde return rc; 143f3bdbd42SAbhijit Gangurde } 144