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

📄 pci.c

📁 ARM 嵌入式 系统 设计与实例开发 实验教材 二源码
💻 C
📖 第 1 页 / 共 3 页
字号:
/* * BK Id: SCCS/s.pci.c 1.40 01/25/02 15:15:24 benh *//* * Common pmac/prep/chrp pci routines. -- Cort */#include <linux/config.h>#include <linux/kernel.h>#include <linux/pci.h>#include <linux/delay.h>#include <linux/string.h>#include <linux/init.h>#include <linux/capability.h>#include <linux/sched.h>#include <linux/errno.h>#include <linux/bootmem.h>#include <asm/processor.h>#include <asm/io.h>#include <asm/prom.h>#include <asm/sections.h>#include <asm/pci-bridge.h>#include <asm/residual.h>#include <asm/byteorder.h>#include <asm/irq.h>#include <asm/gg2.h>#include <asm/uaccess.h>#include "pci.h"#define DEBUG#ifdef DEBUG#define DBG(x...) printk(x)#else#define DBG(x...)#endifunsigned long isa_io_base     = 0;unsigned long isa_mem_base    = 0;unsigned long pci_dram_offset = 0;void pcibios_make_OF_bus_map(void);static void pcibios_fixup_resources(struct pci_dev* dev);static void fixup_broken_pcnet32(struct pci_dev* dev);static void fixup_rev1_53c810(struct pci_dev* dev);#ifdef CONFIG_ALL_PPCstatic void pcibios_fixup_cardbus(struct pci_dev* dev);static u8* pci_to_OF_bus_map;#endif/* By default, we don't re-assign bus numbers. We do this only on * some pmacs */int pci_assign_all_busses;struct pci_controller* hose_head;struct pci_controller** hose_tail = &hose_head;static int pci_bus_count;struct pci_fixup pcibios_fixups[] = {	{ PCI_FIXUP_HEADER,	PCI_VENDOR_ID_TRIDENT,	PCI_ANY_ID,			fixup_broken_pcnet32 },	{ PCI_FIXUP_HEADER,	PCI_VENDOR_ID_NCR,	PCI_DEVICE_ID_NCR_53C810,	fixup_rev1_53c810 },	{ PCI_FIXUP_HEADER,	PCI_ANY_ID,		PCI_ANY_ID,			pcibios_fixup_resources },#ifdef CONFIG_ALL_PPC	/* We should add per-machine fixup support in xxx_setup.c or xxx_pci.c */	{ PCI_FIXUP_FINAL,	PCI_VENDOR_ID_TI,	PCI_ANY_ID,			pcibios_fixup_cardbus }, #endif /* CONFIG_ALL_PPC */ 	{ 0 }};static voidfixup_rev1_53c810(struct pci_dev* dev){	/* rev 1 ncr53c810 chips don't set the class at all which means	 * they don't get their resources remapped. Fix that here.	 */	if ((dev->class == PCI_CLASS_NOT_DEFINED)) {		printk("NCR 53c810 rev 1 detected, setting PCI class.\n");		dev->class = PCI_CLASS_STORAGE_SCSI;	}}static voidfixup_broken_pcnet32(struct pci_dev* dev){	if ((dev->class>>8 == PCI_CLASS_NETWORK_ETHERNET)) {		dev->vendor = PCI_VENDOR_ID_AMD;		pci_write_config_word(dev, PCI_VENDOR_ID, PCI_VENDOR_ID_AMD);		pci_name_device(dev);	}}voidpcibios_update_resource(struct pci_dev *dev, struct resource *root,			     struct resource *res, int resource){	u32 new, check;	int reg;	struct pci_controller* hose = dev->sysdata;	unsigned long io_offset;		new = res->start;	if (hose && res->flags & IORESOURCE_IO) {		io_offset = (unsigned long)hose->io_base_virt - isa_io_base;		new -= io_offset;	}	if (hose && res->flags & IORESOURCE_MEM)		new -= hose->pci_mem_offset;	new |= (res->flags & PCI_REGION_FLAG_MASK);	if (resource < 6) {		reg = PCI_BASE_ADDRESS_0 + 4*resource;	} else if (resource == PCI_ROM_RESOURCE) {		res->flags |= PCI_ROM_ADDRESS_ENABLE;		reg = dev->rom_base_reg;	} else {		/* Somebody might have asked allocation of a non-standard resource */		return;	}	pci_write_config_dword(dev, reg, new);	pci_read_config_dword(dev, reg, &check);	if ((new ^ check) & ((new & PCI_BASE_ADDRESS_SPACE_IO) ? PCI_BASE_ADDRESS_IO_MASK : PCI_BASE_ADDRESS_MEM_MASK)) {		printk(KERN_ERR "PCI: Error while updating region "		       "%s/%d (%08x != %08x)\n", dev->slot_name, resource,		       new, check);	}}static voidpcibios_fixup_resources(struct pci_dev *dev){	struct pci_controller* hose = (struct pci_controller *)dev->sysdata;	int i;	unsigned long offset;	if (!hose) {		printk(KERN_ERR "No hose for PCI dev %s!\n", dev->slot_name);		return;	}	for (i = 0; i < DEVICE_COUNT_RESOURCE; i++) {		struct resource *res = dev->resource + i;		if (!res->start || !res->flags)			continue;		if (res->end == 0xffffffff) {			DBG("PCI:%s Resource %d [%08lx-%08lx] is unassigned\n",			    dev->slot_name, i, res->start, res->end);			res->end -= res->start;			res->start = 0;			continue;		}		offset = 0;		if (res->flags & IORESOURCE_MEM) {			offset = hose->pci_mem_offset;		} else if (res->flags & IORESOURCE_IO) {			offset = (unsigned long) hose->io_base_virt				- isa_io_base;		}		if (offset != 0) {			res->start += offset;			res->end += offset;#ifdef DEBUG			printk("Fixup res %d (%lx) of dev %s: %lx -> %lx\n",			       i, res->flags, dev->slot_name,			       res->start - offset, res->start);#endif		}	}}#ifdef CONFIG_ALL_PPCstatic voidpcibios_fixup_cardbus(struct pci_dev* dev){	if (_machine != _MACH_Pmac)		return;	/*	 * Fix the interrupt routing on the TI1211 chip on the 1999	 * G3 powerbook, which doesn't get initialized properly by OF.	 * Same problem with the 1410 of the new titanium pbook which	 * has the same register.	 */	if (dev->vendor == PCI_VENDOR_ID_TI	    && (dev->device == PCI_DEVICE_ID_TI_1211 ||	        dev->device == PCI_DEVICE_ID_TI_1410)) {		u8 val;		/* 0x8c == TI122X_IRQMUX, 2 says to route the INTA		   signal out the MFUNC0 pin */		if (pci_read_config_byte(dev, 0x8c, &val) == 0)			pci_write_config_byte(dev, 0x8c, (val & ~0x0f) | 2);		/* Disable ISA interrupt mode */			if (pci_read_config_byte(dev, 0x92, &val) == 0)			pci_write_config_byte(dev, 0x92, val & ~0x06);	}}#endif /* CONFIG_ALL_PPC *//* * We need to avoid collisions with `mirrored' VGA ports * and other strange ISA hardware, so we always want the * addresses to be allocated in the 0x000-0x0ff region * modulo 0x400. * * Why? Because some silly external IO cards only decode * the low 10 bits of the IO address. The 0x00-0xff region * is reserved for motherboard devices that decode all 16 * bits, so it's ok to allocate at, say, 0x2800-0x28ff, * but we want to try to avoid allocating at 0x2900-0x2bff * which might have be mirrored at 0x0100-0x03ff.. */voidpcibios_align_resource(void *data, struct resource *res, unsigned long size){	struct pci_dev *dev = data;	if (res->flags & IORESOURCE_IO) {		unsigned long start = res->start;		if (size > 0x100) {			printk(KERN_ERR "PCI: I/O Region %s/%d too large"			       " (%ld bytes)\n", dev->slot_name,			       dev->resource - res, size);		}		if (start & 0x300) {			start = (start + 0x3ff) & ~0x3ff;			res->start = start;		}	}}/* *  Handle resources of PCI devices.  If the world were perfect, we could *  just allocate all the resource regions and do nothing more.  It isn't. *  On the other hand, we cannot just re-allocate all devices, as it would *  require us to know lots of host bridge internals.  So we attempt to *  keep as much of the original configuration as possible, but tweak it *  when it's found to be wrong. * *  Known BIOS problems we have to work around: *	- I/O or memory regions not configured *	- regions configured, but not enabled in the command register *	- bogus I/O addresses above 64K used *	- expansion ROMs left enabled (this may sound harmless, but given *	  the fact the PCI specs explicitly allow address decoders to be *	  shared between expansion ROMs and other resource regions, it's *	  at least dangerous) * *  Our solution: *	(1) Allocate resources for all buses behind PCI-to-PCI bridges. *	    This gives us fixed barriers on where we can allocate. *	(2) Allocate resources for all enabled devices.  If there is *	    a collision, just mark the resource as unallocated. Also *	    disable expansion ROMs during this step. *	(3) Try to allocate resources for disabled devices.  If the *	    resources were assigned correctly, everything goes well, *	    if they weren't, they won't disturb allocation of other *	    resources. *	(4) Assign new addresses to resources which were either *	    not configured at all or misconfigured.  If explicitly *	    requested by the user, configure expansion ROM address *	    as well. */static void __initpcibios_allocate_bus_resources(struct list_head *bus_list){	struct list_head *ln;	struct pci_bus *bus;	int i;	struct resource *res, *pr;	/* Depth-First Search on bus tree */	for (ln = bus_list->next; ln != bus_list; ln=ln->next) {		bus = pci_bus_b(ln);		for (i = 0; i < 4; ++i) {			if ((res = bus->resource[i]) == NULL || !res->flags)				continue;			if (bus->parent == NULL)				pr = (res->flags & IORESOURCE_IO)?					&ioport_resource: &iomem_resource;			else {				pr = pci_find_parent_resource(bus->self, res);				if (pr == res) {					/* this happens when the generic PCI					 * code (wrongly) decides that this					 * bridge is transparent  -- paulus					 */					continue;				}			}						if (pr && request_resource(pr, res) == 0)				continue;			printk(KERN_ERR "PCI: Cannot allocate resource region "			       "%d of PCI bridge %d\n", i, bus->number);			DBG("PCI: resource is %lx..%lx (%lx), parent %p\n",			    res->start, res->end, res->flags, pr);		}		pcibios_allocate_bus_resources(&bus->children);	}}static inline void alloc_resource(struct pci_dev *dev, int idx){	struct resource *pr, *r = &dev->resource[idx];	DBG("PCI:%s: Resource %d: %08lx-%08lx (f=%lx)\n",	    dev->slot_name, idx, r->start, r->end, r->flags);	pr = pci_find_parent_resource(dev, r);	if (!pr || request_resource(pr, r) < 0) {		printk(KERN_ERR "PCI: Cannot allocate resource region %d"		       " of device %s\n", idx, dev->slot_name);		if (pr)			DBG("PCI:  parent is %p: %08lx-%08lx (f=%lx)\n",			    pr, pr->start, pr->end, pr->flags);		/* We'll assign a new address later */		r->end -= r->start;		r->start = 0;	}}static void __initpcibios_allocate_resources(int pass){	struct pci_dev *dev;	int idx, disabled;	u16 command;	struct resource *r;	pci_for_each_dev(dev) {		pci_read_config_word(dev, PCI_COMMAND, &command);		for (idx = 0; idx < 6; idx++) {			r = &dev->resource[idx];			if (r->parent)		/* Already allocated */				continue;			if (!r->start)		/* Not assigned at all */				continue;			if (r->flags & IORESOURCE_IO)				disabled = !(command & PCI_COMMAND_IO);			else				disabled = !(command & PCI_COMMAND_MEMORY);			if (pass == disabled)				alloc_resource(dev, idx);		}		if (pass)			continue;		r = &dev->resource[PCI_ROM_RESOURCE];		if (r->flags & PCI_ROM_ADDRESS_ENABLE) {			/* Turn the ROM off, leave the resource region, but keep it unregistered. */			u32 reg;			DBG("PCI: Switching off ROM of %s\n", dev->slot_name);			r->flags &= ~PCI_ROM_ADDRESS_ENABLE;			pci_read_config_dword(dev, dev->rom_base_reg, &reg);			pci_write_config_dword(dev, dev->rom_base_reg,					       reg & ~PCI_ROM_ADDRESS_ENABLE);		}	}}static void __initpcibios_assign_resources(void){	struct pci_dev *dev;	int idx;	struct resource *r;	pci_for_each_dev(dev) {		int class = dev->class >> 8;		/* Don't touch classless devices and host bridges */		if (!class || class == PCI_CLASS_BRIDGE_HOST)			continue;		for (idx = 0; idx < 6; idx++) {			r = &dev->resource[idx];			/*			 * We shall assign a new address to this resource,			 * either because the BIOS (sic) forgot to do so			 * or because we have decided the old address was			 * unusable for some reason.			 */			if (!r->start && r->end &&			    (!ppc_md.pcibios_enable_device_hook ||			     !ppc_md.pcibios_enable_device_hook(dev, 1)))				pci_assign_resource(dev, idx);		}#if 0 /* don't assign ROMs */		r = &dev->resource[PCI_ROM_RESOURCE];		r->end -= r->start;		r->start = 0;		if (r->end)			pci_assign_resource(dev, PCI_ROM_RESOURCE);#endif	}}intpcibios_enable_resources(struct pci_dev *dev){	u16 cmd, old_cmd;	int idx;	struct resource *r;	pci_read_config_word(dev, PCI_COMMAND, &cmd);	old_cmd = cmd;	for(idx=0; idx<6; idx++) {		r = &dev->resource[idx];		if (!r->start && r->end) {			printk(KERN_ERR "PCI: Device %s not available because of resource collisions\n", dev->slot_name);			return -EINVAL;		}		if (r->flags & IORESOURCE_IO)			cmd |= PCI_COMMAND_IO;		if (r->flags & IORESOURCE_MEM)			cmd |= PCI_COMMAND_MEMORY;	}	if (dev->resource[PCI_ROM_RESOURCE].start)		cmd |= PCI_COMMAND_MEMORY;	if (cmd != old_cmd) {

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -