pci-dma_32.c 4.2 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13
/*
 * Dynamic DMA mapping support.
 *
 * On i386 there is no hardware dynamic DMA address translation,
 * so consistent alloc/free are merely page allocation/freeing.
 * The rest of the dynamic DMA mapping interface is implemented
 * in asm/pci.h.
 */

#include <linux/types.h>
#include <linux/mm.h>
#include <linux/string.h>
#include <linux/pci.h>
14
#include <linux/module.h>
L
Linus Torvalds 已提交
15 16
#include <asm/io.h>

G
Glauber Costa 已提交
17 18 19 20 21 22 23 24 25 26
/* Dummy device used for NULL arguments (normally ISA). Better would
   be probably a smaller DMA mask, but this is bug-to-bug compatible
   to i386. */
struct device fallback_dev = {
	.bus_id = "fallback device",
	.coherent_dma_mask = DMA_32BIT_MASK,
	.dma_mask = &fallback_dev.coherent_dma_mask,
};


27 28
static int dma_alloc_from_coherent_mem(struct device *dev, ssize_t size,
				       dma_addr_t *dma_handle, void **ret)
L
Linus Torvalds 已提交
29 30 31 32 33 34 35 36 37
{
	struct dma_coherent_mem *mem = dev ? dev->dma_mem : NULL;
	int order = get_order(size);

	if (mem) {
		int page = bitmap_find_free_region(mem->bitmap, mem->size,
						     order);
		if (page >= 0) {
			*dma_handle = mem->device_base + (page << PAGE_SHIFT);
38 39
			*ret = mem->virt_base + (page << PAGE_SHIFT);
			memset(*ret, 0, size);
L
Linus Torvalds 已提交
40 41
		}
		if (mem->flags & DMA_MEMORY_EXCLUSIVE)
42 43 44 45 46 47 48 49 50 51 52 53 54 55 56
			*ret = NULL;
	}
	return (mem != NULL);
}

static int dma_release_coherent(struct device *dev, int order, void *vaddr)
{
	struct dma_coherent_mem *mem = dev ? dev->dma_mem : NULL;

	if (mem && vaddr >= mem->virt_base && vaddr <
		   (mem->virt_base + (mem->size << PAGE_SHIFT))) {
		int page = (vaddr - mem->virt_base) >> PAGE_SHIFT;

		bitmap_release_region(mem->bitmap, page, order);
		return 1;
L
Linus Torvalds 已提交
57
	}
58 59 60
	return 0;
}

61 62 63 64 65 66 67 68 69 70 71
/* Allocate DMA memory on node near device */
noinline struct page *
dma_alloc_pages(struct device *dev, gfp_t gfp, unsigned order)
{
	int node;

	node = dev_to_node(dev);

	return alloc_pages_node(node, gfp, order);
}

72 73 74 75
void *dma_alloc_coherent(struct device *dev, size_t size,
			   dma_addr_t *dma_handle, gfp_t gfp)
{
	void *ret = NULL;
76 77
	struct page *page;
	dma_addr_t bus;
78
	int order = get_order(size);
G
Glauber Costa 已提交
79 80
	unsigned long dma_mask = 0;

81
	/* ignore region specifiers */
G
Glauber Costa 已提交
82
	gfp &= ~(__GFP_DMA | __GFP_HIGHMEM | __GFP_DMA32);
83 84 85

	if (dma_alloc_from_coherent_mem(dev, size, dma_handle, &ret))
		return ret;
L
Linus Torvalds 已提交
86

G
Glauber Costa 已提交
87 88 89
	if (!dev)
		dev = &fallback_dev;

G
Glauber Costa 已提交
90 91 92 93
	dma_mask = dev->coherent_dma_mask;
	if (dma_mask == 0)
		dma_mask = DMA_32BIT_MASK;

94 95 96
	if (dev->dma_mask == NULL)
		return NULL;

G
Glauber Costa 已提交
97 98 99
	/* Don't invoke OOM killer */
	gfp |= __GFP_NORETRY;
again:
100 101 102 103
	page = dma_alloc_pages(dev, gfp, order);
	if (page == NULL)
		return NULL;

G
Glauber Costa 已提交
104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121
	{
		int high, mmu;
		bus = page_to_phys(page);
		ret = page_address(page);
		high = (bus + size) >= dma_mask;
		mmu = high;
		if (force_iommu && !(gfp & GFP_DMA))
			mmu = 1;
		else if (high) {
			free_pages((unsigned long)ret,
				   get_order(size));

			/* Don't use the 16MB ZONE_DMA unless absolutely
			   needed. It's better to use remapping first. */
			if (dma_mask < DMA_32BIT_MASK && !(gfp & GFP_DMA)) {
				gfp = (gfp & ~GFP_DMA32) | GFP_DMA;
				goto again;
			}
G
Glauber Costa 已提交
122 123 124 125 126 127 128 129 130

			/* Let low level make its own zone decisions */
			gfp &= ~(GFP_DMA32|GFP_DMA);

			if (dma_ops->alloc_coherent)
				return dma_ops->alloc_coherent(dev, size,
							   dma_handle, gfp);
			return NULL;

G
Glauber Costa 已提交
131 132
		}
		memset(ret, 0, size);
G
Glauber Costa 已提交
133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150
		if (!mmu) {
			*dma_handle = bus;
			return ret;
		}
	}

	if (dma_ops->alloc_coherent) {
		free_pages((unsigned long)ret, get_order(size));
		gfp &= ~(GFP_DMA|GFP_DMA32);
		return dma_ops->alloc_coherent(dev, size, dma_handle, gfp);
	}

	if (dma_ops->map_simple) {
		*dma_handle = dma_ops->map_simple(dev, virt_to_phys(ret),
					      size,
					      PCI_DMA_BIDIRECTIONAL);
		if (*dma_handle != bad_dma_address)
			return ret;
G
Glauber Costa 已提交
151
	}
L
Linus Torvalds 已提交
152

G
Glauber Costa 已提交
153 154 155 156 157
	if (panic_on_overflow)
		panic("dma_alloc_coherent: IOMMU overflow by %lu bytes\n",
		      (unsigned long)size);
	free_pages((unsigned long)ret, get_order(size));
	return NULL;
L
Linus Torvalds 已提交
158
}
159
EXPORT_SYMBOL(dma_alloc_coherent);
L
Linus Torvalds 已提交
160 161 162 163 164

void dma_free_coherent(struct device *dev, size_t size,
			 void *vaddr, dma_addr_t dma_handle)
{
	int order = get_order(size);
165 166

	WARN_ON(irqs_disabled());	/* for portability */
167 168
	if (dma_release_coherent(dev, order, vaddr))
		return;
169 170
	if (dma_ops->unmap_single)
		dma_ops->unmap_single(dev, dma_handle, size, 0);
171
	free_pages((unsigned long)vaddr, order);
L
Linus Torvalds 已提交
172
}
173
EXPORT_SYMBOL(dma_free_coherent);