blob: 7442775ef2a1dd864a01b515d79b611bd075fcaf [file] [log] [blame]
Arnd Bergmannc5a3c2e2005-06-23 09:43:23 +10001/*
2 * arch/ppc64/kernel/rtas_pci.c
3 *
4 * Copyright (C) 2001 Dave Engebretsen, IBM Corporation
5 * Copyright (C) 2003 Anton Blanchard <anton@au.ibm.com>, IBM
6 *
7 * RTAS specific routines for PCI.
linasae65a392005-11-03 18:42:26 -06008 *
Arnd Bergmannc5a3c2e2005-06-23 09:43:23 +10009 * Based on code from pci.c, chrp_pci.c and pSeries_pci.c
10 *
11 * This program is free software; you can redistribute it and/or modify
12 * it under the terms of the GNU General Public License as published by
13 * the Free Software Foundation; either version 2 of the License, or
14 * (at your option) any later version.
linasae65a392005-11-03 18:42:26 -060015 *
Arnd Bergmannc5a3c2e2005-06-23 09:43:23 +100016 * This program is distributed in the hope that it will be useful,
17 * but WITHOUT ANY WARRANTY; without even the implied warranty of
18 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
19 * GNU General Public License for more details.
linasae65a392005-11-03 18:42:26 -060020 *
Arnd Bergmannc5a3c2e2005-06-23 09:43:23 +100021 * You should have received a copy of the GNU General Public License
22 * along with this program; if not, write to the Free Software
23 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
24 */
25
26#include <linux/kernel.h>
27#include <linux/threads.h>
28#include <linux/pci.h>
29#include <linux/string.h>
30#include <linux/init.h>
31#include <linux/bootmem.h>
32
33#include <asm/io.h>
34#include <asm/pgtable.h>
35#include <asm/irq.h>
36#include <asm/prom.h>
37#include <asm/machdep.h>
38#include <asm/pci-bridge.h>
39#include <asm/iommu.h>
40#include <asm/rtas.h>
Stephen Rothwellbbeb3f42005-09-27 13:51:59 +100041#include <asm/mpic.h>
Stephen Rothwelld3878992005-09-28 02:50:25 +100042#include <asm/ppc-pci.h>
Arnd Bergmannc5a3c2e2005-06-23 09:43:23 +100043
44/* RTAS tokens */
45static int read_pci_config;
46static int write_pci_config;
47static int ibm_read_pci_config;
48static int ibm_write_pci_config;
49
linasae65a392005-11-03 18:42:26 -060050static inline int config_access_valid(struct pci_dn *dn, int where)
Arnd Bergmannc5a3c2e2005-06-23 09:43:23 +100051{
52 if (where < 256)
53 return 1;
54 if (where < 4096 && dn->pci_ext_config_space)
55 return 1;
56
57 return 0;
58}
59
Jake Moilanen293da762005-06-09 09:31:12 -050060static int of_device_available(struct device_node * dn)
61{
62 char * status;
63
64 status = get_property(dn, "status", NULL);
65
66 if (!status)
67 return 1;
68
69 if (!strcmp(status, "okay"))
70 return 1;
71
72 return 0;
73}
74
Linas Vepstas7684b402005-11-03 18:55:19 -060075int rtas_read_config(struct pci_dn *pdn, int where, int size, u32 *val)
Arnd Bergmannc5a3c2e2005-06-23 09:43:23 +100076{
77 int returnval = -1;
78 unsigned long buid, addr;
79 int ret;
80
linasae65a392005-11-03 18:42:26 -060081 if (!pdn)
Arnd Bergmannc5a3c2e2005-06-23 09:43:23 +100082 return PCIBIOS_DEVICE_NOT_FOUND;
Paul Mackerras16353172005-09-06 13:17:54 +100083 if (!config_access_valid(pdn, where))
Arnd Bergmannc5a3c2e2005-06-23 09:43:23 +100084 return PCIBIOS_BAD_REGISTER_NUMBER;
85
Paul Mackerras16353172005-09-06 13:17:54 +100086 addr = ((where & 0xf00) << 20) | (pdn->busno << 16) |
87 (pdn->devfn << 8) | (where & 0xff);
88 buid = pdn->phb->buid;
Arnd Bergmannc5a3c2e2005-06-23 09:43:23 +100089 if (buid) {
90 ret = rtas_call(ibm_read_pci_config, 4, 2, &returnval,
linasae65a392005-11-03 18:42:26 -060091 addr, BUID_HI(buid), BUID_LO(buid), size);
Arnd Bergmannc5a3c2e2005-06-23 09:43:23 +100092 } else {
93 ret = rtas_call(read_pci_config, 2, 2, &returnval, addr, size);
94 }
95 *val = returnval;
96
97 if (ret)
98 return PCIBIOS_DEVICE_NOT_FOUND;
99
Paul Mackerras16353172005-09-06 13:17:54 +1000100 if (returnval == EEH_IO_ERROR_VALUE(size) &&
linasae65a392005-11-03 18:42:26 -0600101 eeh_dn_check_failure (pdn->node, NULL))
Arnd Bergmannc5a3c2e2005-06-23 09:43:23 +1000102 return PCIBIOS_DEVICE_NOT_FOUND;
103
104 return PCIBIOS_SUCCESSFUL;
105}
106
107static int rtas_pci_read_config(struct pci_bus *bus,
108 unsigned int devfn,
109 int where, int size, u32 *val)
110{
111 struct device_node *busdn, *dn;
112
113 if (bus->self)
114 busdn = pci_device_to_OF_node(bus->self);
115 else
116 busdn = bus->sysdata; /* must be a phb */
117
118 /* Search only direct children of the bus */
linasae65a392005-11-03 18:42:26 -0600119 for (dn = busdn->child; dn; dn = dn->sibling) {
120 struct pci_dn *pdn = PCI_DN(dn);
121 if (pdn && pdn->devfn == devfn
Paul Mackerras16353172005-09-06 13:17:54 +1000122 && of_device_available(dn))
linasae65a392005-11-03 18:42:26 -0600123 return rtas_read_config(pdn, where, size, val);
124 }
Paul Mackerras16353172005-09-06 13:17:54 +1000125
Arnd Bergmannc5a3c2e2005-06-23 09:43:23 +1000126 return PCIBIOS_DEVICE_NOT_FOUND;
127}
128
linasae65a392005-11-03 18:42:26 -0600129int rtas_write_config(struct pci_dn *pdn, int where, int size, u32 val)
Arnd Bergmannc5a3c2e2005-06-23 09:43:23 +1000130{
131 unsigned long buid, addr;
132 int ret;
133
linasae65a392005-11-03 18:42:26 -0600134 if (!pdn)
Arnd Bergmannc5a3c2e2005-06-23 09:43:23 +1000135 return PCIBIOS_DEVICE_NOT_FOUND;
Paul Mackerras16353172005-09-06 13:17:54 +1000136 if (!config_access_valid(pdn, where))
Arnd Bergmannc5a3c2e2005-06-23 09:43:23 +1000137 return PCIBIOS_BAD_REGISTER_NUMBER;
138
Paul Mackerras16353172005-09-06 13:17:54 +1000139 addr = ((where & 0xf00) << 20) | (pdn->busno << 16) |
140 (pdn->devfn << 8) | (where & 0xff);
141 buid = pdn->phb->buid;
Arnd Bergmannc5a3c2e2005-06-23 09:43:23 +1000142 if (buid) {
linasae65a392005-11-03 18:42:26 -0600143 ret = rtas_call(ibm_write_pci_config, 5, 1, NULL, addr,
144 BUID_HI(buid), BUID_LO(buid), size, (ulong) val);
Arnd Bergmannc5a3c2e2005-06-23 09:43:23 +1000145 } else {
146 ret = rtas_call(write_pci_config, 3, 1, NULL, addr, size, (ulong)val);
147 }
148
149 if (ret)
150 return PCIBIOS_DEVICE_NOT_FOUND;
151
152 return PCIBIOS_SUCCESSFUL;
153}
154
155static int rtas_pci_write_config(struct pci_bus *bus,
156 unsigned int devfn,
157 int where, int size, u32 val)
158{
159 struct device_node *busdn, *dn;
160
161 if (bus->self)
162 busdn = pci_device_to_OF_node(bus->self);
163 else
164 busdn = bus->sysdata; /* must be a phb */
165
166 /* Search only direct children of the bus */
linasae65a392005-11-03 18:42:26 -0600167 for (dn = busdn->child; dn; dn = dn->sibling) {
168 struct pci_dn *pdn = PCI_DN(dn);
169 if (pdn && pdn->devfn == devfn
Paul Mackerras16353172005-09-06 13:17:54 +1000170 && of_device_available(dn))
linasae65a392005-11-03 18:42:26 -0600171 return rtas_write_config(pdn, where, size, val);
172 }
Arnd Bergmannc5a3c2e2005-06-23 09:43:23 +1000173 return PCIBIOS_DEVICE_NOT_FOUND;
174}
175
176struct pci_ops rtas_pci_ops = {
177 rtas_pci_read_config,
178 rtas_pci_write_config
179};
180
181int is_python(struct device_node *dev)
182{
183 char *model = (char *)get_property(dev, "model", NULL);
184
185 if (model && strstr(model, "Python"))
186 return 1;
187
188 return 0;
189}
190
Benjamin Herrenschmidtcc5d0182005-12-13 18:01:21 +1100191static void python_countermeasures(struct device_node *dev)
Arnd Bergmannc5a3c2e2005-06-23 09:43:23 +1000192{
Benjamin Herrenschmidtcc5d0182005-12-13 18:01:21 +1100193 struct resource registers;
Arnd Bergmannc5a3c2e2005-06-23 09:43:23 +1000194 void __iomem *chip_regs;
195 volatile u32 val;
196
Benjamin Herrenschmidtcc5d0182005-12-13 18:01:21 +1100197 if (of_address_to_resource(dev, 0, &registers)) {
198 printk(KERN_ERR "Can't get address for Python workarounds !\n");
Arnd Bergmannc5a3c2e2005-06-23 09:43:23 +1000199 return;
Benjamin Herrenschmidtcc5d0182005-12-13 18:01:21 +1100200 }
Arnd Bergmannc5a3c2e2005-06-23 09:43:23 +1000201
202 /* Python's register file is 1 MB in size. */
Benjamin Herrenschmidtcc5d0182005-12-13 18:01:21 +1100203 chip_regs = ioremap(registers.start & ~(0xfffffUL), 0x100000);
Arnd Bergmannc5a3c2e2005-06-23 09:43:23 +1000204
linasae65a392005-11-03 18:42:26 -0600205 /*
Arnd Bergmannc5a3c2e2005-06-23 09:43:23 +1000206 * Firmware doesn't always clear this bit which is critical
207 * for good performance - Anton
208 */
209
210#define PRG_CL_RESET_VALID 0x00010000
211
212 val = in_be32(chip_regs + 0xf6030);
213 if (val & PRG_CL_RESET_VALID) {
214 printk(KERN_INFO "Python workaround: ");
215 val &= ~PRG_CL_RESET_VALID;
216 out_be32(chip_regs + 0xf6030, val);
217 /*
218 * We must read it back for changes to
219 * take effect
220 */
221 val = in_be32(chip_regs + 0xf6030);
222 printk("reg0: %x\n", val);
223 }
224
225 iounmap(chip_regs);
226}
227
228void __init init_pci_config_tokens (void)
229{
230 read_pci_config = rtas_token("read-pci-config");
231 write_pci_config = rtas_token("write-pci-config");
232 ibm_read_pci_config = rtas_token("ibm,read-pci-config");
233 ibm_write_pci_config = rtas_token("ibm,write-pci-config");
234}
235
236unsigned long __devinit get_phb_buid (struct device_node *phb)
237{
238 int addr_cells;
239 unsigned int *buid_vals;
240 unsigned int len;
241 unsigned long buid;
242
243 if (ibm_read_pci_config == -1) return 0;
244
245 /* PHB's will always be children of the root node,
246 * or so it is promised by the current firmware. */
247 if (phb->parent == NULL)
248 return 0;
249 if (phb->parent->parent)
250 return 0;
251
252 buid_vals = (unsigned int *) get_property(phb, "reg", &len);
253 if (buid_vals == NULL)
254 return 0;
255
256 addr_cells = prom_n_addr_cells(phb);
257 if (addr_cells == 1) {
258 buid = (unsigned long) buid_vals[0];
259 } else {
260 buid = (((unsigned long)buid_vals[0]) << 32UL) |
261 (((unsigned long)buid_vals[1]) & 0xffffffff);
262 }
263 return buid;
264}
265
266static int phb_set_bus_ranges(struct device_node *dev,
267 struct pci_controller *phb)
268{
269 int *bus_range;
270 unsigned int len;
271
272 bus_range = (int *) get_property(dev, "bus-range", &len);
273 if (bus_range == NULL || len < 2 * sizeof(int)) {
274 return 1;
275 }
linasae65a392005-11-03 18:42:26 -0600276
Arnd Bergmannc5a3c2e2005-06-23 09:43:23 +1000277 phb->first_busno = bus_range[0];
278 phb->last_busno = bus_range[1];
279
280 return 0;
281}
282
John Rose92eb4602006-03-14 17:46:45 -0600283int __devinit setup_phb(struct device_node *dev, struct pci_controller *phb)
Arnd Bergmannc5a3c2e2005-06-23 09:43:23 +1000284{
Arnd Bergmannc5a3c2e2005-06-23 09:43:23 +1000285 if (is_python(dev))
Benjamin Herrenschmidtcc5d0182005-12-13 18:01:21 +1100286 python_countermeasures(dev);
Arnd Bergmannc5a3c2e2005-06-23 09:43:23 +1000287
288 if (phb_set_bus_ranges(dev, phb))
289 return 1;
290
Arnd Bergmannc5a3c2e2005-06-23 09:43:23 +1000291 phb->ops = &rtas_pci_ops;
292 phb->buid = get_phb_buid(dev);
293
294 return 0;
295}
296
Arnd Bergmannc5a3c2e2005-06-23 09:43:23 +1000297unsigned long __init find_and_init_phbs(void)
298{
299 struct device_node *node;
300 struct pci_controller *phb;
Arnd Bergmannc5a3c2e2005-06-23 09:43:23 +1000301 unsigned int index;
Benjamin Herrenschmidtcc5d0182005-12-13 18:01:21 +1100302 unsigned int root_size_cells = 0;
Arnd Bergmannc5a3c2e2005-06-23 09:43:23 +1000303 unsigned int *opprop = NULL;
304 struct device_node *root = of_find_node_by_path("/");
305
306 if (ppc64_interrupt_controller == IC_OPEN_PIC) {
307 opprop = (unsigned int *)get_property(root,
308 "platform-open-pic", NULL);
309 }
310
311 root_size_cells = prom_n_size_cells(root);
312
313 index = 0;
314
315 for (node = of_get_next_child(root, NULL);
316 node != NULL;
317 node = of_get_next_child(root, node)) {
318 if (node->type == NULL || strcmp(node->type, "pci") != 0)
319 continue;
320
Benjamin Herrenschmidtb5166cc2005-11-15 16:05:33 +1100321 phb = pcibios_alloc_controller(node);
Arnd Bergmannc5a3c2e2005-06-23 09:43:23 +1000322 if (!phb)
323 continue;
Benjamin Herrenschmidtcc5d0182005-12-13 18:01:21 +1100324 setup_phb(node, phb);
Paul Mackerrasf7abbc12005-10-22 15:03:21 +1000325 pci_process_bridge_OF_ranges(phb, node, 0);
Arnd Bergmannc5a3c2e2005-06-23 09:43:23 +1000326 pci_setup_phb_io(phb, index == 0);
327#ifdef CONFIG_PPC_PSERIES
Benjamin Herrenschmidtcc5d0182005-12-13 18:01:21 +1100328 /* XXX This code need serious fixing ... --BenH */
Arnd Bergmannc5a3c2e2005-06-23 09:43:23 +1000329 if (ppc64_interrupt_controller == IC_OPEN_PIC && pSeries_mpic) {
330 int addr = root_size_cells * (index + 2) - 1;
331 mpic_assign_isu(pSeries_mpic, index, opprop[addr]);
332 }
333#endif
334 index++;
335 }
336
337 of_node_put(root);
338 pci_devs_phb_init();
339
340 /*
341 * pci_probe_only and pci_assign_all_buses can be set via properties
342 * in chosen.
343 */
344 if (of_chosen) {
345 int *prop;
346
347 prop = (int *)get_property(of_chosen, "linux,pci-probe-only",
348 NULL);
349 if (prop)
350 pci_probe_only = *prop;
351
352 prop = (int *)get_property(of_chosen,
353 "linux,pci-assign-all-buses", NULL);
354 if (prop)
355 pci_assign_all_buses = *prop;
356 }
357
358 return 0;
359}
360
Arnd Bergmannc5a3c2e2005-06-23 09:43:23 +1000361/* RPA-specific bits for removing PHBs */
362int pcibios_remove_root_bus(struct pci_controller *phb)
363{
364 struct pci_bus *b = phb->bus;
365 struct resource *res;
366 int rc, i;
367
368 res = b->resource[0];
369 if (!res->flags) {
370 printk(KERN_ERR "%s: no IO resource for PHB %s\n", __FUNCTION__,
371 b->name);
372 return 1;
373 }
374
375 rc = unmap_bus_range(b);
376 if (rc) {
377 printk(KERN_ERR "%s: failed to unmap IO on bus %s\n",
378 __FUNCTION__, b->name);
379 return 1;
380 }
381
382 if (release_resource(res)) {
383 printk(KERN_ERR "%s: failed to release IO on bus %s\n",
384 __FUNCTION__, b->name);
385 return 1;
386 }
387
388 for (i = 1; i < 3; ++i) {
389 res = b->resource[i];
390 if (!res->flags && i == 0) {
391 printk(KERN_ERR "%s: no MEM resource for PHB %s\n",
392 __FUNCTION__, b->name);
393 return 1;
394 }
395 if (res->flags && release_resource(res)) {
396 printk(KERN_ERR
397 "%s: failed to release IO %d on bus %s\n",
398 __FUNCTION__, i, b->name);
399 return 1;
400 }
401 }
402
403 list_del(&phb->list_node);
Benjamin Herrenschmidtb5166cc2005-11-15 16:05:33 +1100404 pcibios_free_controller(phb);
Arnd Bergmannc5a3c2e2005-06-23 09:43:23 +1000405
406 return 0;
407}
408EXPORT_SYMBOL(pcibios_remove_root_bus);