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