diff options
author | FUJITA Tomonori <fujita.tomonori@lab.ntt.co.jp> | 2008-09-08 05:10:13 -0400 |
---|---|---|
committer | Ingo Molnar <mingo@elte.hu> | 2008-09-08 09:50:07 -0400 |
commit | 823e7e8c6ef12cd1943dc42fe7595ca74e8cc3d7 (patch) | |
tree | 56f4b01741353ad4d24f24adeeedd68eaa041354 /include | |
parent | 8a53ad675f86ee003482b557da944e070d3c4859 (diff) |
x86: dma_alloc_coherent sets gfp flags properly
Non real IOMMU implemenations (which doesn't do virtual mappings,
e.g. swiotlb, pci-nommu, etc) need to use proper gfp flags and
dma_mask to allocate pages in their own dma_alloc_coherent()
(allocated page need to be suitable for device's coherent_dma_mask).
This patch makes dma_alloc_coherent do this job so that IOMMUs don't
need to take care of it any more.
Real IOMMU implemenataions can simply ignore the gfp flags.
Signed-off-by: FUJITA Tomonori <fujita.tomonori@lab.ntt.co.jp>
Acked-by: Joerg Roedel <joerg.roedel@amd.com>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
Diffstat (limited to 'include')
-rw-r--r-- | include/asm-x86/dma-mapping.h | 32 |
1 files changed, 28 insertions, 4 deletions
diff --git a/include/asm-x86/dma-mapping.h b/include/asm-x86/dma-mapping.h index ad8b49032d18..0cc022b9a4a1 100644 --- a/include/asm-x86/dma-mapping.h +++ b/include/asm-x86/dma-mapping.h | |||
@@ -239,6 +239,29 @@ static inline int dma_get_cache_alignment(void) | |||
239 | return boot_cpu_data.x86_clflush_size; | 239 | return boot_cpu_data.x86_clflush_size; |
240 | } | 240 | } |
241 | 241 | ||
242 | static inline unsigned long dma_alloc_coherent_mask(struct device *dev, | ||
243 | gfp_t gfp) | ||
244 | { | ||
245 | unsigned long dma_mask = 0; | ||
246 | |||
247 | dma_mask = dev->coherent_dma_mask; | ||
248 | if (!dma_mask) | ||
249 | dma_mask = (gfp & GFP_DMA) ? DMA_24BIT_MASK : DMA_32BIT_MASK; | ||
250 | |||
251 | return dma_mask; | ||
252 | } | ||
253 | |||
254 | static inline gfp_t dma_alloc_coherent_gfp_flags(struct device *dev, gfp_t gfp) | ||
255 | { | ||
256 | unsigned long dma_mask = dma_alloc_coherent_mask(dev, gfp); | ||
257 | |||
258 | #ifdef CONFIG_X86_64 | ||
259 | if (dma_mask <= DMA_32BIT_MASK && !(gfp & GFP_DMA)) | ||
260 | gfp |= GFP_DMA32; | ||
261 | #endif | ||
262 | return gfp; | ||
263 | } | ||
264 | |||
242 | static inline void * | 265 | static inline void * |
243 | dma_alloc_coherent(struct device *dev, size_t size, dma_addr_t *dma_handle, | 266 | dma_alloc_coherent(struct device *dev, size_t size, dma_addr_t *dma_handle, |
244 | gfp_t gfp) | 267 | gfp_t gfp) |
@@ -259,10 +282,11 @@ dma_alloc_coherent(struct device *dev, size_t size, dma_addr_t *dma_handle, | |||
259 | if (!dev->dma_mask) | 282 | if (!dev->dma_mask) |
260 | return NULL; | 283 | return NULL; |
261 | 284 | ||
262 | if (ops->alloc_coherent) | 285 | if (!ops->alloc_coherent) |
263 | return ops->alloc_coherent(dev, size, | 286 | return NULL; |
264 | dma_handle, gfp); | 287 | |
265 | return NULL; | 288 | return ops->alloc_coherent(dev, size, dma_handle, |
289 | dma_alloc_coherent_gfp_flags(dev, gfp)); | ||
266 | } | 290 | } |
267 | 291 | ||
268 | static inline void dma_free_coherent(struct device *dev, size_t size, | 292 | static inline void dma_free_coherent(struct device *dev, size_t size, |