2 * Coherent per-device memory handling.
5 #include <linux/slab.h>
6 #include <linux/kernel.h>
7 #include <linux/module.h>
8 #include <linux/dma-mapping.h>
10 struct dma_coherent_mem
{
12 dma_addr_t device_base
;
16 unsigned long *bitmap
;
19 int dma_declare_coherent_memory(struct device
*dev
, dma_addr_t bus_addr
,
20 dma_addr_t device_addr
, size_t size
, int flags
)
22 void __iomem
*mem_base
= NULL
;
23 int pages
= size
>> PAGE_SHIFT
;
24 int bitmap_size
= BITS_TO_LONGS(pages
) * sizeof(long);
26 if ((flags
& (DMA_MEMORY_MAP
| DMA_MEMORY_IO
)) == 0)
33 /* FIXME: this routine just ignores DMA_MEMORY_INCLUDES_CHILDREN */
35 mem_base
= ioremap(bus_addr
, size
);
39 dev
->dma_mem
= kzalloc(sizeof(struct dma_coherent_mem
), GFP_KERNEL
);
42 dev
->dma_mem
->bitmap
= kzalloc(bitmap_size
, GFP_KERNEL
);
43 if (!dev
->dma_mem
->bitmap
)
46 dev
->dma_mem
->virt_base
= mem_base
;
47 dev
->dma_mem
->device_base
= device_addr
;
48 dev
->dma_mem
->pfn_base
= PFN_DOWN(bus_addr
);
49 dev
->dma_mem
->size
= pages
;
50 dev
->dma_mem
->flags
= flags
;
52 if (flags
& DMA_MEMORY_MAP
)
53 return DMA_MEMORY_MAP
;
64 EXPORT_SYMBOL(dma_declare_coherent_memory
);
66 void dma_release_declared_memory(struct device
*dev
)
68 struct dma_coherent_mem
*mem
= dev
->dma_mem
;
73 iounmap(mem
->virt_base
);
77 EXPORT_SYMBOL(dma_release_declared_memory
);
79 void *dma_mark_declared_memory_occupied(struct device
*dev
,
80 dma_addr_t device_addr
, size_t size
)
82 struct dma_coherent_mem
*mem
= dev
->dma_mem
;
85 size
+= device_addr
& ~PAGE_MASK
;
88 return ERR_PTR(-EINVAL
);
90 pos
= (device_addr
- mem
->device_base
) >> PAGE_SHIFT
;
91 err
= bitmap_allocate_region(mem
->bitmap
, pos
, get_order(size
));
94 return mem
->virt_base
+ (pos
<< PAGE_SHIFT
);
96 EXPORT_SYMBOL(dma_mark_declared_memory_occupied
);
99 * dma_alloc_from_coherent() - try to allocate memory from the per-device coherent area
101 * @dev: device from which we allocate memory
102 * @size: size of requested memory area
103 * @dma_handle: This will be filled with the correct dma handle
104 * @ret: This pointer will be filled with the virtual address
107 * This function should be only called from per-arch dma_alloc_coherent()
108 * to support allocation from per-device coherent memory pools.
110 * Returns 0 if dma_alloc_coherent should continue with allocating from
111 * generic memory areas, or !0 if dma_alloc_coherent should return @ret.
113 int dma_alloc_from_coherent(struct device
*dev
, ssize_t size
,
114 dma_addr_t
*dma_handle
, void **ret
)
116 struct dma_coherent_mem
*mem
;
117 int order
= get_order(size
);
128 if (unlikely(size
> (mem
->size
<< PAGE_SHIFT
)))
131 pageno
= bitmap_find_free_region(mem
->bitmap
, mem
->size
, order
);
132 if (unlikely(pageno
< 0))
136 * Memory was found in the per-device area.
138 *dma_handle
= mem
->device_base
+ (pageno
<< PAGE_SHIFT
);
139 *ret
= mem
->virt_base
+ (pageno
<< PAGE_SHIFT
);
140 memset(*ret
, 0, size
);
146 * In the case where the allocation can not be satisfied from the
147 * per-device area, try to fall back to generic memory if the
148 * constraints allow it.
150 return mem
->flags
& DMA_MEMORY_EXCLUSIVE
;
152 EXPORT_SYMBOL(dma_alloc_from_coherent
);
155 * dma_release_from_coherent() - try to free the memory allocated from per-device coherent memory pool
156 * @dev: device from which the memory was allocated
157 * @order: the order of pages allocated
158 * @vaddr: virtual address of allocated pages
160 * This checks whether the memory was allocated from the per-device
161 * coherent memory pool and if so, releases that memory.
163 * Returns 1 if we correctly released the memory, or 0 if
164 * dma_release_coherent() should proceed with releasing memory from
167 int dma_release_from_coherent(struct device
*dev
, int order
, void *vaddr
)
169 struct dma_coherent_mem
*mem
= dev
? dev
->dma_mem
: NULL
;
171 if (mem
&& vaddr
>= mem
->virt_base
&& vaddr
<
172 (mem
->virt_base
+ (mem
->size
<< PAGE_SHIFT
))) {
173 int page
= (vaddr
- mem
->virt_base
) >> PAGE_SHIFT
;
175 bitmap_release_region(mem
->bitmap
, page
, order
);
180 EXPORT_SYMBOL(dma_release_from_coherent
);
183 * dma_mmap_from_coherent() - try to mmap the memory allocated from
184 * per-device coherent memory pool to userspace
185 * @dev: device from which the memory was allocated
186 * @vma: vm_area for the userspace memory
187 * @vaddr: cpu address returned by dma_alloc_from_coherent
188 * @size: size of the memory buffer allocated by dma_alloc_from_coherent
189 * @ret: result from remap_pfn_range()
191 * This checks whether the memory was allocated from the per-device
192 * coherent memory pool and if so, maps that memory to the provided vma.
194 * Returns 1 if we correctly mapped the memory, or 0 if
195 * dma_release_coherent() should proceed with mapping memory from
198 int dma_mmap_from_coherent(struct device
*dev
, struct vm_area_struct
*vma
,
199 void *vaddr
, size_t size
, int *ret
)
201 struct dma_coherent_mem
*mem
= dev
? dev
->dma_mem
: NULL
;
203 if (mem
&& vaddr
>= mem
->virt_base
&& vaddr
+ size
<=
204 (mem
->virt_base
+ (mem
->size
<< PAGE_SHIFT
))) {
205 unsigned long off
= vma
->vm_pgoff
;
206 int start
= (vaddr
- mem
->virt_base
) >> PAGE_SHIFT
;
207 int user_count
= (vma
->vm_end
- vma
->vm_start
) >> PAGE_SHIFT
;
208 int count
= size
>> PAGE_SHIFT
;
211 if (off
< count
&& user_count
<= count
- off
) {
212 unsigned pfn
= mem
->pfn_base
+ start
+ off
;
213 *ret
= remap_pfn_range(vma
, vma
->vm_start
, pfn
,
214 user_count
<< PAGE_SHIFT
,
221 EXPORT_SYMBOL(dma_mmap_from_coherent
);