patch-2.3.35 linux/arch/sparc/kernel/pcic.c

Next file: linux/arch/sparc/kernel/process.c
Previous file: linux/arch/sparc/kernel/irq.c
Back to the patch index
Back to the overall index

diff -u --recursive --new-file v2.3.34/linux/arch/sparc/kernel/pcic.c linux/arch/sparc/kernel/pcic.c
@@ -1,4 +1,4 @@
-/* $Id: pcic.c,v 1.8 1999/08/31 06:54:22 davem Exp $
+/* $Id: pcic.c,v 1.11 1999/11/25 05:22:05 zaitcev Exp $
  * pcic.c: Sparc/PCI controller support
  *
  * Copyright (C) 1998 V. Roganov and G. Raiko
@@ -22,17 +22,6 @@
 #include <asm/swift.h> /* for cache flushing. */
 #include <asm/io.h>
 
-#undef PROM_DEBUG
-#undef FIXUP_REGS_DEBUG
-#undef FIXUP_IRQ_DEBUG
-#undef FIXUP_VMA_DEBUG
-
-#ifdef PROM_DEBUG
-#define dprintf	prom_printf
-#else
-#define dprintf printk
-#endif
-
 #include <linux/ctype.h>
 #include <linux/pci.h>
 #include <linux/timex.h>
@@ -57,7 +46,7 @@
 				  unsigned long len,
 				  unsigned char *buf)
 {
-	return 0;
+	return -EINVAL;
 }
 
 asmlinkage int sys_pciconfig_write(unsigned long bus,
@@ -66,11 +55,19 @@
 				   unsigned long len,
 				   unsigned char *buf)
 {
-	return 0;
+	return -EINVAL;
 }
 
 #else
 
+#ifdef CONFIG_SUN_JSFLASH
+extern int jsflash_init(void);
+#endif
+
+struct pci_fixup pcibios_fixups[] = {
+	{ 0 }
+};
+
 unsigned int pcic_pin_to_irq(unsigned int pin, char *name);
 
 /*
@@ -120,7 +117,8 @@
 	{ 0, 0x00, 0, 13, 0 },		/* Ebus - serial and keyboard */
 	{ 0, 0x01, 1,  6, 0 },		/* hme */
 	{ 0, 0x08, 2,  9, 0 },		/* VGA - we hope not used :) */
-	{ 0, 0x18, 6,  8, 0 },		/* PCI INTA# in Slot 1 */
+	{ 0, 0x10, 6,  8, 0 },		/* PCI INTA# in Slot 1 */
+	{ 0, 0x18, 7, 12, 0 },		/* PCI INTA# in Slot 2, shared w. RTC */
 	{ 0, 0x38, 4,  9, 0 },		/* All ISA devices. Read 8259. */
 	{ 0, 0x80, 5, 11, 0 },		/* EIDE */
 	/* {0,0x88, 0,0,0} - unknown device... PMU? Probably no interrupt. */
@@ -144,6 +142,16 @@
 };
 
 /*
+ * Krups (courtesy of Varol Kaptan)
+ * No documentation available, so we guess it, based on Espresso layout.
+ * Since we always run PROLL on Krups we may put map in there.
+ */
+static struct pcic_ca2irq pcic_i_jk[] = {
+	{ 0, 0x00, 0, 13, 0 },		/* Ebus - serial and keyboard */
+	{ 0, 0x01, 1,  6, 0 },		/* hme */
+};
+
+/*
  * Several entries in this list may point to the same routing map
  * as several PROMs may be installed on the same physical board.
  */
@@ -154,11 +162,17 @@
 	SN2L_INIT("JE-1-name", pcic_i_je1),  /* XXX Gleb, put name here, pls */
 	SN2L_INIT("SUNW,JS-E", pcic_i_jse),	/* PROLL JavaStation-E */
 	SN2L_INIT("SUNW,SPARCengine-6", pcic_i_se6), /* SPARCengine-6/CP-1200 */
+	SN2L_INIT("SUNW,JS-NC", pcic_i_jk),	/* PROLL JavaStation-NC */
+	SN2L_INIT("SUNW,JSIIep", pcic_i_jk),	/* OBP JavaStation-NC */
 	{ NULL, NULL, 0 }
 };
 
-static struct linux_pcic PCIC;
-static struct linux_pcic *pcic = NULL;
+/*
+ * Only one PCIC per IIep,
+ * and since we have no SMP IIep, only one per system.
+ */
+static int pcic0_up = 0;
+static struct linux_pcic pcic0;
 
 unsigned int pcic_regs;
 volatile int pcic_speculative;
@@ -167,23 +181,144 @@
 static void pci_do_gettimeofday(struct timeval *tv);
 static void pci_do_settimeofday(struct timeval *tv);
 
-void __init pcic_probe(void)
+#define CONFIG_CMD(bus, device_fn, where) (0x80000000 | (((unsigned int)bus) << 16) | (((unsigned int)device_fn) << 8) | (where & ~3))
+
+static int pcic_read_config_dword(struct pci_dev *dev, int where, u32 *value);
+static int pcic_write_config_dword(struct pci_dev *dev, int where, u32 value);
+
+static int pcic_read_config_byte(struct pci_dev *dev, int where, u8 *value)
+{
+	unsigned int v;
+
+	pcic_read_config_dword(dev, where&~3, &v);
+	*value = 0xff & (v >> (8*(where & 3)));
+	return PCIBIOS_SUCCESSFUL;
+}
+
+static int pcic_read_config_word(struct pci_dev *dev, int where, u16 *value)
+{
+	unsigned int v;
+	if (where&1) return PCIBIOS_BAD_REGISTER_NUMBER;
+
+	pcic_read_config_dword(dev, where&~3, &v);
+	*value = 0xffff & (v >> (8*(where & 3)));
+	return PCIBIOS_SUCCESSFUL;
+}
+
+static int pcic_read_config_dword(struct pci_dev *dev, int where, u32 *value)
 {
+	unsigned char bus = dev->bus->number;
+	unsigned char device_fn = dev->devfn;
+	/* unsigned char where; */
+
+	struct linux_pcic *pcic;
+	unsigned long flags;
+
+	if (where&3) return PCIBIOS_BAD_REGISTER_NUMBER;
+	if (bus != 0) return PCIBIOS_DEVICE_NOT_FOUND;
+	pcic = &pcic0;
+
+	save_and_cli(flags);
+#if 0 /* does not fail here */
+	pcic_speculative = 1;
+	pcic_trapped = 0;
+#endif
+	writel(CONFIG_CMD(bus,device_fn,where), pcic->pcic_config_space_addr);
+#if 0 /* does not fail here */
+	nop();
+	if (pcic_trapped) {
+		restore_flags(flags);
+		*value = ~0;
+		return PCIBIOS_SUCCESSFUL;
+	}
+#endif
+	pcic_speculative = 2;
+	pcic_trapped = 0;
+	*value = readl(pcic->pcic_config_space_data + (where&4));
+	nop();
+	if (pcic_trapped) {
+		pcic_speculative = 0;
+		restore_flags(flags);
+		*value = ~0;
+		return PCIBIOS_SUCCESSFUL;
+	}
+	pcic_speculative = 0;
+	restore_flags(flags);
+	return PCIBIOS_SUCCESSFUL;
+}
+
+static int pcic_write_config_byte(struct pci_dev *dev, int where, u8 value)
+{
+	unsigned int v;
+
+	pcic_read_config_dword(dev, where&~3, &v);
+        v = (v & ~(0xff << (8*(where&3)))) |
+            ((0xff&(unsigned)value) << (8*(where&3)));
+	return pcic_write_config_dword(dev, where&~3, v);
+}
+
+static int pcic_write_config_word(struct pci_dev *dev, int where, u16 value)
+{
+	unsigned int v;
+
+	if (where&1) return PCIBIOS_BAD_REGISTER_NUMBER;
+	pcic_read_config_dword(dev, where&~3, &v);
+	v = (v & ~(0xffff << (8*(where&3)))) |
+	    ((0xffff&(unsigned)value) << (8*(where&3)));
+	return pcic_write_config_dword(dev, where&~3, v);
+}
+
+static int pcic_write_config_dword(struct pci_dev *dev, int where, u32 value)
+{
+	unsigned char bus = dev->bus->number;
+	unsigned char devfn = dev->devfn;
+	struct linux_pcic *pcic;
+	unsigned long flags;
+
+	if (where&3) return PCIBIOS_BAD_REGISTER_NUMBER;
+	if (bus != 0) return PCIBIOS_DEVICE_NOT_FOUND;
+	pcic = &pcic0;
+
+	save_and_cli(flags);
+	writel(CONFIG_CMD(bus,devfn,where), pcic->pcic_config_space_addr);
+	writel(value, pcic->pcic_config_space_data + (where&4));
+	restore_flags(flags);
+	return PCIBIOS_SUCCESSFUL;
+}
+
+static struct pci_ops pcic_ops = {
+	pcic_read_config_byte,
+	pcic_read_config_word,
+	pcic_read_config_dword,
+	pcic_write_config_byte,
+	pcic_write_config_word,
+	pcic_write_config_dword,
+};
+
+/*
+ * On sparc64 pcibios_init() calls pci_controller_probe().
+ * We want PCIC probed little ahead so that interrupt controller
+ * would be operational.
+ */
+int __init pcic_probe(void)
+{
+	struct linux_pcic *pcic;
 	struct linux_prom_registers regs[PROMREG_MAX];
 	struct linux_pbm_info* pbm;
 	char namebuf[64];
 	int node;
 	int err;
 
-	if (pcibios_present()) {
+	if (pcic0_up) {
 		prom_printf("PCIC: called twice!\n");
 		prom_halt();
 	}
+	pcic = &pcic0;
 
 	node = prom_getchild (prom_root_node);
 	node = prom_searchsiblings (node, "pci");
 	if (node == 0)
-		return;
+		return -ENODEV;
 	/*
 	 * Map in PCIC register set, config space, and IO base
 	 */
@@ -193,31 +328,27 @@
 			    "from PROM.\n");
 		prom_halt();
 	}
-	
-	pcic = &PCIC;
 
-	pcic->pcic_regs = (unsigned long)sparc_alloc_io(regs[0].phys_addr, NULL,
-					      regs[0].reg_size,
-					      "PCIC Registers", 0, 0);
+	pcic0_up = 1;
+
+	pcic->pcic_res_regs.name = "pcic_registers";
+	pcic->pcic_regs = (unsigned long)
+	    ioremap(regs[0].phys_addr, regs[0].reg_size);
 	if (!pcic->pcic_regs) {
 		prom_printf("PCIC: Error, cannot map PCIC registers.\n");
 		prom_halt();
 	}
 
-	pcic->pcic_io_phys = regs[1].phys_addr;
-	pcic->pcic_io = (unsigned long)sparc_alloc_io(regs[1].phys_addr, NULL,
-					    regs[1].reg_size,
-					    "PCIC IO Base", 0, 0);
-	if (pcic->pcic_io == 0UL) {
+	pcic->pcic_res_io.name = "pcic_io";
+	if ((pcic->pcic_io = (unsigned long)
+	    ioremap(regs[1].phys_addr, 0x10000)) == 0) {
 		prom_printf("PCIC: Error, cannot map PCIC IO Base.\n");
 		prom_halt();
 	}
 
-	pcic->pcic_config_space_addr =
-			(unsigned long)sparc_alloc_io (regs[2].phys_addr, NULL,
-					     regs[2].reg_size * 2,
-					     "PCI Config Space Address", 0, 0);
-	if (pcic->pcic_config_space_addr == 0UL) {
+	pcic->pcic_res_cfg_addr.name = "pcic_cfg_addr";
+	if ((pcic->pcic_config_space_addr = (unsigned long)
+	    ioremap(regs[2].phys_addr, regs[2].reg_size * 2)) == 0) {
 		prom_printf("PCIC: Error, cannot map" 
 			    "PCI Configuration Space Address.\n");
 		prom_halt();
@@ -227,11 +358,9 @@
 	 * Docs say three least significant bits in address and data
 	 * must be the same. Thus, we need adjust size of data.
 	 */
-	pcic->pcic_config_space_data =
-			(unsigned long)sparc_alloc_io (regs[3].phys_addr, NULL,
-					     regs[3].reg_size * 2,
-					     "PCI Config Space Data", 0, 0);
-	if (pcic->pcic_config_space_data == 0UL) {
+	pcic->pcic_res_cfg_data.name = "pcic_cfg_data";
+	if ((pcic->pcic_config_space_data = (unsigned long)
+	    ioremap(regs[3].phys_addr, regs[3].reg_size * 2)) == 0) {
 		prom_printf("PCIC: Error, cannot map" 
 			    "PCI Configuration Space Data.\n");
 		prom_halt();
@@ -239,7 +368,7 @@
 
 	pbm = &pcic->pbm;
 	pbm->prom_node = node;
-	prom_getstring(node, "name", namebuf, sizeof(namebuf));
+	prom_getstring(node, "name", namebuf, 63);  namebuf[63] = 0;
 	strcpy(pbm->prom_name, namebuf);
 
 	{
@@ -254,7 +383,7 @@
 		pcic_regs = pcic->pcic_regs;
 	}
 
-	prom_getstring(prom_root_node, "name", namebuf, sizeof(namebuf));
+	prom_getstring(prom_root_node, "name", namebuf, 63);  namebuf[63] = 0;
 	{
 		struct pcic_sn2list *p;
 
@@ -272,21 +401,37 @@
 		printk("PCIC: System %s is unknown, cannot route interrupts\n",
 		    namebuf);
 	}
+
+	return 0;
+}
+
+static void __init pcic_pbm_scan_bus(struct linux_pcic *pcic)
+{
+	struct linux_pbm_info *pbm = &pcic->pbm;
+
+	pbm->pci_bus = pci_scan_bus(pbm->pci_first_busno, &pcic_ops, pbm);
+#if 0 /* deadwood transplanted from sparc64 */
+	pci_fill_in_pbm_cookies(pbm->pci_bus, pbm, pbm->prom_node);
+	pci_record_assignments(pbm, pbm->pci_bus);
+	pci_assign_unassigned(pbm, pbm->pci_bus);
+	pci_fixup_irq(pbm, pbm->pci_bus);
+#endif
 }
 
+/*
+ * Main entry point from the PCI subsystem.
+ */
 void __init pcibios_init(void)
 {
+	struct linux_pcic *pcic;
+
 	/*
 	 * PCIC should be initialized at start of the timer.
 	 * So, here we report the presence of PCIC and do some magic passes.
 	 */
-	if(!pcic)
+	if(!pcic0_up)
 		return;
-
-	printk("PCIC MAP: config addr=0x%lx; config data=0x%lx, "
-	       "regs=0x%lx io=0x%lx\n",
-	       pcic->pcic_config_space_addr, pcic->pcic_config_space_data,
-	       pcic->pcic_regs, pcic->pcic_io);
+	pcic = &pcic0;
 
 	/*
 	 *      Switch off IOTLB translation.
@@ -302,11 +447,18 @@
 	writel(0xF0000000UL, pcic->pcic_regs+PCI_SIZE_0);
 	writel(0+PCI_BASE_ADDRESS_SPACE_MEMORY, 
 	       pcic->pcic_regs+PCI_BASE_ADDRESS_0);
+
+	pcic_pbm_scan_bus(pcic);
+
+	ebus_init();
+#ifdef CONFIG_SUN_JSFLASH
+	jsflash_init();
+#endif
 }
 
-int pcibios_present(void)
+int pcic_present(void)
 {
-	return pcic != NULL;
+	return pcic0_up;
 }
 
 static int __init pdev_to_pnode(struct linux_pbm_info *pbm, 
@@ -334,91 +486,77 @@
 	return kmalloc(sizeof(struct pcidev_cookie), GFP_ATOMIC);
 }
 
-static void pcic_map_pci_device (struct pci_dev *dev, int node) {
-	struct linux_prom_pci_assigned_addresses addrs[6];
-	int addrlen;
-	int i, j;
-
-	/* Is any valid address present ? */
-	i = 0;
-	for(j = 0; j < 6; j++)
-		if (dev->base_address[j]) i++;
-	if (!i) return; /* nothing to do */
+static void pcic_map_pci_device(struct linux_pcic *pcic,
+    struct pci_dev *dev, int node)
+{
+	char namebuf[64];
+	unsigned long address;
+	unsigned long flags;
+	int j;
 
 	if (node == 0 || node == -1) {
-		printk("PCIC: no prom node for device ID (%x,%x)\n",
-		    dev->device, dev->vendor);
-		return;
-	}
-
-	/*
-	 * find related address and get it's window length
-	 */
-	addrlen = prom_getproperty(node,"assigned-addresses",
-					       (char*)addrs, sizeof(addrs));
-	if (addrlen == -1) {
-		printk("PCIC: no \"assigned-addresses\" for device (%x,%x)\n",
-		    dev->device, dev->vendor);
-		return;
+		strcpy(namebuf, "???");
+	} else {
+		prom_getstring(node, "name", namebuf, 63); namebuf[63] = 0;
 	}
 
-	addrlen /= sizeof(struct linux_prom_pci_assigned_addresses);
-	for (i = 0; i < addrlen; i++ )
-	    for (j = 0; j < 6; j++) {
-		if (!dev->base_address[j] || !addrs[i].phys_lo)
-			continue;
-		if (addrs[i].phys_lo == dev->base_address[j]) {
-			unsigned long address = dev->base_address[j];
-			int length  = addrs[i].size_lo;
-			char namebuf[128] = { 0, };
-			unsigned long mapaddr, addrflags;
-
-			prom_getstring(node, "name", namebuf, sizeof(namebuf));
-
-			/*
-			 *      failure in allocation too large space
-			 */
-			if (length > 0x200000) {
-				length = 0x200000;
-				prom_printf("PCIC: map window for device '%s' "
-					    "reduced to 2MB !\n", namebuf);
-			}
-
-			/*
-			 *  Be careful with MEM/IO address flags
-			 */
-			if ((address & PCI_BASE_ADDRESS_SPACE) ==
-				 PCI_BASE_ADDRESS_SPACE_IO) {
-				mapaddr = address & PCI_BASE_ADDRESS_IO_MASK;
+	for (j = 0; j < 6; j++) {
+		address = dev->resource[j].start;
+		if (address == 0) break;	/* are sequential */
+		flags = dev->resource[j].flags;
+		if ((flags & IORESOURCE_IO) != 0) {
+			if (address < 0x10000) {
+				/*
+				 * A device responds to I/O cycles on PCI.
+				 * We generate these cycles with memory
+				 * access into the fixed map (phys 0x30000000).
+				 *
+				 * Since a device driver does not want to
+				 * do ioremap() before accessing PC-style I/O,
+				 * we supply virtual, ready to access address.
+				 *
+				 * Ebus devices do not come here even if
+				 * CheerIO makes a similar conversion.
+				 * See ebus.c for details.
+				 *
+				 * Note that check_region()/request_region()
+				 * work for these devices.
+				 *
+				 * XXX Neat trick, but it's a *bad* idea
+				 * to shit into regions like that.
+				 * What if we want to allocate one more
+				 * PCI base address...
+				 */
+				dev->resource[j].start =
+				    pcic->pcic_io + address;
+				dev->resource[j].end = 1;  /* XXX */
+				dev->resource[j].flags =
+				    (flags & ~IORESOURCE_IO) | IORESOURCE_MEM;
 			} else {
-				mapaddr = address & PCI_BASE_ADDRESS_MEM_MASK;
+				/*
+				 * OOPS... PCI Spec allows this. Sun does
+				 * not have any devices getting above 64K
+				 * so it must be user with a weird I/O
+				 * board in a PCI slot. We must remap it
+				 * under 64K but it is not done yet. XXX
+				 */
+				printk("PCIC: Skipping I/O space at 0x%lx,"
+				    "this will Oops if a driver attaches;"
+				    "device '%s' (%x,%x)\n", address, namebuf,
+	    			    dev->device, dev->vendor);
 			}
-			addrflags = address ^ mapaddr;
-
-			dev->base_address[j] =
-				(unsigned long)sparc_alloc_io(address, 0, 
-							      length,
-							      namebuf, 0, 0);
-			if ( dev->base_address[j] == 0 )
-				panic("PCIC: failed make mapping for "
-				      "pci device '%s' with address %lx\n",
-				       namebuf, address);
-
-			dev->base_address[j] ^= addrflags;
-			return;
 		}
-	    }
-
-	printk("PCIC: unable to match addresses for device (%x,%x)\n",
-	    dev->device, dev->vendor);
+	}
 }
 
-static void pcic_fill_irq(struct pci_dev *dev, int node) {
+static void
+pcic_fill_irq(struct linux_pcic *pcic, struct pci_dev *dev, int node)
+{
 	struct pcic_ca2irq *p;
 	int i, ivec;
 	char namebuf[64];  /* P3 remove */
 
-	if (node == -1) {
+	if (node == 0 || node == -1) {
 		strcpy(namebuf, "???");
 	} else {
 		prom_getstring(node, "name", namebuf, sizeof(namebuf)); /* P3 remove */
@@ -474,53 +612,33 @@
 }
 
 /*
- * Assign IO space for a device.
- * This is a chance for devices which have the same IO and Mem Space to
- * fork access to IO and Mem.
- *
- * Now, we assume there is one such device only (IGA 1682) but code below
- * should work in cases when space of all such devices is less then 16MB.
- */
-unsigned long pcic_alloc_io( unsigned long* addr )
-{
-	unsigned long paddr = *addr;
-	unsigned long offset;
-
-	if(pcic->pcic_mapped_io == 0) {
-		pcic->pcic_mapped_io = paddr & ~(PCI_SPACE_SIZE-1) ;
-		writeb((pcic->pcic_mapped_io>>24) & 0xff, 
-		       pcic->pcic_regs+PCI_PIBAR);
-		writeb((pcic->pcic_io_phys>>24) & PCI_SIBAR_ADDRESS_MASK,
-		       pcic->pcic_regs+PCI_SIBAR);
-		writeb(PCI_ISIZE_16M, pcic->pcic_regs+PCI_ISIZE);
-
-	}
-	if(paddr < pcic->pcic_mapped_io ||
-	   paddr >= pcic->pcic_mapped_io + 0x10000)
-		return 0;
-	offset = paddr - pcic->pcic_mapped_io;
-	*addr = pcic->pcic_io_phys + offset;
-	return pcic->pcic_io + offset;
-}
-
-/*
- * Stolen from both i386 and sparc64 branch 
+ * Normally called from {do_}pci_scan_bus...
  */
-void __init pcibios_fixup(void)
+void __init pcibios_fixup_bus(struct pci_bus *bus)
 {
-  struct pci_dev *dev;
-  int i, has_io, has_mem;
-  unsigned short cmd;
-	struct linux_pbm_info* pbm = &pcic->pbm;
+	struct pci_dev *dev;
+	int i, has_io, has_mem;
+	unsigned short cmd;
+	struct linux_pcic *pcic;
+	/* struct linux_pbm_info* pbm = &pcic->pbm; */
 	int node;
 	struct pcidev_cookie *pcp;
 
-	if(pcic == NULL) {
-		prom_printf("PCI: Error, PCIC not found.\n");
-		prom_halt();
+	if (!pcic0_up) {
+		printk("pcibios_fixup_bus: no PCIC\n");
+		return;
 	}
+	pcic = &pcic0;
 
-	for (dev = pci_devices; dev; dev=dev->next) {
+	/*
+	 * Next crud is an equivalent of pbm = pcic_bus_to_pbm(bus);
+	 */
+	if (bus->number != 0) {
+		printk("pcibios_fixup_bus: nonzero bus 0x%x\n", bus->number);
+		return;
+	}
+
+	for (dev = bus->devices; dev; dev = dev->sibling) {
 		/*
 		 * Comment from i386 branch:
 		 *     There are buggy BIOSes that forget to enable I/O and memory
@@ -531,44 +649,42 @@
 		 */
 		has_io = has_mem = 0;
 		for(i=0; i<6; i++) {
-			unsigned long a = dev->base_address[i];
-			if (a & PCI_BASE_ADDRESS_SPACE_IO) {
+			unsigned long f = dev->resource[i].flags;
+			if (f & IORESOURCE_IO) {
 				has_io = 1;
-			} else if (a & PCI_BASE_ADDRESS_MEM_MASK)
+			} else if (f & IORESOURCE_MEM)
 				has_mem = 1;
 		}
-		pci_read_config_word(dev, PCI_COMMAND, &cmd);
+		pcic_read_config_word(dev, PCI_COMMAND, &cmd);
 		if (has_io && !(cmd & PCI_COMMAND_IO)) {
 			printk("PCIC: Enabling I/O for device %02x:%02x\n",
 				dev->bus->number, dev->devfn);
 			cmd |= PCI_COMMAND_IO;
-			pci_write_config_word(dev, PCI_COMMAND, cmd);
+			pcic_write_config_word(dev, PCI_COMMAND, cmd);
 		}
 		if (has_mem && !(cmd & PCI_COMMAND_MEMORY)) {
 			printk("PCIC: Enabling memory for device %02x:%02x\n",
 				dev->bus->number, dev->devfn);
 			cmd |= PCI_COMMAND_MEMORY;
-			pci_write_config_word(dev, PCI_COMMAND, cmd);
+			pcic_write_config_word(dev, PCI_COMMAND, cmd);
 		}    
 
-		node = pdev_to_pnode(pbm, dev);
+		node = pdev_to_pnode(&pcic->pbm, dev);
 		if(node == 0)
 			node = -1;
 
 		/* cookies */
 		pcp = pci_devcookie_alloc();
-		pcp->pbm = pbm;
+		pcp->pbm = &pcic->pbm;
 		pcp->prom_node = node;
 		dev->sysdata = pcp;
 
-		/* memory mapping */
+		/* fixing I/O to look like memory */
 		if ((dev->class>>16) != PCI_BASE_CLASS_BRIDGE)
-			pcic_map_pci_device(dev, node);
+			pcic_map_pci_device(pcic, dev, node);
 
-		pcic_fill_irq(dev, node);
+		pcic_fill_irq(pcic, dev, node);
 	}
-
-	ebus_init();
 }
 
 /*
@@ -577,6 +693,7 @@
 unsigned int
 pcic_pin_to_irq(unsigned int pin, char *name)
 {
+	struct linux_pcic *pcic = &pcic0;
 	unsigned int irq;
 	unsigned int ivec;
 
@@ -599,7 +716,7 @@
 
 static void pcic_clear_clock_irq(void)
 {
-	pcic_timer_dummy = readl(pcic->pcic_regs+PCI_SYS_LIMIT);
+	pcic_timer_dummy = readl(pcic0.pcic_regs+PCI_SYS_LIMIT);
 }
 
 static void pcic_timer_handler (int irq, void *h, struct pt_regs *regs)
@@ -613,6 +730,7 @@
 
 void __init pci_time_init(void)
 {
+	struct linux_pcic *pcic = &pcic0;
 	unsigned long v;
 	int timer_irq, irq;
 
@@ -620,9 +738,9 @@
 	/* A hack until do_gettimeofday prototype is moved to arch specific headers
 	   and btfixupped. Patch do_gettimeofday with ba pci_do_gettimeofday; nop */
 	((unsigned int *)do_gettimeofday)[0] = 
-		0x10800000 | ((((unsigned long)pci_do_gettimeofday - (unsigned long)do_gettimeofday) >> 2) & 0x003fffff);
-	((unsigned int *)do_gettimeofday)[1] =
-		0x01000000;
+	    0x10800000 | ((((unsigned long)pci_do_gettimeofday -
+	     (unsigned long)do_gettimeofday) >> 2) & 0x003fffff);
+	((unsigned int *)do_gettimeofday)[1] = 0x01000000;
 	BTFIXUPSET_CALL(bus_do_settimeofday, pci_do_settimeofday, BTFIXUPCALL_NORM);
 	btfixup();
 
@@ -650,7 +768,7 @@
 	 * to have microsecond resolution and to avoid overflow
 	 */
 	unsigned long count = 
-	    readl(pcic->pcic_regs+PCI_SYS_COUNTER) & ~PCI_SYS_COUNTER_OVERFLOW;
+	    readl(pcic0.pcic_regs+PCI_SYS_COUNTER) & ~PCI_SYS_COUNTER_OVERFLOW;
 	count = ((count/100)*USECS_PER_JIFFY) / (TICK_TIMER_LIMIT/100);
 
 	if(test_bit(TIMER_BH, &bh_active))
@@ -705,103 +823,9 @@
 }
 #endif
 
-#define CONFIG_CMD(bus, device_fn, where) (0x80000000 | (((unsigned int)bus) << 16) | (((unsigned int)device_fn) << 8) | (where & ~3))
-
-int pcibios_read_config_byte(unsigned char bus, unsigned char device_fn,
-			     unsigned char where, unsigned char *value)
-{
-	unsigned int v;
-
-	pcibios_read_config_dword (bus, device_fn, where&~3, &v);
-	*value = 0xff & (v >> (8*(where & 3)));
-	return PCIBIOS_SUCCESSFUL;
-}
-
-int pcibios_read_config_word (unsigned char bus,
-			      unsigned char device_fn, 
-			      unsigned char where, unsigned short *value)
-{
-	unsigned int v;
-	if (where&1) return PCIBIOS_BAD_REGISTER_NUMBER;
-  
-	pcibios_read_config_dword (bus, device_fn, where&~3, &v);
-	*value = 0xffff & (v >> (8*(where & 3)));
-	return PCIBIOS_SUCCESSFUL;
-}
-
-int pcibios_read_config_dword (unsigned char bus, unsigned char device_fn,
-			       unsigned char where, unsigned int *value)
-{
-	unsigned long flags;
-
-	if (where&3) return PCIBIOS_BAD_REGISTER_NUMBER;
-
-	save_and_cli(flags);
-#if 0
-	pcic_speculative = 1;
-	pcic_trapped = 0;
-#endif
-	writel(CONFIG_CMD(bus,device_fn,where), pcic->pcic_config_space_addr);
-#if 0
-	nop();
-	if (pcic_trapped) {
-		restore_flags(flags);
-		*value = ~0;
-		return PCIBIOS_SUCCESSFUL;
-	}
-#endif
-	pcic_speculative = 2;
-	pcic_trapped = 0;
-	*value = readl(pcic->pcic_config_space_data + (where&4));
-	nop();
-	if (pcic_trapped) {
-		pcic_speculative = 0;
-		restore_flags(flags);
-		*value = ~0;
-		return PCIBIOS_SUCCESSFUL;
-	}
-	pcic_speculative = 0;
-	restore_flags(flags);
-	return PCIBIOS_SUCCESSFUL;
-}
-
-int pcibios_write_config_byte (unsigned char bus, unsigned char devfn,
-			       unsigned char where, unsigned char value)
-{
-	unsigned int v;
-
-	pcibios_read_config_dword (bus, devfn, where&~3, &v);
-	v = (v & ~(0xff << (8*(where&3)))) | 
-	    ((0xff&(unsigned)value) << (8*(where&3)));
-	return pcibios_write_config_dword (bus, devfn, where&~3, v);
-}
-
-int pcibios_write_config_word (unsigned char bus, unsigned char devfn,
-			       unsigned char where, unsigned short value)
-{
-	unsigned int v;
-	if (where&1) return PCIBIOS_BAD_REGISTER_NUMBER;
-
-	pcibios_read_config_dword (bus, devfn, where&~3, &v);
-	v = (v & ~(0xffff << (8*(where&3)))) | 
-	    ((0xffff&(unsigned)value) << (8*(where&3)));
-	return pcibios_write_config_dword (bus, devfn, where&~3, v);
-}
-  
-int pcibios_write_config_dword (unsigned char bus, unsigned char devfn,
-				unsigned char where, unsigned int value)
-{
-	unsigned long flags;
-
-	if (where&3) return PCIBIOS_BAD_REGISTER_NUMBER;
-
-	save_and_cli(flags);
-	writel(CONFIG_CMD(bus,devfn,where),pcic->pcic_config_space_addr);
-	writel(value, pcic->pcic_config_space_data + (where&4));
-	restore_flags(flags);
-	return PCIBIOS_SUCCESSFUL;
-}
-
+/*
+ * Other archs parse arguments here.
+ */
 char * __init pcibios_setup(char *str)
 {
 	return str;
@@ -831,6 +855,9 @@
 }
 
 /*
+ * XXX  Gleb wrote me that he needs this for X server (only).
+ * Since we successfuly use XF86_FBDev, we do not need these anymore.
+ *
  *  Following code added to handle extra PCI-related system calls 
  */
 asmlinkage int sys_pciconfig_read(unsigned long bus,
@@ -870,7 +897,7 @@
 
 	return err;
 }
-   
+
 asmlinkage int sys_pciconfig_write(unsigned long bus,
 				   unsigned long dfn,
 				   unsigned long off,
@@ -936,7 +963,7 @@
 
 	mask = get_irqmask(irq_nr);
 	save_and_cli(flags);
-	writel(mask, pcic->pcic_regs+PCI_SYS_INT_TARGET_MASK_SET);
+	writel(mask, pcic0.pcic_regs+PCI_SYS_INT_TARGET_MASK_SET);
 	restore_flags(flags);
 }
 
@@ -946,7 +973,7 @@
 
 	mask = get_irqmask(irq_nr);
 	save_and_cli(flags);
-	writel(mask, pcic->pcic_regs+PCI_SYS_INT_TARGET_MASK_CLEAR);
+	writel(mask, pcic0.pcic_regs+PCI_SYS_INT_TARGET_MASK_CLEAR);
 	restore_flags(flags);
 }
 
@@ -965,12 +992,12 @@
  */
 static void pcic_disable_pil_irq(unsigned int pil)
 {
-	writel(get_irqmask(pil), pcic->pcic_regs+PCI_SYS_INT_TARGET_MASK_SET);
+	writel(get_irqmask(pil), pcic0.pcic_regs+PCI_SYS_INT_TARGET_MASK_SET);
 }
 
 static void pcic_enable_pil_irq(unsigned int pil)
 {
-	writel(get_irqmask(pil), pcic->pcic_regs+PCI_SYS_INT_TARGET_MASK_CLEAR);
+	writel(get_irqmask(pil), pcic0.pcic_regs+PCI_SYS_INT_TARGET_MASK_CLEAR);
 }
 
 void __init sun4m_pci_init_IRQ(void)
@@ -985,8 +1012,9 @@
 	BTFIXUPSET_CALL(__irq_itoa, pcic_irq_itoa, BTFIXUPCALL_NORM);
 }
 
-void __init pcibios_fixup_bus(struct pci_bus *bus)
+int pcibios_assign_resource(struct pci_dev *pdev, int resource)
 {
+	return -ENXIO;
 }
 
 #endif

FUNET's LINUX-ADM group, linux-adm@nic.funet.fi
TCL-scripts by Sam Shen (who was at: slshen@lbl.gov)