summaryrefslogtreecommitdiffstats
path: root/lib/dma-direct.c
diff options
context:
space:
mode:
authorChristoph Hellwig <hch@lst.de>2018-01-09 10:30:23 -0500
committerChristoph Hellwig <hch@lst.de>2018-01-15 03:35:06 -0500
commit002e67454f61bb67d8071ac4d0cacb86a01d18e0 (patch)
treeef15f40222144d529cc2e2b2bfdb72fad7fd4510 /lib/dma-direct.c
parentc5cd037d1c8044fbd131c57822a67a1576eb16e9 (diff)
dma-direct: rename dma_noop to dma_direct
The trivial direct mapping implementation already does a virtual to physical translation which isn't strictly a noop, and will soon learn to do non-direct but linear physical to dma translations through the device offset and a few small tricks. Rename it to a better fitting name. Signed-off-by: Christoph Hellwig <hch@lst.de> Reviewed-by: Vladimir Murzin <vladimir.murzin@arm.com>
Diffstat (limited to 'lib/dma-direct.c')
-rw-r--r--lib/dma-direct.c63
1 files changed, 63 insertions, 0 deletions
diff --git a/lib/dma-direct.c b/lib/dma-direct.c
new file mode 100644
index 000000000000..0ec3262a3148
--- /dev/null
+++ b/lib/dma-direct.c
@@ -0,0 +1,63 @@
1// SPDX-License-Identifier: GPL-2.0
2/*
3 * lib/dma-noop.c
4 *
5 * DMA operations that map to physical addresses without flushing memory.
6 */
7#include <linux/export.h>
8#include <linux/mm.h>
9#include <linux/dma-mapping.h>
10#include <linux/scatterlist.h>
11#include <linux/pfn.h>
12
13static void *dma_direct_alloc(struct device *dev, size_t size,
14 dma_addr_t *dma_handle, gfp_t gfp, unsigned long attrs)
15{
16 void *ret;
17
18 ret = (void *)__get_free_pages(gfp, get_order(size));
19 if (ret)
20 *dma_handle = virt_to_phys(ret) - PFN_PHYS(dev->dma_pfn_offset);
21
22 return ret;
23}
24
25static void dma_direct_free(struct device *dev, size_t size, void *cpu_addr,
26 dma_addr_t dma_addr, unsigned long attrs)
27{
28 free_pages((unsigned long)cpu_addr, get_order(size));
29}
30
31static dma_addr_t dma_direct_map_page(struct device *dev, struct page *page,
32 unsigned long offset, size_t size, enum dma_data_direction dir,
33 unsigned long attrs)
34{
35 return page_to_phys(page) + offset - PFN_PHYS(dev->dma_pfn_offset);
36}
37
38static int dma_direct_map_sg(struct device *dev, struct scatterlist *sgl,
39 int nents, enum dma_data_direction dir, unsigned long attrs)
40{
41 int i;
42 struct scatterlist *sg;
43
44 for_each_sg(sgl, sg, nents, i) {
45 dma_addr_t offset = PFN_PHYS(dev->dma_pfn_offset);
46 void *va;
47
48 BUG_ON(!sg_page(sg));
49 va = sg_virt(sg);
50 sg_dma_address(sg) = (dma_addr_t)virt_to_phys(va) - offset;
51 sg_dma_len(sg) = sg->length;
52 }
53
54 return nents;
55}
56
57const struct dma_map_ops dma_direct_ops = {
58 .alloc = dma_direct_alloc,
59 .free = dma_direct_free,
60 .map_page = dma_direct_map_page,
61 .map_sg = dma_direct_map_sg,
62};
63EXPORT_SYMBOL(dma_direct_ops);