⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 iommu.c

📁 linux内核源码
💻 C
📖 第 1 页 / 共 2 页
字号:
/* * 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 + -