📄 iommu.c
字号:
/* * IOMMU implementation for Cell Broadband Processor Architecture * * (C) Copyright IBM Corporation 2006 * * Author: Jeremy Kerr <jk@ozlabs.org> * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2, or (at your option) * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */#undef DEBUG#include <linux/kernel.h>#include <linux/init.h>#include <linux/interrupt.h>#include <linux/notifier.h>#include <asm/prom.h>#include <asm/iommu.h>#include <asm/machdep.h>#include <asm/pci-bridge.h>#include <asm/udbg.h>#include <asm/of_platform.h>#include <asm/lmb.h>#include <asm/cell-regs.h>#include "interrupt.h"/* Define CELL_IOMMU_REAL_UNMAP to actually unmap non-used pages * instead of leaving them mapped to some dummy page. This can be * enabled once the appropriate workarounds for spider bugs have * been enabled */#define CELL_IOMMU_REAL_UNMAP/* Define CELL_IOMMU_STRICT_PROTECTION to enforce protection of * IO PTEs based on the transfer direction. That can be enabled * once spider-net has been fixed to pass the correct direction * to the DMA mapping functions */#define CELL_IOMMU_STRICT_PROTECTION#define NR_IOMMUS 2/* IOC mmap registers */#define IOC_Reg_Size 0x2000#define IOC_IOPT_CacheInvd 0x908#define IOC_IOPT_CacheInvd_NE_Mask 0xffe0000000000000ul#define IOC_IOPT_CacheInvd_IOPTE_Mask 0x000003fffffffff8ul#define IOC_IOPT_CacheInvd_Busy 0x0000000000000001ul#define IOC_IOST_Origin 0x918#define IOC_IOST_Origin_E 0x8000000000000000ul#define IOC_IOST_Origin_HW 0x0000000000000800ul#define IOC_IOST_Origin_HL 0x0000000000000400ul#define IOC_IO_ExcpStat 0x920#define IOC_IO_ExcpStat_V 0x8000000000000000ul#define IOC_IO_ExcpStat_SPF_Mask 0x6000000000000000ul#define IOC_IO_ExcpStat_SPF_S 0x6000000000000000ul#define IOC_IO_ExcpStat_SPF_P 0x4000000000000000ul#define IOC_IO_ExcpStat_ADDR_Mask 0x00000007fffff000ul#define IOC_IO_ExcpStat_RW_Mask 0x0000000000000800ul#define IOC_IO_ExcpStat_IOID_Mask 0x00000000000007fful#define IOC_IO_ExcpMask 0x928#define IOC_IO_ExcpMask_SFE 0x4000000000000000ul#define IOC_IO_ExcpMask_PFE 0x2000000000000000ul#define IOC_IOCmd_Offset 0x1000#define IOC_IOCmd_Cfg 0xc00#define IOC_IOCmd_Cfg_TE 0x0000800000000000ul/* Segment table entries */#define IOSTE_V 0x8000000000000000ul /* valid */#define IOSTE_H 0x4000000000000000ul /* cache hint */#define IOSTE_PT_Base_RPN_Mask 0x3ffffffffffff000ul /* base RPN of IOPT */#define IOSTE_NPPT_Mask 0x0000000000000fe0ul /* no. pages in IOPT */#define IOSTE_PS_Mask 0x0000000000000007ul /* page size */#define IOSTE_PS_4K 0x0000000000000001ul /* - 4kB */#define IOSTE_PS_64K 0x0000000000000003ul /* - 64kB */#define IOSTE_PS_1M 0x0000000000000005ul /* - 1MB */#define IOSTE_PS_16M 0x0000000000000007ul /* - 16MB *//* Page table entries */#define IOPTE_PP_W 0x8000000000000000ul /* protection: write */#define IOPTE_PP_R 0x4000000000000000ul /* protection: read */#define IOPTE_M 0x2000000000000000ul /* coherency required */#define IOPTE_SO_R 0x1000000000000000ul /* ordering: writes */#define IOPTE_SO_RW 0x1800000000000000ul /* ordering: r & w */#define IOPTE_RPN_Mask 0x07fffffffffff000ul /* RPN */#define IOPTE_H 0x0000000000000800ul /* cache hint */#define IOPTE_IOID_Mask 0x00000000000007fful /* ioid *//* IOMMU sizing */#define IO_SEGMENT_SHIFT 28#define IO_PAGENO_BITS (IO_SEGMENT_SHIFT - IOMMU_PAGE_SHIFT)/* The high bit needs to be set on every DMA address */#define SPIDER_DMA_OFFSET 0x80000000ulstruct iommu_window { struct list_head list; struct cbe_iommu *iommu; unsigned long offset; unsigned long size; unsigned long pte_offset; unsigned int ioid; struct iommu_table table;};#define NAMESIZE 8struct cbe_iommu { int nid; char name[NAMESIZE]; void __iomem *xlate_regs; void __iomem *cmd_regs; unsigned long *stab; unsigned long *ptab; void *pad_page; struct list_head windows;};/* Static array of iommus, one per node * each contains a list of windows, keyed from dma_window property * - on bus setup, look for a matching window, or create one * - on dev setup, assign iommu_table ptr */static struct cbe_iommu iommus[NR_IOMMUS];static int cbe_nr_iommus;static void invalidate_tce_cache(struct cbe_iommu *iommu, unsigned long *pte, long n_ptes){ unsigned long __iomem *reg; unsigned long val; long n; reg = iommu->xlate_regs + IOC_IOPT_CacheInvd; while (n_ptes > 0) { /* we can invalidate up to 1 << 11 PTEs at once */ n = min(n_ptes, 1l << 11); val = (((n /*- 1*/) << 53) & IOC_IOPT_CacheInvd_NE_Mask) | (__pa(pte) & IOC_IOPT_CacheInvd_IOPTE_Mask) | IOC_IOPT_CacheInvd_Busy; out_be64(reg, val); while (in_be64(reg) & IOC_IOPT_CacheInvd_Busy) ; n_ptes -= n; pte += n; }}static void tce_build_cell(struct iommu_table *tbl, long index, long npages, unsigned long uaddr, enum dma_data_direction direction){ int i; unsigned long *io_pte, base_pte; struct iommu_window *window = container_of(tbl, struct iommu_window, table); /* implementing proper protection causes problems with the spidernet * driver - check mapping directions later, but allow read & write by * default for now.*/#ifdef CELL_IOMMU_STRICT_PROTECTION /* to avoid referencing a global, we use a trick here to setup the * protection bit. "prot" is setup to be 3 fields of 4 bits apprended * together for each of the 3 supported direction values. It is then * shifted left so that the fields matching the desired direction * lands on the appropriate bits, and other bits are masked out. */ const unsigned long prot = 0xc48; base_pte = ((prot << (52 + 4 * direction)) & (IOPTE_PP_W | IOPTE_PP_R)) | IOPTE_M | IOPTE_SO_RW | (window->ioid & IOPTE_IOID_Mask);#else base_pte = IOPTE_PP_W | IOPTE_PP_R | IOPTE_M | IOPTE_SO_RW | (window->ioid & IOPTE_IOID_Mask);#endif io_pte = (unsigned long *)tbl->it_base + (index - window->pte_offset); for (i = 0; i < npages; i++, uaddr += IOMMU_PAGE_SIZE) io_pte[i] = base_pte | (__pa(uaddr) & IOPTE_RPN_Mask); mb(); invalidate_tce_cache(window->iommu, io_pte, npages); pr_debug("tce_build_cell(index=%lx,n=%lx,dir=%d,base_pte=%lx)\n", index, npages, direction, base_pte);}static void tce_free_cell(struct iommu_table *tbl, long index, long npages){ int i; unsigned long *io_pte, pte; struct iommu_window *window = container_of(tbl, struct iommu_window, table); pr_debug("tce_free_cell(index=%lx,n=%lx)\n", index, npages);#ifdef CELL_IOMMU_REAL_UNMAP pte = 0;#else /* spider bridge does PCI reads after freeing - insert a mapping * to a scratch page instead of an invalid entry */ pte = IOPTE_PP_R | IOPTE_M | IOPTE_SO_RW | __pa(window->iommu->pad_page) | (window->ioid & IOPTE_IOID_Mask);#endif io_pte = (unsigned long *)tbl->it_base + (index - window->pte_offset); for (i = 0; i < npages; i++) io_pte[i] = pte; mb(); invalidate_tce_cache(window->iommu, io_pte, npages);}static irqreturn_t ioc_interrupt(int irq, void *data){ unsigned long stat; struct cbe_iommu *iommu = data; stat = in_be64(iommu->xlate_regs + IOC_IO_ExcpStat); /* Might want to rate limit it */ printk(KERN_ERR "iommu: DMA exception 0x%016lx\n", stat); printk(KERN_ERR " V=%d, SPF=[%c%c], RW=%s, IOID=0x%04x\n", !!(stat & IOC_IO_ExcpStat_V), (stat & IOC_IO_ExcpStat_SPF_S) ? 'S' : ' ', (stat & IOC_IO_ExcpStat_SPF_P) ? 'P' : ' ', (stat & IOC_IO_ExcpStat_RW_Mask) ? "Read" : "Write", (unsigned int)(stat & IOC_IO_ExcpStat_IOID_Mask)); printk(KERN_ERR " page=0x%016lx\n", stat & IOC_IO_ExcpStat_ADDR_Mask); /* clear interrupt */ stat &= ~IOC_IO_ExcpStat_V; out_be64(iommu->xlate_regs + IOC_IO_ExcpStat, stat); return IRQ_HANDLED;}static int cell_iommu_find_ioc(int nid, unsigned long *base){ struct device_node *np; struct resource r; *base = 0; /* First look for new style /be nodes */ for_each_node_by_name(np, "ioc") { if (of_node_to_nid(np) != nid) continue; if (of_address_to_resource(np, 0, &r)) { printk(KERN_ERR "iommu: can't get address for %s\n", np->full_name); continue; } *base = r.start; of_node_put(np); return 0; } /* Ok, let's try the old way */ for_each_node_by_type(np, "cpu") { const unsigned int *nidp; const unsigned long *tmp; nidp = of_get_property(np, "node-id", NULL); if (nidp && *nidp == nid) { tmp = of_get_property(np, "ioc-translation", NULL); if (tmp) { *base = *tmp; of_node_put(np); return 0; } } } return -ENODEV;}static void cell_iommu_setup_hardware(struct cbe_iommu *iommu, unsigned long size){ struct page *page; int ret, i; unsigned long reg, segments, pages_per_segment, ptab_size, n_pte_pages; unsigned long xlate_base; unsigned int virq; if (cell_iommu_find_ioc(iommu->nid, &xlate_base)) panic("%s: missing IOC register mappings for node %d\n", __FUNCTION__, iommu->nid); iommu->xlate_regs = ioremap(xlate_base, IOC_Reg_Size); iommu->cmd_regs = iommu->xlate_regs + IOC_IOCmd_Offset; segments = size >> IO_SEGMENT_SHIFT; pages_per_segment = 1ull << IO_PAGENO_BITS; pr_debug("%s: iommu[%d]: segments: %lu, pages per segment: %lu\n", __FUNCTION__, iommu->nid, segments, pages_per_segment); /* set up the segment table */ page = alloc_pages_node(iommu->nid, GFP_KERNEL, 0); BUG_ON(!page); iommu->stab = page_address(page); clear_page(iommu->stab); /* ... and the page tables. Since these are contiguous, we can treat * the page tables as one array of ptes, like pSeries does. */ ptab_size = segments * pages_per_segment * sizeof(unsigned long); pr_debug("%s: iommu[%d]: ptab_size: %lu, order: %d\n", __FUNCTION__, iommu->nid, ptab_size, get_order(ptab_size)); page = alloc_pages_node(iommu->nid, GFP_KERNEL, get_order(ptab_size)); BUG_ON(!page); iommu->ptab = page_address(page); memset(iommu->ptab, 0, ptab_size); /* allocate a bogus page for the end of each mapping */ page = alloc_pages_node(iommu->nid, GFP_KERNEL, 0); BUG_ON(!page); iommu->pad_page = page_address(page); clear_page(iommu->pad_page); /* number of pages needed for a page table */ n_pte_pages = (pages_per_segment * sizeof(unsigned long)) >> IOMMU_PAGE_SHIFT; pr_debug("%s: iommu[%d]: stab at %p, ptab at %p, n_pte_pages: %lu\n", __FUNCTION__, iommu->nid, iommu->stab, iommu->ptab, n_pte_pages); /* initialise the STEs */ reg = IOSTE_V | ((n_pte_pages - 1) << 5); if (IOMMU_PAGE_SIZE == 0x1000) reg |= IOSTE_PS_4K; else if (IOMMU_PAGE_SIZE == 0x10000) reg |= IOSTE_PS_64K; else { extern void __unknown_page_size_error(void); __unknown_page_size_error(); } pr_debug("Setting up IOMMU stab:\n"); for (i = 0; i * (1ul << IO_SEGMENT_SHIFT) < size; i++) { iommu->stab[i] = reg |
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -