亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频

? 歡迎來到蟲蟲下載站! | ?? 資源下載 ?? 資源專輯 ?? 關于我們
? 蟲蟲下載站

?? io-unit.c

?? 底層驅動開發
?? C
字號:
/* $Id: io-unit.c,v 1.24 2001/12/17 07:05:09 davem Exp $ * io-unit.c:  IO-UNIT specific routines for memory management. * * Copyright (C) 1997,1998 Jakub Jelinek    (jj@sunsite.mff.cuni.cz) */ #include <linux/config.h>#include <linux/kernel.h>#include <linux/init.h>#include <linux/slab.h>#include <linux/spinlock.h>#include <linux/mm.h>#include <linux/highmem.h>	/* pte_offset_map => kmap_atomic */#include <linux/bitops.h>#include <asm/scatterlist.h>#include <asm/pgalloc.h>#include <asm/pgtable.h>#include <asm/sbus.h>#include <asm/io.h>#include <asm/io-unit.h>#include <asm/mxcc.h>#include <asm/cacheflush.h>#include <asm/tlbflush.h>#include <asm/dma.h>/* #define IOUNIT_DEBUG */#ifdef IOUNIT_DEBUG#define IOD(x) printk(x)#else#define IOD(x) do { } while (0)#endif#define IOPERM        (IOUPTE_CACHE | IOUPTE_WRITE | IOUPTE_VALID)#define MKIOPTE(phys) __iopte((((phys)>>4) & IOUPTE_PAGE) | IOPERM)void __initiounit_init(int sbi_node, int io_node, struct sbus_bus *sbus){	iopte_t *xpt, *xptend;	struct iounit_struct *iounit;	struct linux_prom_registers iommu_promregs[PROMREG_MAX];	struct resource r;	iounit = kmalloc(sizeof(struct iounit_struct), GFP_ATOMIC);	memset(iounit, 0, sizeof(*iounit));	iounit->limit[0] = IOUNIT_BMAP1_START;	iounit->limit[1] = IOUNIT_BMAP2_START;	iounit->limit[2] = IOUNIT_BMAPM_START;	iounit->limit[3] = IOUNIT_BMAPM_END;	iounit->rotor[1] = IOUNIT_BMAP2_START;	iounit->rotor[2] = IOUNIT_BMAPM_START;	xpt = NULL;	if(prom_getproperty(sbi_node, "reg", (void *) iommu_promregs,			    sizeof(iommu_promregs)) != -1) {		prom_apply_generic_ranges(io_node, 0, iommu_promregs, 3);		memset(&r, 0, sizeof(r));		r.flags = iommu_promregs[2].which_io;		r.start = iommu_promregs[2].phys_addr;		xpt = (iopte_t *) sbus_ioremap(&r, 0, PAGE_SIZE * 16, "XPT");	}	if(!xpt) panic("Cannot map External Page Table.");		sbus->iommu = (struct iommu_struct *)iounit;	iounit->page_table = xpt;		for (xptend = iounit->page_table + (16 * PAGE_SIZE) / sizeof(iopte_t);	     xpt < xptend;)	     	iopte_val(*xpt++) = 0;}/* One has to hold iounit->lock to call this */static unsigned long iounit_get_area(struct iounit_struct *iounit, unsigned long vaddr, int size){	int i, j, k, npages;	unsigned long rotor, scan, limit;	iopte_t iopte;        npages = ((vaddr & ~PAGE_MASK) + size + (PAGE_SIZE-1)) >> PAGE_SHIFT;	/* A tiny bit of magic ingredience :) */	switch (npages) {	case 1: i = 0x0231; break;	case 2: i = 0x0132; break;	default: i = 0x0213; break;	}		IOD(("iounit_get_area(%08lx,%d[%d])=", vaddr, size, npages));	next:	j = (i & 15);	rotor = iounit->rotor[j - 1];	limit = iounit->limit[j];	scan = rotor;nexti:	scan = find_next_zero_bit(iounit->bmap, limit, scan);	if (scan + npages > limit) {		if (limit != rotor) {			limit = rotor;			scan = iounit->limit[j - 1];			goto nexti;		}		i >>= 4;		if (!(i & 15))			panic("iounit_get_area: Couldn't find free iopte slots for (%08lx,%d)\n", vaddr, size);		goto next;	}	for (k = 1, scan++; k < npages; k++)		if (test_bit(scan++, iounit->bmap))			goto nexti;	iounit->rotor[j - 1] = (scan < limit) ? scan : iounit->limit[j - 1];	scan -= npages;	iopte = MKIOPTE(__pa(vaddr & PAGE_MASK));	vaddr = IOUNIT_DMA_BASE + (scan << PAGE_SHIFT) + (vaddr & ~PAGE_MASK);	for (k = 0; k < npages; k++, iopte = __iopte(iopte_val(iopte) + 0x100), scan++) {		set_bit(scan, iounit->bmap);		iounit->page_table[scan] = iopte;	}	IOD(("%08lx\n", vaddr));	return vaddr;}static __u32 iounit_get_scsi_one(char *vaddr, unsigned long len, struct sbus_bus *sbus){	unsigned long ret, flags;	struct iounit_struct *iounit = (struct iounit_struct *)sbus->iommu;		spin_lock_irqsave(&iounit->lock, flags);	ret = iounit_get_area(iounit, (unsigned long)vaddr, len);	spin_unlock_irqrestore(&iounit->lock, flags);	return ret;}static void iounit_get_scsi_sgl(struct scatterlist *sg, int sz, struct sbus_bus *sbus){	unsigned long flags;	struct iounit_struct *iounit = (struct iounit_struct *)sbus->iommu;	/* FIXME: Cache some resolved pages - often several sg entries are to the same page */	spin_lock_irqsave(&iounit->lock, flags);	while (sz != 0) {		--sz;		sg[sz].dvma_address = iounit_get_area(iounit, (unsigned long)page_address(sg[sz].page) + sg[sz].offset, sg[sz].length);		sg[sz].dvma_length = sg[sz].length;	}	spin_unlock_irqrestore(&iounit->lock, flags);}static void iounit_release_scsi_one(__u32 vaddr, unsigned long len, struct sbus_bus *sbus){	unsigned long flags;	struct iounit_struct *iounit = (struct iounit_struct *)sbus->iommu;		spin_lock_irqsave(&iounit->lock, flags);	len = ((vaddr & ~PAGE_MASK) + len + (PAGE_SIZE-1)) >> PAGE_SHIFT;	vaddr = (vaddr - IOUNIT_DMA_BASE) >> PAGE_SHIFT;	IOD(("iounit_release %08lx-%08lx\n", (long)vaddr, (long)len+vaddr));	for (len += vaddr; vaddr < len; vaddr++)		clear_bit(vaddr, iounit->bmap);	spin_unlock_irqrestore(&iounit->lock, flags);}static void iounit_release_scsi_sgl(struct scatterlist *sg, int sz, struct sbus_bus *sbus){	unsigned long flags;	unsigned long vaddr, len;	struct iounit_struct *iounit = (struct iounit_struct *)sbus->iommu;	spin_lock_irqsave(&iounit->lock, flags);	while (sz != 0) {		--sz;		len = ((sg[sz].dvma_address & ~PAGE_MASK) + sg[sz].length + (PAGE_SIZE-1)) >> PAGE_SHIFT;		vaddr = (sg[sz].dvma_address - IOUNIT_DMA_BASE) >> PAGE_SHIFT;		IOD(("iounit_release %08lx-%08lx\n", (long)vaddr, (long)len+vaddr));		for (len += vaddr; vaddr < len; vaddr++)			clear_bit(vaddr, iounit->bmap);	}	spin_unlock_irqrestore(&iounit->lock, flags);}#ifdef CONFIG_SBUSstatic int iounit_map_dma_area(dma_addr_t *pba, unsigned long va, __u32 addr, int len){	unsigned long page, end;	pgprot_t dvma_prot;	iopte_t *iopte;	struct sbus_bus *sbus;	*pba = addr;	dvma_prot = __pgprot(SRMMU_CACHE | SRMMU_ET_PTE | SRMMU_PRIV);	end = PAGE_ALIGN((addr + len));	while(addr < end) {		page = va;		{			pgd_t *pgdp;			pmd_t *pmdp;			pte_t *ptep;			long i;			pgdp = pgd_offset(&init_mm, addr);			pmdp = pmd_offset(pgdp, addr);			ptep = pte_offset_map(pmdp, addr);			set_pte(ptep, mk_pte(virt_to_page(page), dvma_prot));						i = ((addr - IOUNIT_DMA_BASE) >> PAGE_SHIFT);			for_each_sbus(sbus) {				struct iounit_struct *iounit = (struct iounit_struct *)sbus->iommu;				iopte = (iopte_t *)(iounit->page_table + i);				*iopte = MKIOPTE(__pa(page));			}		}		addr += PAGE_SIZE;		va += PAGE_SIZE;	}	flush_cache_all();	flush_tlb_all();	return 0;}static void iounit_unmap_dma_area(unsigned long addr, int len){	/* XXX Somebody please fill this in */}/* XXX We do not pass sbus device here, bad. */static struct page *iounit_translate_dvma(unsigned long addr){	struct sbus_bus *sbus = sbus_root;	/* They are all the same */	struct iounit_struct *iounit = (struct iounit_struct *)sbus->iommu;	int i;	iopte_t *iopte;	i = ((addr - IOUNIT_DMA_BASE) >> PAGE_SHIFT);	iopte = (iopte_t *)(iounit->page_table + i);	return pfn_to_page(iopte_val(*iopte) >> (PAGE_SHIFT-4)); /* XXX sun4d guru, help */}#endifstatic char *iounit_lockarea(char *vaddr, unsigned long len){/* FIXME: Write this */	return vaddr;}static void iounit_unlockarea(char *vaddr, unsigned long len){/* FIXME: Write this */}void __init ld_mmu_iounit(void){	BTFIXUPSET_CALL(mmu_lockarea, iounit_lockarea, BTFIXUPCALL_RETO0);	BTFIXUPSET_CALL(mmu_unlockarea, iounit_unlockarea, BTFIXUPCALL_NOP);	BTFIXUPSET_CALL(mmu_get_scsi_one, iounit_get_scsi_one, BTFIXUPCALL_NORM);	BTFIXUPSET_CALL(mmu_get_scsi_sgl, iounit_get_scsi_sgl, BTFIXUPCALL_NORM);	BTFIXUPSET_CALL(mmu_release_scsi_one, iounit_release_scsi_one, BTFIXUPCALL_NORM);	BTFIXUPSET_CALL(mmu_release_scsi_sgl, iounit_release_scsi_sgl, BTFIXUPCALL_NORM);#ifdef CONFIG_SBUS	BTFIXUPSET_CALL(mmu_map_dma_area, iounit_map_dma_area, BTFIXUPCALL_NORM);	BTFIXUPSET_CALL(mmu_unmap_dma_area, iounit_unmap_dma_area, BTFIXUPCALL_NORM);	BTFIXUPSET_CALL(mmu_translate_dvma, iounit_translate_dvma, BTFIXUPCALL_NORM);#endif}__u32 iounit_map_dma_init(struct sbus_bus *sbus, int size){	int i, j, k, npages;	unsigned long rotor, scan, limit;	unsigned long flags;	__u32 ret;	struct iounit_struct *iounit = (struct iounit_struct *)sbus->iommu;        npages = (size + (PAGE_SIZE-1)) >> PAGE_SHIFT;	i = 0x0213;	spin_lock_irqsave(&iounit->lock, flags);next:	j = (i & 15);	rotor = iounit->rotor[j - 1];	limit = iounit->limit[j];	scan = rotor;nexti:	scan = find_next_zero_bit(iounit->bmap, limit, scan);	if (scan + npages > limit) {		if (limit != rotor) {			limit = rotor;			scan = iounit->limit[j - 1];			goto nexti;		}		i >>= 4;		if (!(i & 15))			panic("iounit_map_dma_init: Couldn't find free iopte slots for %d bytes\n", size);		goto next;	}	for (k = 1, scan++; k < npages; k++)		if (test_bit(scan++, iounit->bmap))			goto nexti;	iounit->rotor[j - 1] = (scan < limit) ? scan : iounit->limit[j - 1];	scan -= npages;	ret = IOUNIT_DMA_BASE + (scan << PAGE_SHIFT);	for (k = 0; k < npages; k++, scan++)		set_bit(scan, iounit->bmap);	spin_unlock_irqrestore(&iounit->lock, flags);	return ret;}__u32 iounit_map_dma_page(__u32 vaddr, void *addr, struct sbus_bus *sbus){	int scan = (vaddr - IOUNIT_DMA_BASE) >> PAGE_SHIFT;	struct iounit_struct *iounit = (struct iounit_struct *)sbus->iommu;		iounit->page_table[scan] = MKIOPTE(__pa(((unsigned long)addr) & PAGE_MASK));	return vaddr + (((unsigned long)addr) & ~PAGE_MASK);}

?? 快捷鍵說明

復制代碼 Ctrl + C
搜索代碼 Ctrl + F
全屏模式 F11
切換主題 Ctrl + Shift + D
顯示快捷鍵 ?
增大字號 Ctrl + =
減小字號 Ctrl + -
亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频
一个色综合av| 麻豆国产欧美日韩综合精品二区 | 五月婷婷激情综合网| 国产一区二区三区免费| 欧美三电影在线| 中文字幕第一页久久| 蜜桃精品视频在线| 欧美女孩性生活视频| 国产精品国模大尺度视频| 精品一区二区三区影院在线午夜 | 国产99久久久国产精品潘金网站| 精品久久人人做人人爱| 亚洲影视在线播放| av电影在线观看完整版一区二区| 欧美刺激午夜性久久久久久久| 亚洲视频免费观看| www.欧美日韩国产在线| 国产午夜精品一区二区三区四区| 蜜臀av一区二区在线免费观看| 欧美日韩视频不卡| 亚洲国产日韩精品| 欧美日韩国产高清一区二区三区 | a在线欧美一区| 欧美激情一区不卡| 成人影视亚洲图片在线| 国产午夜精品福利| 成人免费毛片app| 国产精品入口麻豆原神| 成人av网站免费| 国产精品久久久久久久久图文区 | 91 com成人网| 日日夜夜精品视频免费| 欧美日韩视频专区在线播放| 香蕉成人伊视频在线观看| 欧美午夜理伦三级在线观看| 亚洲成人一区在线| 日韩一级黄色大片| 国产曰批免费观看久久久| 久久伊人蜜桃av一区二区| 国产乱淫av一区二区三区| 日本一区二区免费在线观看视频| 不卡视频在线观看| 亚洲最新在线观看| 91精品国产综合久久精品图片| 日本不卡视频在线| 久久久久久久久久久99999| 国产一区二区视频在线播放| 国产日韩欧美精品综合| 91视频免费看| 日韩在线卡一卡二| 久久久噜噜噜久久中文字幕色伊伊| 国产成人综合亚洲91猫咪| 亚洲欧洲另类国产综合| 色屁屁一区二区| 免费在线观看一区| 中文一区二区完整视频在线观看| 99国产精品国产精品久久| 亚洲乱码精品一二三四区日韩在线 | 99re这里只有精品6| 亚洲美女一区二区三区| 91精品国产综合久久久蜜臀图片| 国产综合色在线视频区| 亚洲人成电影网站色mp4| 欧美一级黄色片| 丁香六月久久综合狠狠色| 亚洲国产精品久久人人爱| 精品捆绑美女sm三区| 99国产一区二区三精品乱码| 天天色天天操综合| 中文字幕在线观看一区| 91精品国产麻豆国产自产在线| 国产成人精品网址| 亚洲最新在线观看| 国产精品欧美一级免费| 日韩一区二区三区av| av动漫一区二区| 国产精品一区二区在线观看不卡| 亚洲午夜在线电影| 国产精品天天看| 精品国产一二三| 欧美日韩亚洲综合在线 欧美亚洲特黄一级| 精品在线观看免费| 亚洲一级二级三级在线免费观看| 国产校园另类小说区| 欧美一区二区精美| 色悠悠久久综合| 成人蜜臀av电影| 激情六月婷婷综合| 天堂精品中文字幕在线| 亚洲另类一区二区| 中文一区二区完整视频在线观看| 777a∨成人精品桃花网| 色综合av在线| 91污片在线观看| 成人午夜伦理影院| 国产一区二区三区观看| 裸体歌舞表演一区二区| 亚洲高清免费观看| 亚洲中国最大av网站| 亚洲国产精华液网站w| 亚洲精品在线观| 日韩女优制服丝袜电影| 91.com在线观看| 制服丝袜成人动漫| 这里只有精品99re| 欧美伦理影视网| 欧美亚洲一区二区三区四区| 99视频有精品| 色婷婷综合五月| 色天天综合色天天久久| 色综合天天视频在线观看| 一本久久a久久精品亚洲 | 成人激情免费视频| 国产精品77777竹菊影视小说| 麻豆精品久久久| 国内精品视频一区二区三区八戒| 日本午夜精品视频在线观看| 日本午夜精品视频在线观看| 日韩成人午夜精品| 狂野欧美性猛交blacked| 精品一二线国产| 国产一区不卡在线| 成人精品鲁一区一区二区| 不卡的av中国片| 99精品国产一区二区三区不卡| 日本精品一区二区三区高清| 91久久精品一区二区二区| 欧美视频一二三区| 在线播放欧美女士性生活| 欧美大胆一级视频| 日本一区二区三区在线不卡| 亚洲欧美在线高清| 午夜精品久久久久久久 | 国产人成一区二区三区影院| 国产亲近乱来精品视频| 国产精品久久久久久久久图文区| 亚洲欧美综合网| 亚洲高清免费在线| 韩国视频一区二区| 99国产精品国产精品毛片| 欧美巨大另类极品videosbest | 国产黄色精品视频| 99久久综合国产精品| 欧美日韩国产不卡| 国产日本欧美一区二区| 亚洲自拍偷拍欧美| 麻豆成人久久精品二区三区红 | 在线播放91灌醉迷j高跟美女| 亚洲精品一区二区三区四区高清 | 亚洲特黄一级片| 天堂久久久久va久久久久| 高清成人在线观看| 欧美日本一区二区三区| 日本一区二区久久| 天堂久久一区二区三区| www.欧美日韩国产在线| 日韩欧美激情在线| 尤物在线观看一区| 国产一区二区三区在线观看免费视频| 色欲综合视频天天天| 久久精品视频在线看| 日韩精品电影在线| 99v久久综合狠狠综合久久| 欧美精品一区男女天堂| 亚洲最色的网站| 成人av高清在线| 欧美精品一区二区不卡| 亚洲国产日韩综合久久精品| 高清视频一区二区| 2021国产精品久久精品| 婷婷一区二区三区| 色综合久久久久| 国产精品乱码久久久久久| 美女精品自拍一二三四| 欧美肥妇毛茸茸| 亚洲激情校园春色| 99视频在线精品| 亚洲国产精品高清| 国产一区二区不卡老阿姨| 在线不卡一区二区| 亚洲综合色成人| 色综合久久88色综合天天6| 久久久无码精品亚洲日韩按摩| 青青国产91久久久久久| 欧美理论电影在线| 午夜伦欧美伦电影理论片| 一本色道久久综合亚洲aⅴ蜜桃 | 国产69精品一区二区亚洲孕妇| 日韩欧美成人激情| 另类的小说在线视频另类成人小视频在线| 91女神在线视频| 亚洲视频一二区| 99久久精品国产毛片| 亚洲素人一区二区| 91香蕉国产在线观看软件| 亚洲欧美激情小说另类| www.一区二区| 亚洲精品伦理在线| 在线免费观看日本欧美| 亚洲一二三四区不卡|