pci_64.c 8.4 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
/*
 * Port for PPC64 David Engebretsen, IBM Corp.
 * Contains common pci routines for ppc64 platform, pSeries and iSeries brands.
 * 
 * Copyright (C) 2003 Anton Blanchard <anton@au.ibm.com>, IBM
 *   Rework, based on alpha PCI code.
 *
 *      This program is free software; you can redistribute it and/or
 *      modify it under the terms of the GNU General Public License
 *      as published by the Free Software Foundation; either version
 *      2 of the License, or (at your option) any later version.
 */

#undef DEBUG

#include <linux/kernel.h>
#include <linux/pci.h>
#include <linux/string.h>
#include <linux/init.h>
#include <linux/bootmem.h>
#include <linux/mm.h>
#include <linux/list.h>
23
#include <linux/syscalls.h>
24
#include <linux/irq.h>
25
#include <linux/vmalloc.h>
L
Linus Torvalds 已提交
26 27 28 29 30 31 32

#include <asm/processor.h>
#include <asm/io.h>
#include <asm/prom.h>
#include <asm/pci-bridge.h>
#include <asm/byteorder.h>
#include <asm/machdep.h>
33
#include <asm/ppc-pci.h>
L
Linus Torvalds 已提交
34 35 36 37 38 39

unsigned long pci_probe_only = 1;

/* pci_io_base -- the base address from which io bars are offsets.
 * This is the lowest I/O base address (so bar values are always positive),
 * and it *must* be the start of ISA space if an ISA bus exists because
40 41
 * ISA drivers use hard coded offsets.  If no ISA bus exists nothing
 * is mapped on the first 64K of IO space
L
Linus Torvalds 已提交
42
 */
43
unsigned long pci_io_base = ISA_IO_BASE;
L
Linus Torvalds 已提交
44 45
EXPORT_SYMBOL(pci_io_base);

J
John Rose 已提交
46
void __devinit scan_phb(struct pci_controller *hose)
47 48
{
	struct pci_bus *bus;
49
	struct device_node *node = hose->dn;
50
	int mode;
51

52 53
	pr_debug("PCI: Scanning PHB %s\n",
		 node ? node->full_name : "<NO NAME>");
54

55
	/* Create an empty bus for the toplevel */
56
	bus = pci_create_bus(hose->parent, hose->first_busno, hose->ops, node);
57 58 59 60 61 62 63 64
	if (bus == NULL) {
		printk(KERN_ERR "Failed to create bus for PCI domain %04x\n",
		       hose->global_number);
		return;
	}
	bus->secondary = hose->first_busno;
	hose->bus = bus;

65
	/* Get some IO space for the new PHB */
66
	pcibios_map_io_space(bus);
67

68
	/* Wire up PHB bus resources */
69
	pcibios_setup_phb_resources(hose);
70

71
	/* Get probe mode and perform scan */
72
	mode = PCI_PROBE_NORMAL;
73
	if (node && ppc_md.pci_probe_mode)
74
		mode = ppc_md.pci_probe_mode(bus);
75
	pr_debug("    probe mode: %d\n", mode);
76 77 78 79
	if (mode == PCI_PROBE_DEVTREE) {
		bus->subordinate = hose->last_busno;
		of_scan_bus(node, bus);
	}
80

81 82 83 84
	if (mode == PCI_PROBE_NORMAL)
		hose->last_busno = bus->subordinate = pci_scan_child_bus(bus);
}

L
Linus Torvalds 已提交
85 86 87 88
static int __init pcibios_init(void)
{
	struct pci_controller *hose, *tmp;

89 90
	printk(KERN_INFO "PCI: Probing PCI hardware\n");

91
	/* For now, override phys_mem_access_prot. If we need it,g
L
Linus Torvalds 已提交
92 93 94 95
	 * later, we may move that initialization to each ppc_md
	 */
	ppc_md.phys_mem_access_prot = pci_phys_mem_access_prot;

96 97
	if (pci_probe_only)
		ppc_pci_flags |= PPC_PCI_PROBE_ONLY;
L
Linus Torvalds 已提交
98

99 100 101 102 103
	/* On ppc64, we always enable PCI domains and we keep domain 0
	 * backward compatible in /proc for video cards
	 */
	ppc_pci_flags |= PPC_PCI_ENABLE_PROC_DOMAINS | PPC_PCI_COMPAT_DOMAIN_0;

L
Linus Torvalds 已提交
104
	/* Scan all of the recorded PCI controllers.  */
105
	list_for_each_entry_safe(hose, tmp, &hose_list, list_node) {
106
		scan_phb(hose);
107 108
		pci_bus_add_devices(hose->bus);
	}
L
Linus Torvalds 已提交
109

110 111
	/* Call common code to handle resource allocation */
	pcibios_resource_survey();
L
Linus Torvalds 已提交
112

113
	printk(KERN_DEBUG "PCI: Probing PCI hardware done\n");
L
Linus Torvalds 已提交
114 115 116 117 118 119

	return 0;
}

subsys_initcall(pcibios_init);

120 121 122
#ifdef CONFIG_HOTPLUG

int pcibios_unmap_io_space(struct pci_bus *bus)
L
Linus Torvalds 已提交
123
{
124
	struct pci_controller *hose;
L
Linus Torvalds 已提交
125

126
	WARN_ON(bus == NULL);
127

128 129 130 131 132 133
	/* If this is not a PHB, we only flush the hash table over
	 * the area mapped by this bridge. We don't play with the PTE
	 * mappings since we might have to deal with sub-page alignemnts
	 * so flushing the hash table is the only sane way to make sure
	 * that no hash entries are covering that removed bridge area
	 * while still allowing other busses overlapping those pages
134 135 136
	 *
	 * Note: If we ever support P2P hotplug on Book3E, we'll have
	 * to do an appropriate TLB flush here too
137 138 139
	 */
	if (bus->self) {
		struct resource *res = bus->resource[0];
L
Linus Torvalds 已提交
140

141 142
		pr_debug("IO unmapping for PCI-PCI bridge %s\n",
			 pci_name(bus->self));
143

144
#ifdef CONFIG_PPC_STD_MMU_64
145
		__flush_hash_table_range(&init_mm, res->start + _IO_BASE,
146
					 res->end + _IO_BASE + 1);
147
#endif
148 149
		return 0;
	}
L
Linus Torvalds 已提交
150

151 152
	/* Get the host bridge */
	hose = pci_bus_to_host(bus);
L
Linus Torvalds 已提交
153

154 155 156
	/* Check if we have IOs allocated */
	if (hose->io_base_alloc == 0)
		return 0;
157

158 159
	pr_debug("IO unmapping for PHB %s\n", hose->dn->full_name);
	pr_debug("  alloc=0x%p\n", hose->io_base_alloc);
L
Linus Torvalds 已提交
160

161 162
	/* This is a PHB, we fully unmap the IO area */
	vunmap(hose->io_base_alloc);
L
Linus Torvalds 已提交
163

164
	return 0;
L
Linus Torvalds 已提交
165
}
166
EXPORT_SYMBOL_GPL(pcibios_unmap_io_space);
L
Linus Torvalds 已提交
167

168
#endif /* CONFIG_HOTPLUG */
L
Linus Torvalds 已提交
169

170
int __devinit pcibios_map_io_space(struct pci_bus *bus)
L
Linus Torvalds 已提交
171
{
172 173 174 175 176
	struct vm_struct *area;
	unsigned long phys_page;
	unsigned long size_page;
	unsigned long io_virt_offset;
	struct pci_controller *hose;
177

178
	WARN_ON(bus == NULL);
179

180 181 182 183
	/* If this not a PHB, nothing to do, page tables still exist and
	 * thus HPTEs will be faulted in when needed
	 */
	if (bus->self) {
184 185
		pr_debug("IO mapping for PCI-PCI bridge %s\n",
			 pci_name(bus->self));
186
		pr_debug("  virt=0x%016llx...0x%016llx\n",
187 188
			 bus->resource[0]->start + _IO_BASE,
			 bus->resource[0]->end + _IO_BASE);
189
		return 0;
L
Linus Torvalds 已提交
190 191
	}

192 193 194 195
	/* Get the host bridge */
	hose = pci_bus_to_host(bus);
	phys_page = _ALIGN_DOWN(hose->io_base_phys, PAGE_SIZE);
	size_page = _ALIGN_UP(hose->pci_io_size, PAGE_SIZE);
L
Linus Torvalds 已提交
196

197 198
	/* Make sure IO area address is clear */
	hose->io_base_alloc = NULL;
L
Linus Torvalds 已提交
199

200 201 202
	/* If there's no IO to map on that bus, get away too */
	if (hose->pci_io_size == 0 || hose->io_base_phys == 0)
		return 0;
L
Linus Torvalds 已提交
203

204 205 206 207 208 209 210 211 212 213 214 215 216
	/* Let's allocate some IO space for that guy. We don't pass
	 * VM_IOREMAP because we don't care about alignment tricks that
	 * the core does in that case. Maybe we should due to stupid card
	 * with incomplete address decoding but I'd rather not deal with
	 * those outside of the reserved 64K legacy region.
	 */
	area = __get_vm_area(size_page, 0, PHB_IO_BASE, PHB_IO_END);
	if (area == NULL)
		return -ENOMEM;
	hose->io_base_alloc = area->addr;
	hose->io_base_virt = (void __iomem *)(area->addr +
					      hose->io_base_phys - phys_page);

217
	pr_debug("IO mapping for PHB %s\n", hose->dn->full_name);
218
	pr_debug("  phys=0x%016llx, virt=0x%p (alloc=0x%p)\n",
219
		 hose->io_base_phys, hose->io_base_virt, hose->io_base_alloc);
220
	pr_debug("  size=0x%016llx (alloc=0x%016lx)\n",
221
		 hose->pci_io_size, size_page);
222 223 224 225 226 227 228 229 230 231 232

	/* Establish the mapping */
	if (__ioremap_at(phys_page, area->addr, size_page,
			 _PAGE_NO_CACHE | _PAGE_GUARDED) == NULL)
		return -ENOMEM;

	/* Fixup hose IO resource */
	io_virt_offset = (unsigned long)hose->io_base_virt - _IO_BASE;
	hose->io_resource.start += io_virt_offset;
	hose->io_resource.end += io_virt_offset;

233
	pr_debug("  hose->io_resource=0x%016llx...0x%016llx\n",
234
		 hose->io_resource.start, hose->io_resource.end);
L
Linus Torvalds 已提交
235 236 237

	return 0;
}
238
EXPORT_SYMBOL_GPL(pcibios_map_io_space);
L
Linus Torvalds 已提交
239

240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258
#define IOBASE_BRIDGE_NUMBER	0
#define IOBASE_MEMORY		1
#define IOBASE_IO		2
#define IOBASE_ISA_IO		3
#define IOBASE_ISA_MEM		4

long sys_pciconfig_iobase(long which, unsigned long in_bus,
			  unsigned long in_devfn)
{
	struct pci_controller* hose;
	struct list_head *ln;
	struct pci_bus *bus = NULL;
	struct device_node *hose_node;

	/* Argh ! Please forgive me for that hack, but that's the
	 * simplest way to get existing XFree to not lockup on some
	 * G5 machines... So when something asks for bus 0 io base
	 * (bus 0 is HT root), we return the AGP one instead.
	 */
259 260 261 262 263
	if (in_bus == 0 && machine_is_compatible("MacRISC4")) {
		struct device_node *agp;

		agp = of_find_compatible_node(NULL, NULL, "u3-agp");
		if (agp)
264
			in_bus = 0xf0;
265 266
		of_node_put(agp);
	}
267 268 269 270 271 272 273

	/* That syscall isn't quite compatible with PCI domains, but it's
	 * used on pre-domains setup. We return the first match
	 */

	for (ln = pci_root_buses.next; ln != &pci_root_buses; ln = ln->next) {
		bus = pci_bus_b(ln);
274
		if (in_bus >= bus->number && in_bus <= bus->subordinate)
275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298
			break;
		bus = NULL;
	}
	if (bus == NULL || bus->sysdata == NULL)
		return -ENODEV;

	hose_node = (struct device_node *)bus->sysdata;
	hose = PCI_DN(hose_node)->phb;

	switch (which) {
	case IOBASE_BRIDGE_NUMBER:
		return (long)hose->first_busno;
	case IOBASE_MEMORY:
		return (long)hose->pci_mem_offset;
	case IOBASE_IO:
		return (long)hose->io_base_phys;
	case IOBASE_ISA_IO:
		return (long)isa_io_base;
	case IOBASE_ISA_MEM:
		return -EINVAL;
	}

	return -EOPNOTSUPP;
}
A
Anton Blanchard 已提交
299 300 301 302 303 304 305 306 307

#ifdef CONFIG_NUMA
int pcibus_to_node(struct pci_bus *bus)
{
	struct pci_controller *phb = pci_bus_to_host(bus);
	return phb->node;
}
EXPORT_SYMBOL(pcibus_to_node);
#endif