xref: /linux/drivers/infiniband/hw/ionic/ionic_pgtbl.c (revision b83c62055b6faabb444b2f8f3355420927cc39fd)
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