diff options
Diffstat (limited to 'sys/arch/mvme88k/dev')
-rw-r--r-- | sys/arch/mvme88k/dev/mainbus.c | 91 |
1 files changed, 86 insertions, 5 deletions
diff --git a/sys/arch/mvme88k/dev/mainbus.c b/sys/arch/mvme88k/dev/mainbus.c index a394254f76d..bc82805a4ce 100644 --- a/sys/arch/mvme88k/dev/mainbus.c +++ b/sys/arch/mvme88k/dev/mainbus.c @@ -1,4 +1,4 @@ -/* $OpenBSD: mainbus.c,v 1.13 2004/08/02 08:35:00 miod Exp $ */ +/* $OpenBSD: mainbus.c,v 1.14 2004/11/09 15:02:23 miod Exp $ */ /* * Copyright (c) 1998 Steve Murphree, Jr. * Copyright (c) 2004, Miodrag Vallat. @@ -31,11 +31,15 @@ #include <sys/conf.h> #include <sys/device.h> #include <sys/disklabel.h> +#include <sys/extent.h> + +#include <uvm/uvm_extern.h> #include <machine/bus.h> #include <machine/autoconf.h> #include <machine/cmmu.h> #include <machine/cpu.h> +#include <machine/cpu_number.h> #ifdef MVME188 #include <machine/mvme188.h> @@ -66,6 +70,12 @@ const struct mvme88k_bus_space_tag mainbus_bustag = { mainbus_vaddr }; +extern struct extent *iomap_extent; +extern struct vm_map *iomap_map; + +void *mapiodev(void *, int); +void unmapiodev(void *, int); + /* * Obio (internal IO) space is mapped 1:1 (see pmap_bootstrap() for details). * @@ -103,11 +113,7 @@ mainbus_map(bus_addr_t addr, bus_size_t size, int flags, if (addr >= threshold) map = (vaddr_t)addr; else { -#if 0 - map = iomap_mapin(addr, size, 0); -#else map = (vaddr_t)mapiodev((void *)addr, size); -#endif } if (map == NULL) @@ -138,6 +144,81 @@ mainbus_vaddr(bus_space_handle_t handle) } /* + * Map a range [pa, pa+size] in the given map to a kernel address + * in iomap space. + * + * Note: To be flexible, I did not put a restriction on the alignment + * of pa. However, it is advisable to have pa page aligned since otherwise, + * we might have several mappings for a given chunk of the IO page. + */ +void * +mapiodev(pa, size) + void *pa; + int size; +{ + vaddr_t iova, tva, off; + paddr_t ppa; + int s, error; + + if (size <= 0) + return NULL; + + ppa = trunc_page((paddr_t)pa); + off = (paddr_t)pa & PGOFSET; + size = round_page(off + size); + + s = splhigh(); + error = extent_alloc(iomap_extent, size, PAGE_SIZE, 0, EX_NOBOUNDARY, + EX_WAITSPACE, &iova); + splx(s); + + if (error != 0) + return NULL; + + cmmu_flush_tlb(cpu_number(), 1, iova, size); /* necessary? */ + + tva = iova; + while (size != 0) { + pmap_enter(vm_map_pmap(iomap_map), tva, ppa, + VM_PROT_WRITE | VM_PROT_READ, + VM_PROT_WRITE | VM_PROT_READ | PMAP_WIRED); + size -= PAGE_SIZE; + tva += PAGE_SIZE; + ppa += PAGE_SIZE; + } + pmap_update(vm_map_pmap(iomap_map)); + + return (void *)(iova + off); +} + +/* + * Free up the mapping in iomap. + */ +void +unmapiodev(va, size) + void *va; + int size; +{ + vaddr_t kva, off; + int s, error; + + off = (vaddr_t)va & PGOFSET; + kva = trunc_page((vaddr_t)va); + size = round_page(off + size); + + pmap_remove(vm_map_pmap(iomap_map), kva, kva + size); + pmap_update(vm_map_pmap(iomap_map)); + + s = splhigh(); + error = extent_free(iomap_extent, kva, size, EX_NOWAIT); +#ifdef DIAGNOSTIC + if (error != 0) + printf("unmapiodev: extent_free failed\n"); +#endif + splx(s); +} + +/* * Configuration glue */ |