ioremap.c 4.2 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16
/*
 * arch/cris/mm/ioremap.c
 *
 * Re-map IO memory to kernel address space so that we can access it.
 * Needed for memory-mapped I/O devices mapped outside our normal DRAM
 * window (that is, all memory-mapped I/O devices).
 *
 * (C) Copyright 1995 1996 Linus Torvalds
 * CRIS-port by Axis Communications AB
 */

#include <linux/vmalloc.h>
#include <asm/io.h>
#include <asm/pgalloc.h>
#include <asm/cacheflush.h>
#include <asm/tlbflush.h>
M
Mikael Starvik 已提交
17
#include <asm/arch/memmap.h>
L
Linus Torvalds 已提交
18 19

extern inline void remap_area_pte(pte_t * pte, unsigned long address, unsigned long size,
M
Mikael Starvik 已提交
20
	unsigned long phys_addr, pgprot_t prot)
L
Linus Torvalds 已提交
21 22 23 24 25 26 27 28 29 30 31 32 33 34
{
	unsigned long end;

	address &= ~PMD_MASK;
	end = address + size;
	if (end > PMD_SIZE)
		end = PMD_SIZE;
	if (address >= end)
		BUG();
	do {
		if (!pte_none(*pte)) {
			printk("remap_area_pte: page already exists\n");
			BUG();
		}
M
Mikael Starvik 已提交
35
		set_pte(pte, mk_pte_phys(phys_addr, prot));
L
Linus Torvalds 已提交
36 37 38 39 40 41 42
		address += PAGE_SIZE;
		phys_addr += PAGE_SIZE;
		pte++;
	} while (address && (address < end));
}

static inline int remap_area_pmd(pmd_t * pmd, unsigned long address, unsigned long size,
M
Mikael Starvik 已提交
43
	unsigned long phys_addr, pgprot_t prot)
L
Linus Torvalds 已提交
44 45 46 47 48 49 50 51 52 53 54 55 56 57
{
	unsigned long end;

	address &= ~PGDIR_MASK;
	end = address + size;
	if (end > PGDIR_SIZE)
		end = PGDIR_SIZE;
	phys_addr -= address;
	if (address >= end)
		BUG();
	do {
		pte_t * pte = pte_alloc_kernel(&init_mm, pmd, address);
		if (!pte)
			return -ENOMEM;
M
Mikael Starvik 已提交
58
		remap_area_pte(pte, address, end - address, address + phys_addr, prot);
L
Linus Torvalds 已提交
59 60 61 62 63 64 65
		address = (address + PMD_SIZE) & PMD_MASK;
		pmd++;
	} while (address && (address < end));
	return 0;
}

static int remap_area_pages(unsigned long address, unsigned long phys_addr,
M
Mikael Starvik 已提交
66
				 unsigned long size, pgprot_t prot)
L
Linus Torvalds 已提交
67 68 69 70 71 72 73 74 75 76 77 78
{
	int error;
	pgd_t * dir;
	unsigned long end = address + size;

	phys_addr -= address;
	dir = pgd_offset(&init_mm, address);
	flush_cache_all();
	if (address >= end)
		BUG();
	spin_lock(&init_mm.page_table_lock);
	do {
M
Mikael Starvik 已提交
79
		pud_t *pud;
L
Linus Torvalds 已提交
80
		pmd_t *pmd;
M
Mikael Starvik 已提交
81

L
Linus Torvalds 已提交
82
		error = -ENOMEM;
M
Mikael Starvik 已提交
83 84 85 86 87
		pud = pud_alloc(&init_mm, dir, address);
		if (!pud)
			break;
		pmd = pmd_alloc(&init_mm, pud, address);

L
Linus Torvalds 已提交
88 89 90
		if (!pmd)
			break;
		if (remap_area_pmd(pmd, address, end - address,
M
Mikael Starvik 已提交
91
				   phys_addr + address, prot))
L
Linus Torvalds 已提交
92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114
			break;
		error = 0;
		address = (address + PGDIR_SIZE) & PGDIR_MASK;
		dir++;
	} while (address && (address < end));
	spin_unlock(&init_mm.page_table_lock);
	flush_tlb_all();
	return error;
}

/*
 * Generic mapping function (not visible outside):
 */

/*
 * Remap an arbitrary physical address space into the kernel virtual
 * address space. Needed when the kernel wants to access high addresses
 * directly.
 *
 * NOTE! We need to allow non-page-aligned mappings too: we will obviously
 * have to convert them into an offset in a page-aligned mapping, but the
 * caller shouldn't need to know that small detail.
 */
M
Mikael Starvik 已提交
115
void __iomem * __ioremap_prot(unsigned long phys_addr, unsigned long size, pgprot_t prot)
L
Linus Torvalds 已提交
116
{
M
Mikael Starvik 已提交
117
	void __iomem * addr;
L
Linus Torvalds 已提交
118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138
	struct vm_struct * area;
	unsigned long offset, last_addr;

	/* Don't allow wraparound or zero size */
	last_addr = phys_addr + size - 1;
	if (!size || last_addr < phys_addr)
		return NULL;

	/*
	 * Mappings have to be page-aligned
	 */
	offset = phys_addr & ~PAGE_MASK;
	phys_addr &= PAGE_MASK;
	size = PAGE_ALIGN(last_addr+1) - phys_addr;

	/*
	 * Ok, go for it..
	 */
	area = get_vm_area(size, VM_IOREMAP);
	if (!area)
		return NULL;
M
Mikael Starvik 已提交
139 140 141
	addr = (void __iomem *)area->addr;
	if (remap_area_pages((unsigned long) addr, phys_addr, size, prot)) {
		vfree((void __force *)addr);
L
Linus Torvalds 已提交
142 143
		return NULL;
	}
M
Mikael Starvik 已提交
144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165
	return (void __iomem *) (offset + (char __iomem *)addr);
}

void __iomem * __ioremap(unsigned long phys_addr, unsigned long size, unsigned long flags)
{
	return __ioremap_prot(phys_addr, size,
		              __pgprot(_PAGE_PRESENT | __READABLE |
				       __WRITEABLE | _PAGE_GLOBAL |
				       _PAGE_KERNEL | flags));
}

/**
 * ioremap_nocache     -   map bus memory into CPU space
 * @offset:    bus address of the memory
 * @size:      size of the resource to map
 *
 * Must be freed with iounmap.
 */

void __iomem *ioremap_nocache (unsigned long phys_addr, unsigned long size)
{
        return __ioremap(phys_addr | MEM_NON_CACHEABLE, size, 0);
L
Linus Torvalds 已提交
166 167
}

M
Mikael Starvik 已提交
168
void iounmap(volatile void __iomem *addr)
L
Linus Torvalds 已提交
169 170 171 172
{
	if (addr > high_memory)
		return vfree((void *) (PAGE_MASK & (unsigned long) addr));
}