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

📄 pci.c

📁 powerpc内核mpc8241linux系统下pci驱动程序
💻 C
字号:
/* *	$Id: pci.c,v 1.91 1999/01/21 13:34:01 davem Exp $ * *	PCI Bus Services, see include/linux/pci.h for further explanation. * *	Copyright 1993 -- 1997 Drew Eckhardt, Frederic Potter, *	David Mosberger-Tang * *	Copyright 1997 -- 1999 Martin Mares <mj@atrey.karlin.mff.cuni.cz> */#include <linux/config.h>#include <linux/types.h>#include <linux/kernel.h>#include <linux/pci.h>#include <linux/string.h>#include <linux/init.h>#include <linux/malloc.h>#include <asm/page.h>#undef DEBUG#ifdef DEBUG#define DBG(x...) printk(x)#else#define DBG(x...)#endifstruct pci_bus pci_root;struct pci_dev *pci_devices = NULL;static struct pci_dev **pci_last_dev_p = &pci_devices;static int pci_reverse __initdata = 0;struct pci_dev *pci_find_slot(unsigned int bus, unsigned int devfn){	struct pci_dev *dev;	for(dev=pci_devices; dev; dev=dev->next)		if (dev->bus->number == bus && dev->devfn == devfn)			break;	return dev;}struct pci_dev *pci_find_device(unsigned int vendor, unsigned int device, struct pci_dev *from){	if (!from)		from = pci_devices;	else		from = from->next;	while (from && (from->vendor != vendor || from->device != device))		from = from->next;	return from;}struct pci_dev *pci_find_class(unsigned int class, struct pci_dev *from){	if (!from)		from = pci_devices;	else		from = from->next;	while (from && from->class != class)		from = from->next;	return from;}intpci_read_config_byte(struct pci_dev *dev, u8 where, u8 *val){	return pcibios_read_config_byte(dev->bus->number, dev->devfn, where, val);}intpci_read_config_word(struct pci_dev *dev, u8 where, u16 *val){	return pcibios_read_config_word(dev->bus->number, dev->devfn, where, val);}intpci_read_config_dword(struct pci_dev *dev, u8 where, u32 *val){	return pcibios_read_config_dword(dev->bus->number, dev->devfn, where, val);}intpci_write_config_byte(struct pci_dev *dev, u8 where, u8 val){	return pcibios_write_config_byte(dev->bus->number, dev->devfn, where, val);}intpci_write_config_word(struct pci_dev *dev, u8 where, u16 val){	return pcibios_write_config_word(dev->bus->number, dev->devfn, where, val);}intpci_write_config_dword(struct pci_dev *dev, u8 where, u32 val){	return pcibios_write_config_dword(dev->bus->number, dev->devfn, where, val);}voidpci_set_master(struct pci_dev *dev){	u16 cmd;	u8 lat;	pci_read_config_word(dev, PCI_COMMAND, &cmd);	if (! (cmd & PCI_COMMAND_MASTER)) {		printk("PCI: Enabling bus mastering for device %02x:%02x\n",			dev->bus->number, dev->devfn);		cmd |= PCI_COMMAND_MASTER;		pci_write_config_word(dev, PCI_COMMAND, cmd);	}	pci_read_config_byte(dev, PCI_LATENCY_TIMER, &lat);	if (lat < 16) {		printk("PCI: Increasing latency timer of device %02x:%02x to 64\n",			dev->bus->number, dev->devfn);		pci_write_config_byte(dev, PCI_LATENCY_TIMER, 64);	}}__initfunc(void pci_read_bases(struct pci_dev *dev, unsigned int howmany)){	unsigned int reg;	u32 l;	for(reg=0; reg<howmany; reg++) {		pci_read_config_dword(dev, PCI_BASE_ADDRESS_0 + (reg << 2), &l);		if (l == 0xffffffff)			continue;		dev->base_address[reg] = l;		if ((l & (PCI_BASE_ADDRESS_SPACE | PCI_BASE_ADDRESS_MEM_TYPE_MASK))		    == (PCI_BASE_ADDRESS_SPACE_MEMORY | PCI_BASE_ADDRESS_MEM_TYPE_64)) {			reg++;			pci_read_config_dword(dev, PCI_BASE_ADDRESS_0 + (reg << 2), &l);			if (l) {#if BITS_PER_LONG == 64				dev->base_address[reg-1] |= ((unsigned long) l) << 32;#else				printk("PCI: Unable to handle 64-bit address for device %02x:%02x\n",					dev->bus->number, dev->devfn);				dev->base_address[reg-1] = 0;#endif			}		}	}}__initfunc(unsigned int pci_scan_bus(struct pci_bus *bus)){          	unsigned int devfn, l, max, class;	unsigned char cmd, irq, tmp, hdr_type, is_multi = 0;	struct pci_dev *dev, **bus_last;	struct pci_bus *child;   	DBG("pci_scan_bus for bus %d\n", bus->number);	bus_last = &bus->devices;	max = bus->secondary;	for (devfn = 0; devfn < 0xff; ++devfn) {		if (PCI_FUNC(devfn) && !is_multi) {			/* not a multi-function device */			continue;		}		if (pcibios_read_config_byte(bus->number, devfn, PCI_HEADER_TYPE, &hdr_type))			continue;		if (!PCI_FUNC(devfn))			is_multi = hdr_type & 0x80;		if (pcibios_read_config_dword(bus->number, devfn, PCI_VENDOR_ID, &l) ||		    /* some broken boards return 0 if a slot is empty: */		    l == 0xffffffff || l == 0x00000000 || l == 0x0000ffff || l == 0xffff0000) {			is_multi = 0;			continue;		}		dev = kmalloc(sizeof(*dev), GFP_ATOMIC);		memset(dev, 0, sizeof(*dev));		dev->bus = bus;		dev->devfn  = devfn;		dev->vendor = l & 0xffff;		dev->device = (l >> 16) & 0xffff;		/* non-destructively determine if device can be a master: */#if defined(CONFIG_OCTOPUS) /* for octopus, RTL8139 could not be master */	if (!(dev->vendor==0x10ec && dev->device==0x8129)){#endif		pcibios_read_config_byte(bus->number, devfn, PCI_COMMAND, &cmd);		pcibios_write_config_byte(bus->number, devfn, PCI_COMMAND, cmd | PCI_COMMAND_MASTER);		pcibios_read_config_byte(bus->number, devfn, PCI_COMMAND, &tmp);		dev->master = ((tmp & PCI_COMMAND_MASTER) != 0);		pcibios_write_config_byte(bus->number, devfn, PCI_COMMAND, cmd);#if defined(CONFIG_OCTOPUS)	} else {		dev->master = 0;	}#endif		pcibios_read_config_dword(bus->number, devfn, PCI_CLASS_REVISION, &class);		class >>= 8;				    /* upper 3 bytes */		dev->class = class;		class >>= 8;		dev->hdr_type = hdr_type;		switch (hdr_type & 0x7f) {		    /* header type */		case PCI_HEADER_TYPE_NORMAL:		    /* standard header */			if (class == PCI_CLASS_BRIDGE_PCI)				goto bad;			/*			 * If the card generates interrupts, read IRQ number			 * (some architectures change it during pcibios_fixup())			 */			pcibios_read_config_byte(bus->number, dev->devfn, PCI_INTERRUPT_PIN, &irq);			if (irq)				pcibios_read_config_byte(bus->number, dev->devfn, PCI_INTERRUPT_LINE, &irq);			dev->irq = irq;			/*			 * read base address registers, again pcibios_fixup() can			 * tweak these			 */			pci_read_bases(dev, 6);			pcibios_read_config_dword(bus->number, devfn, PCI_ROM_ADDRESS, &l);			dev->rom_address = (l == 0xffffffff) ? 0 : l;			break;		case PCI_HEADER_TYPE_BRIDGE:		    /* bridge header */			if (class != PCI_CLASS_BRIDGE_PCI)				goto bad;			pci_read_bases(dev, 2);			pcibios_read_config_dword(bus->number, devfn, PCI_ROM_ADDRESS1, &l);			dev->rom_address = (l == 0xffffffff) ? 0 : l;			break;		case PCI_HEADER_TYPE_CARDBUS:		    /* CardBus bridge header */			if (class != PCI_CLASS_BRIDGE_CARDBUS)				goto bad;			pci_read_bases(dev, 1);			break;		default:				    /* unknown header */		bad:			printk(KERN_ERR "PCI: %02x:%02x [%04x/%04x/%06x] has unknown header type %02x, ignoring.\n",			       bus->number, dev->devfn, dev->vendor, dev->device, class, hdr_type);			continue;		}		DBG("PCI: %02x:%02x [%04x/%04x]\n", bus->number, dev->devfn, dev->vendor, dev->device);		#if 0	/* iMALL: for debug purpose, DennisK & sjzhu*/		printk("bus=%x,dvf=%x,vdr=%x,dvc=%x,cls=%x,htp=%x,bdr[0]=%x.\n", 				dev->bus->number, dev->devfn, dev->vendor, 				dev->device, dev->class, dev->hdr_type, 				dev->base_address[0]);#endif		/*		 * Put it into the global PCI device chain. It's used to		 * find devices once everything is set up.		 */		if (!pci_reverse) {			*pci_last_dev_p = dev;			pci_last_dev_p = &dev->next;		} else {			dev->next = pci_devices;			pci_devices = dev;		}		/*		 * Now insert it into the list of devices held		 * by the parent bus.		 */		*bus_last = dev;		bus_last = &dev->sibling;#if 0		/*		 * Setting of latency timer in case it was less than 32 was		 * a great idea, but it confused several broken devices. Grrr.		 */		pcibios_read_config_byte(bus->number, dev->devfn, PCI_LATENCY_TIMER, &tmp);		if (tmp < 32)			pcibios_write_config_byte(bus->number, dev->devfn, PCI_LATENCY_TIMER, 32);#endif	}	/*	 * After performing arch-dependent fixup of the bus, look behind	 * all PCI-to-PCI bridges on this bus.	 */	pcibios_fixup_bus(bus);	for(dev=bus->devices; dev; dev=dev->sibling)		/*		 * If it's a bridge, scan the bus behind it.		 */		if ((dev->class >> 8) == PCI_CLASS_BRIDGE_PCI) {			unsigned int buses;			unsigned int devfn = dev->devfn;			unsigned short cr;			/*			 * Insert it into the tree of buses.			 */			child = kmalloc(sizeof(*child), GFP_ATOMIC);			memset(child, 0, sizeof(*child));			child->next = bus->children;			bus->children = child;			child->self = dev;			child->parent = bus;			/*			 * Set up the primary, secondary and subordinate			 * bus numbers.			 */			child->number = child->secondary = ++max;			child->primary = bus->secondary;			child->subordinate = 0xff;			/*			 * Clear all status bits and turn off memory,			 * I/O and master enables.			 */			pcibios_read_config_word(bus->number, devfn, PCI_COMMAND, &cr);			pcibios_write_config_word(bus->number, devfn, PCI_COMMAND, 0x0000);			pcibios_write_config_word(bus->number, devfn, PCI_STATUS, 0xffff);			/*			 * Read the existing primary/secondary/subordinate bus			 * number configuration to determine if the PCI bridge			 * has already been configured by the system.  If so,			 * do not modify the configuration, merely note it.			 */			pcibios_read_config_dword(bus->number, devfn, PCI_PRIMARY_BUS, &buses);			if ((buses & 0xFFFFFF) != 0)			  {			    unsigned int cmax;			    child->primary = buses & 0xFF;			    child->secondary = (buses >> 8) & 0xFF;			    child->subordinate = (buses >> 16) & 0xFF;			    child->number = child->secondary;			    cmax = pci_scan_bus(child);			    if (cmax > max) max = cmax;			  }			else			  {			    /*			     * Configure the bus numbers for this bridge:			     */			    buses &= 0xff000000;			    buses |=			      (((unsigned int)(child->primary)     <<  0) |			       ((unsigned int)(child->secondary)   <<  8) |			       ((unsigned int)(child->subordinate) << 16));			    pcibios_write_config_dword(bus->number, devfn, PCI_PRIMARY_BUS, buses);			    /*			     * Now we can scan all subordinate buses:			     */			    max = pci_scan_bus(child);			    /*			     * Set the subordinate bus number to its real			     * value:			     */			    child->subordinate = max;			    buses = (buses & 0xff00ffff)			      | ((unsigned int)(child->subordinate) << 16);			    pcibios_write_config_dword(bus->number, devfn, PCI_PRIMARY_BUS, buses);			  }			pcibios_write_config_word(bus->number, devfn, PCI_COMMAND, cr);		}	/*	 * We've scanned the bus and so we know all about what's on	 * the other side of any bridges that may be on this bus plus	 * any devices.	 *	 * Return how far we've got finding sub-buses.	 */	DBG("PCI: pci_scan_bus returning with max=%02x\n", max);	return max;}struct pci_bus * __init pci_scan_peer_bridge(int bus){	struct pci_bus *b;	b = kmalloc(sizeof(*b), GFP_KERNEL);	memset(b, 0, sizeof(*b));	b->next = pci_root.next;	pci_root.next = b;	b->number = b->secondary = bus;	b->subordinate = pci_scan_bus(b);	return b;}__initfunc(void pci_init(void)){	pcibios_init();	if (!pci_present()) {		printk("PCI: No PCI bus detected\n");		return;	}	printk("PCI: Probing PCI hardware\n");	memset(&pci_root, 0, sizeof(pci_root));	pci_root.subordinate = pci_scan_bus(&pci_root);	/* give BIOS a chance to apply platform specific fixes: */	pcibios_fixup();#ifdef CONFIG_PCI_QUIRKS	pci_quirks_init();#endif#ifdef CONFIG_PROC_FS	pci_proc_init();#endif}__initfunc(void pci_setup (char *str, int *ints)){	while (str) {		char *k = strchr(str, ',');		if (k)			*k++ = 0;		if (*str && (str = pcibios_setup(str)) && *str) {			if (!strcmp(str, "reverse"))				pci_reverse = 1;			else printk(KERN_ERR "PCI: Unknown option `%s'\n", str);		}		str = k;	}}

⌨️ 快捷键说明

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