aboutsummaryrefslogtreecommitdiffstats
path: root/roms/openbios/drivers/iommu.c
diff options
context:
space:
mode:
Diffstat (limited to 'roms/openbios/drivers/iommu.c')
-rw-r--r--roms/openbios/drivers/iommu.c272
1 files changed, 272 insertions, 0 deletions
diff --git a/roms/openbios/drivers/iommu.c b/roms/openbios/drivers/iommu.c
new file mode 100644
index 000000000..a6c02b8bb
--- /dev/null
+++ b/roms/openbios/drivers/iommu.c
@@ -0,0 +1,272 @@
+/**
+ ** Proll (PROM replacement)
+ ** iommu.c: Functions for DVMA management.
+ ** Copyright 1999 Pete Zaitcev
+ ** This code is licensed under GNU General Public License.
+ **/
+#include "config.h"
+#include "libopenbios/bindings.h"
+#include "libopenbios/ofmem.h"
+#include "drivers/drivers.h"
+#include "iommu.h"
+#include "arch/sparc32/ofmem_sparc32.h"
+#include "arch/sparc32/asi.h"
+#include "arch/sparc32/pgtsrmmu.h"
+
+#ifdef CONFIG_DEBUG_IOMMU
+#define DPRINTF(fmt, args...) \
+ do { printk(fmt , ##args); } while (0)
+#else
+#define DPRINTF(fmt, args...)
+#endif
+
+/*
+ * IOMMU parameters
+ */
+struct iommu {
+ struct iommu_regs *regs;
+ unsigned int *page_table;
+ unsigned long plow; /* Base bus address */
+ unsigned long pphys; /* Base phys address */
+};
+
+static struct iommu ciommu;
+
+static void
+iommu_invalidate(struct iommu_regs *iregs)
+{
+ iregs->tlbflush = 0;
+}
+
+/*
+ * XXX This is a problematic interface. We alloc _memory_ which is uncached.
+ * So if we ever reuse allocations somebody is going to get uncached pages.
+ * Returned address is always aligned by page.
+ * BTW, we were not going to give away anonymous storage, were we not?
+ */
+void *
+dvma_alloc(int size)
+{
+ void *va;
+ unsigned int pa, iova;
+ unsigned int npages;
+ unsigned int mva, mpa;
+ unsigned int i;
+ unsigned int *iopte;
+ struct iommu *t = &ciommu;
+
+ npages = (size + (PAGE_SIZE-1)) / PAGE_SIZE;
+ iova = (unsigned int)mem_alloc(&cdvmem, npages * PAGE_SIZE, PAGE_SIZE);
+ if (iova == 0)
+ return NULL;
+
+ pa = t->pphys + (iova - t->plow);
+ va = (void *)pa2va((unsigned long)pa);
+
+ /*
+ * Change page attributes in MMU to uncached.
+ */
+ mva = (unsigned int) va;
+ mpa = (unsigned int) pa;
+ ofmem_arch_map_pages(mpa, mva, npages * PAGE_SIZE, ofmem_arch_io_translation_mode(mpa));
+
+ /*
+ * Map into IOMMU page table.
+ */
+ mpa = (unsigned int) pa;
+ iopte = &t->page_table[(iova - t->plow) / PAGE_SIZE];
+ for (i = 0; i < npages; i++) {
+ *iopte++ = MKIOPTE(mpa);
+ mpa += PAGE_SIZE;
+ }
+
+ return va;
+}
+
+void
+dvma_sync(unsigned char *va, int size)
+{
+ /* Synchronise the VA address region after DMA */
+ unsigned long virt = pointer2cell(va);
+ unsigned long page;
+
+ for (page = (unsigned long)virt; page < virt + size; page += PAGE_SIZE) {
+ srmmu_flush_tlb_page(page);
+ }
+}
+
+unsigned int
+dvma_map_in(unsigned char *va)
+{
+ /* Convert from VA to IOVA */
+ unsigned int pa, iova;
+ struct iommu *t = &ciommu;
+
+ pa = va2pa((unsigned int)va);
+ iova = t->plow + (pa - t->pphys);
+
+ return iova;
+}
+
+#define DVMA_SIZE 0x4000
+
+/*
+ * Initialize IOMMU
+ * This looks like initialization of CPU MMU but
+ * the routine is higher in food chain.
+ */
+static struct iommu_regs *
+iommu_init(struct iommu *t, uint64_t base)
+{
+ unsigned int *ptab, pva;
+ int ptsize;
+#ifdef CONFIG_DEBUG_IOMMU
+ unsigned int impl, vers;
+#endif
+ unsigned int tmp;
+ struct iommu_regs *regs;
+ int ret;
+ unsigned long vasize;
+
+ regs = (struct iommu_regs *)ofmem_map_io(base, IOMMU_REGS);
+ if (regs == NULL) {
+ DPRINTF("Cannot map IOMMU\n");
+ for (;;) { }
+ }
+ t->regs = regs;
+#ifdef CONFIG_DEBUG_IOMMU
+ impl = (regs->control & IOMMU_CTRL_IMPL) >> 28;
+ vers = (regs->control & IOMMU_CTRL_VERS) >> 24;
+#endif
+
+ tmp = regs->control;
+ tmp &= ~(IOMMU_CTRL_RNGE);
+
+ tmp |= (IOMMU_RNGE_32MB | IOMMU_CTRL_ENAB);
+ t->plow = 0xfe000000; /* End - 32 MB */
+ /* Size of VA region that we manage */
+ vasize = 0x2000000; /* 32 MB */
+
+ regs->control = tmp;
+ iommu_invalidate(regs);
+
+ /* Allocate IOMMU page table */
+ /* Tremendous alignment causes great waste... */
+ ptsize = (vasize / PAGE_SIZE) * sizeof(int);
+ ret = ofmem_posix_memalign((void *)&ptab, ptsize, ptsize);
+ if (ret != 0) {
+ DPRINTF("Cannot allocate IOMMU table [0x%x]\n", ptsize);
+ for (;;) { }
+ }
+ t->page_table = ptab;
+
+ /* flush_cache_all(); */
+ /** flush_tlb_all(); **/
+ tmp = (unsigned int)va2pa((unsigned long)ptab);
+ regs->base = tmp >> 4;
+ iommu_invalidate(regs);
+
+ DPRINTF("IOMMU: impl %d vers %d page table at 0x%p (pa 0x%x) of size %d bytes\n",
+ impl, vers, t->page_table, tmp, ptsize);
+
+ mem_init(&cdvmem, (char*)t->plow, (char *)(t->plow + DVMA_SIZE));
+ ret = ofmem_posix_memalign((void *)&pva, DVMA_SIZE, PAGE_SIZE);
+ if (ret != 0) {
+ DPRINTF("Cannot allocate IOMMU phys size [0x%x]\n", DVMA_SIZE);
+ for (;;) { }
+ }
+ t->pphys = va2pa(pva);
+ return regs;
+}
+
+/* ( addr.lo addr.hi size -- virt ) */
+
+static void
+ob_iommu_map_in(void)
+{
+ phys_addr_t phys;
+ ucell size, virt;
+
+ size = POP();
+ phys = POP();
+ phys = (phys << 32) + POP();
+
+ virt = ofmem_map_io(phys, size);
+
+ PUSH(virt);
+}
+
+/* ( virt size ) */
+
+static void
+ob_iommu_map_out(void)
+{
+ ucell size = POP();
+ ucell virt = POP();
+
+ ofmem_release_io(virt, size);
+}
+
+static void
+ob_iommu_dma_alloc(void)
+{
+ call_parent_method("dma-alloc");
+}
+
+static void
+ob_iommu_dma_free(void)
+{
+ call_parent_method("dma-free");
+}
+
+static void
+ob_iommu_dma_map_in(void)
+{
+ call_parent_method("dma-map-in");
+}
+
+static void
+ob_iommu_dma_map_out(void)
+{
+ call_parent_method("dma-map-out");
+}
+
+static void
+ob_iommu_dma_sync(void)
+{
+ call_parent_method("dma-sync");
+}
+
+void
+ob_init_iommu(uint64_t base)
+{
+ struct iommu_regs *regs;
+
+ regs = iommu_init(&ciommu, base);
+
+ push_str("/iommu");
+ fword("find-device");
+ PUSH((unsigned long)regs);
+ fword("encode-int");
+ push_str("address");
+ fword("property");
+
+ PUSH(base >> 32);
+ fword("encode-int");
+ PUSH(base & 0xffffffff);
+ fword("encode-int");
+ fword("encode+");
+ PUSH(IOMMU_REGS);
+ fword("encode-int");
+ fword("encode+");
+ push_str("reg");
+ fword("property");
+
+ bind_func("map-in", ob_iommu_map_in);
+ bind_func("map-out", ob_iommu_map_out);
+ bind_func("dma-alloc", ob_iommu_dma_alloc);
+ bind_func("dma-free", ob_iommu_dma_free);
+ bind_func("dma-map-in", ob_iommu_dma_map_in);
+ bind_func("dma-map-out", ob_iommu_dma_map_out);
+ bind_func("dma-sync", ob_iommu_dma_sync);
+}