numa.c 3.9 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5 6 7
/*
 * numa.c - Low-level PCI access for NUMA-Q machines
 */

#include <linux/pci.h>
#include <linux/init.h>
#include <linux/nodemask.h>
8
#include <mach_apic.h>
L
Linus Torvalds 已提交
9 10
#include "pci.h"

11 12 13
#define XQUAD_PORTIO_BASE 0xfe400000
#define XQUAD_PORTIO_QUAD 0x40000  /* 256k per quad. */

L
Linus Torvalds 已提交
14 15
#define BUS2QUAD(global) (mp_bus_id_to_node[global])
#define BUS2LOCAL(global) (mp_bus_id_to_local[global])
16 17

int quad_local_to_mp_bus_id [NR_CPUS/4][4];
L
Linus Torvalds 已提交
18 19
#define QUADLOCAL2BUS(quad,local) (quad_local_to_mp_bus_id[quad][local])

20 21 22
extern void *xquad_portio;    /* Where the IO area was mapped */
#define XQUAD_PORT_ADDR(port, quad) (xquad_portio + (XQUAD_PORTIO_QUAD*quad) + port)

L
Linus Torvalds 已提交
23 24 25
#define PCI_CONF1_MQ_ADDRESS(bus, devfn, reg) \
	(0x80000000 | (BUS2LOCAL(bus) << 16) | (devfn << 8) | (reg & ~3))

26 27 28 29 30 31 32 33 34
static void write_cf8(unsigned bus, unsigned devfn, unsigned reg)
{
	unsigned val = PCI_CONF1_MQ_ADDRESS(bus, devfn, reg);
	if (xquad_portio)
		writel(val, XQUAD_PORT_ADDR(0xcf8, BUS2QUAD(bus)));
	else
		outl(val, 0xCF8);
}

L
Linus Torvalds 已提交
35 36 37 38
static int pci_conf1_mq_read(unsigned int seg, unsigned int bus,
			     unsigned int devfn, int reg, int len, u32 *value)
{
	unsigned long flags;
39
	void *adr __iomem = XQUAD_PORT_ADDR(0xcfc, BUS2QUAD(bus));
L
Linus Torvalds 已提交
40 41 42 43 44 45

	if (!value || (bus >= MAX_MP_BUSSES) || (devfn > 255) || (reg > 255))
		return -EINVAL;

	spin_lock_irqsave(&pci_config_lock, flags);

46
	write_cf8(bus, devfn, reg);
L
Linus Torvalds 已提交
47 48 49

	switch (len) {
	case 1:
50 51 52 53
		if (xquad_portio)
			*value = readb(adr + (reg & 3));
		else
			*value = inb(0xCFC + (reg & 3));
L
Linus Torvalds 已提交
54 55
		break;
	case 2:
56 57 58 59
		if (xquad_portio)
			*value = readw(adr + (reg & 2));
		else
			*value = inw(0xCFC + (reg & 2));
L
Linus Torvalds 已提交
60 61
		break;
	case 4:
62 63 64 65
		if (xquad_portio)
			*value = readl(adr);
		else
			*value = inl(0xCFC);
L
Linus Torvalds 已提交
66 67 68 69 70 71 72 73 74 75 76 77
		break;
	}

	spin_unlock_irqrestore(&pci_config_lock, flags);

	return 0;
}

static int pci_conf1_mq_write(unsigned int seg, unsigned int bus,
			      unsigned int devfn, int reg, int len, u32 value)
{
	unsigned long flags;
78
	void *adr __iomem = XQUAD_PORT_ADDR(0xcfc, BUS2QUAD(bus));
L
Linus Torvalds 已提交
79 80 81 82 83 84

	if ((bus >= MAX_MP_BUSSES) || (devfn > 255) || (reg > 255)) 
		return -EINVAL;

	spin_lock_irqsave(&pci_config_lock, flags);

85
	write_cf8(bus, devfn, reg);
L
Linus Torvalds 已提交
86 87 88

	switch (len) {
	case 1:
89 90 91 92
		if (xquad_portio)
			writeb(value, adr + (reg & 3));
		else
			outb((u8)value, 0xCFC + (reg & 3));
L
Linus Torvalds 已提交
93 94
		break;
	case 2:
95 96 97 98
		if (xquad_portio)
			writew(value, adr + (reg & 2));
		else
			outw((u16)value, 0xCFC + (reg & 2));
L
Linus Torvalds 已提交
99 100
		break;
	case 4:
101 102 103 104
		if (xquad_portio)
			writel(value, adr + reg);
		else
			outl((u32)value, 0xCFC);
L
Linus Torvalds 已提交
105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136
		break;
	}

	spin_unlock_irqrestore(&pci_config_lock, flags);

	return 0;
}

#undef PCI_CONF1_MQ_ADDRESS

static struct pci_raw_ops pci_direct_conf1_mq = {
	.read	= pci_conf1_mq_read,
	.write	= pci_conf1_mq_write
};


static void __devinit pci_fixup_i450nx(struct pci_dev *d)
{
	/*
	 * i450NX -- Find and scan all secondary buses on all PXB's.
	 */
	int pxb, reg;
	u8 busno, suba, subb;
	int quad = BUS2QUAD(d->bus->number);

	printk("PCI: Searching for i450NX host bridges on %s\n", pci_name(d));
	reg = 0xd0;
	for(pxb=0; pxb<2; pxb++) {
		pci_read_config_byte(d, reg++, &busno);
		pci_read_config_byte(d, reg++, &suba);
		pci_read_config_byte(d, reg++, &subb);
		DBG("i450NX PXB %d: %02x/%02x/%02x\n", pxb, busno, suba, subb);
137 138 139 140 141 142 143 144
		if (busno) {
			/* Bus A */
			pci_scan_bus_with_sysdata(QUADLOCAL2BUS(quad, busno));
		}
		if (suba < subb) {
			/* Bus B */
			pci_scan_bus_with_sysdata(QUADLOCAL2BUS(quad, suba+1));
		}
L
Linus Torvalds 已提交
145 146 147 148 149 150 151 152 153 154 155 156 157 158 159
	}
	pcibios_last_bus = -1;
}
DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_INTEL, PCI_DEVICE_ID_INTEL_82451NX, pci_fixup_i450nx);

static int __init pci_numa_init(void)
{
	int quad;

	raw_pci_ops = &pci_direct_conf1_mq;

	if (pcibios_scanned++)
		return 0;

	pci_root_bus = pcibios_scan_root(0);
160 161
	if (pci_root_bus)
		pci_bus_add_devices(pci_root_bus);
L
Linus Torvalds 已提交
162 163 164 165 166 167
	if (num_online_nodes() > 1)
		for_each_online_node(quad) {
			if (quad == 0)
				continue;
			printk("Scanning PCI bus %d for quad %d\n", 
				QUADLOCAL2BUS(quad,0), quad);
168
			pci_scan_bus_with_sysdata(QUADLOCAL2BUS(quad, 0));
L
Linus Torvalds 已提交
169 170 171 172 173
		}
	return 0;
}

subsys_initcall(pci_numa_init);