aboutsummaryrefslogtreecommitdiffstats
path: root/arch/x86
diff options
context:
space:
mode:
Diffstat (limited to 'arch/x86')
-rw-r--r--arch/x86/kernel/pci-dma_32.c34
1 files changed, 32 insertions, 2 deletions
diff --git a/arch/x86/kernel/pci-dma_32.c b/arch/x86/kernel/pci-dma_32.c
index 5450bd142cb0..f134de3833a2 100644
--- a/arch/x86/kernel/pci-dma_32.c
+++ b/arch/x86/kernel/pci-dma_32.c
@@ -116,12 +116,42 @@ again:
116 gfp = (gfp & ~GFP_DMA32) | GFP_DMA; 116 gfp = (gfp & ~GFP_DMA32) | GFP_DMA;
117 goto again; 117 goto again;
118 } 118 }
119
120 /* Let low level make its own zone decisions */
121 gfp &= ~(GFP_DMA32|GFP_DMA);
122
123 if (dma_ops->alloc_coherent)
124 return dma_ops->alloc_coherent(dev, size,
125 dma_handle, gfp);
126 return NULL;
127
119 } 128 }
120 memset(ret, 0, size); 129 memset(ret, 0, size);
121 *dma_handle = bus; 130 if (!mmu) {
131 *dma_handle = bus;
132 return ret;
133 }
134 }
135
136 if (dma_ops->alloc_coherent) {
137 free_pages((unsigned long)ret, get_order(size));
138 gfp &= ~(GFP_DMA|GFP_DMA32);
139 return dma_ops->alloc_coherent(dev, size, dma_handle, gfp);
140 }
141
142 if (dma_ops->map_simple) {
143 *dma_handle = dma_ops->map_simple(dev, virt_to_phys(ret),
144 size,
145 PCI_DMA_BIDIRECTIONAL);
146 if (*dma_handle != bad_dma_address)
147 return ret;
122 } 148 }
123 149
124 return ret; 150 if (panic_on_overflow)
151 panic("dma_alloc_coherent: IOMMU overflow by %lu bytes\n",
152 (unsigned long)size);
153 free_pages((unsigned long)ret, get_order(size));
154 return NULL;
125} 155}
126EXPORT_SYMBOL(dma_alloc_coherent); 156EXPORT_SYMBOL(dma_alloc_coherent);
127 157