diff options
author | Christoph Hellwig <hch@lst.de> | 2018-01-09 10:30:23 -0500 |
---|---|---|
committer | Christoph Hellwig <hch@lst.de> | 2018-01-15 03:35:06 -0500 |
commit | 002e67454f61bb67d8071ac4d0cacb86a01d18e0 (patch) | |
tree | ef15f40222144d529cc2e2b2bfdb72fad7fd4510 /lib/dma-direct.c | |
parent | c5cd037d1c8044fbd131c57822a67a1576eb16e9 (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.c | 63 |
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 | |||
13 | static 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 | |||
25 | static 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 | |||
31 | static 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 | |||
38 | static 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 | |||
57 | const 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 | }; | ||
63 | EXPORT_SYMBOL(dma_direct_ops); | ||