blob: 5c1118e31940bf39d3a88926a16fc883c4f30d5f [file] [log] [blame]
Benjamin Herrenschmidt014da7f2006-11-11 17:25:12 +11001/*
Ishizaki Kou7cfb62a2008-04-24 19:21:10 +10002 * Support PCI IO workaround
3 *
Benjamin Herrenschmidt014da7f2006-11-11 17:25:12 +11004 * Copyright (C) 2006 Benjamin Herrenschmidt <benh@kernel.crashing.org>
5 * IBM, Corp.
Ishizaki Kou7cfb62a2008-04-24 19:21:10 +10006 * (C) Copyright 2007-2008 TOSHIBA CORPORATION
Benjamin Herrenschmidt014da7f2006-11-11 17:25:12 +11007 *
8 * This program is free software; you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License version 2 as
10 * published by the Free Software Foundation.
11 */
12#undef DEBUG
13
14#include <linux/kernel.h>
Ishizaki Kou7cfb62a2008-04-24 19:21:10 +100015
Benjamin Herrenschmidt014da7f2006-11-11 17:25:12 +110016#include <asm/io.h>
17#include <asm/machdep.h>
Ishizaki Kou7cfb62a2008-04-24 19:21:10 +100018#include <asm/pgtable.h>
Benjamin Herrenschmidt014da7f2006-11-11 17:25:12 +110019#include <asm/ppc-pci.h>
20
Ishizaki Kou7cfb62a2008-04-24 19:21:10 +100021#include "io-workarounds.h"
Benjamin Herrenschmidt014da7f2006-11-11 17:25:12 +110022
Ishizaki Kou7cfb62a2008-04-24 19:21:10 +100023#define IOWA_MAX_BUS 8
Benjamin Herrenschmidt014da7f2006-11-11 17:25:12 +110024
Ishizaki Kou7cfb62a2008-04-24 19:21:10 +100025static struct iowa_bus iowa_busses[IOWA_MAX_BUS];
26static unsigned int iowa_bus_count;
Benjamin Herrenschmidt014da7f2006-11-11 17:25:12 +110027
Ishizaki Kou7cfb62a2008-04-24 19:21:10 +100028static struct iowa_bus *iowa_pci_find(unsigned long vaddr, unsigned long paddr)
Benjamin Herrenschmidt014da7f2006-11-11 17:25:12 +110029{
Ishizaki Kou7cfb62a2008-04-24 19:21:10 +100030 int i, j;
31 struct resource *res;
32 unsigned long vstart, vend;
Benjamin Herrenschmidt014da7f2006-11-11 17:25:12 +110033
Ishizaki Kou7cfb62a2008-04-24 19:21:10 +100034 for (i = 0; i < iowa_bus_count; i++) {
35 struct iowa_bus *bus = &iowa_busses[i];
36 struct pci_controller *phb = bus->phb;
37
38 if (vaddr) {
39 vstart = (unsigned long)phb->io_base_virt;
40 vend = vstart + phb->pci_io_size - 1;
41 if ((vaddr >= vstart) && (vaddr <= vend))
42 return bus;
43 }
44
45 if (paddr)
46 for (j = 0; j < 3; j++) {
47 res = &phb->mem_resources[j];
48 if (paddr >= res->start && paddr <= res->end)
49 return bus;
50 }
Benjamin Herrenschmidt014da7f2006-11-11 17:25:12 +110051 }
Ishizaki Kou7cfb62a2008-04-24 19:21:10 +100052
Benjamin Herrenschmidt014da7f2006-11-11 17:25:12 +110053 return NULL;
54}
55
Ishizaki Kou7cfb62a2008-04-24 19:21:10 +100056struct iowa_bus *iowa_mem_find_bus(const PCI_IO_ADDR addr)
Benjamin Herrenschmidt014da7f2006-11-11 17:25:12 +110057{
Ishizaki Kou7cfb62a2008-04-24 19:21:10 +100058 struct iowa_bus *bus;
Benjamin Herrenschmidt014da7f2006-11-11 17:25:12 +110059 int token;
60
Benjamin Herrenschmidt014da7f2006-11-11 17:25:12 +110061 token = PCI_GET_ADDR_TOKEN(addr);
62
Ishizaki Kou7cfb62a2008-04-24 19:21:10 +100063 if (token && token <= iowa_bus_count)
64 bus = &iowa_busses[token - 1];
Benjamin Herrenschmidt014da7f2006-11-11 17:25:12 +110065 else {
66 unsigned long vaddr, paddr;
67 pte_t *ptep;
68
Benjamin Herrenschmidt014da7f2006-11-11 17:25:12 +110069 vaddr = (unsigned long)PCI_FIX_ADDR(addr);
Ishizaki Kou7cfb62a2008-04-24 19:21:10 +100070 if (vaddr < PHB_IO_BASE || vaddr >= PHB_IO_END)
71 return NULL;
Benjamin Herrenschmidt014da7f2006-11-11 17:25:12 +110072
Benjamin Herrenschmidt014da7f2006-11-11 17:25:12 +110073 ptep = find_linux_pte(init_mm.pgd, vaddr);
74 if (ptep == NULL)
75 paddr = 0;
76 else
77 paddr = pte_pfn(*ptep) << PAGE_SHIFT;
Ishizaki Kou7cfb62a2008-04-24 19:21:10 +100078 bus = iowa_pci_find(vaddr, paddr);
Benjamin Herrenschmidt014da7f2006-11-11 17:25:12 +110079
Benjamin Herrenschmidt014da7f2006-11-11 17:25:12 +110080 if (bus == NULL)
Ishizaki Kou7cfb62a2008-04-24 19:21:10 +100081 return NULL;
Benjamin Herrenschmidt014da7f2006-11-11 17:25:12 +110082 }
83
Ishizaki Kou7cfb62a2008-04-24 19:21:10 +100084 return bus;
Benjamin Herrenschmidt014da7f2006-11-11 17:25:12 +110085}
86
Ishizaki Kou7cfb62a2008-04-24 19:21:10 +100087struct iowa_bus *iowa_pio_find_bus(unsigned long port)
Benjamin Herrenschmidt014da7f2006-11-11 17:25:12 +110088{
Ishizaki Kou7cfb62a2008-04-24 19:21:10 +100089 unsigned long vaddr = (unsigned long)pci_io_base + port;
90 return iowa_pci_find(vaddr, 0);
Benjamin Herrenschmidt014da7f2006-11-11 17:25:12 +110091}
92
93
Ishizaki Kou7cfb62a2008-04-24 19:21:10 +100094#define DEF_PCI_AC_RET(name, ret, at, al, space, aa) \
95static ret iowa_##name at \
96{ \
97 struct iowa_bus *bus; \
98 bus = iowa_##space##_find_bus(aa); \
99 if (bus && bus->ops && bus->ops->name) \
100 return bus->ops->name al; \
101 return __do_##name al; \
102}
103
104#define DEF_PCI_AC_NORET(name, at, al, space, aa) \
105static void iowa_##name at \
106{ \
107 struct iowa_bus *bus; \
108 bus = iowa_##space##_find_bus(aa); \
109 if (bus && bus->ops && bus->ops->name) { \
110 bus->ops->name al; \
111 return; \
112 } \
113 __do_##name al; \
114}
115
116#include <asm/io-defs.h>
117
118#undef DEF_PCI_AC_RET
119#undef DEF_PCI_AC_NORET
120
Ishizaki Kou70122552008-05-14 19:05:19 +1000121static const struct ppc_pci_io __devinitconst iowa_pci_io = {
Ishizaki Kou7cfb62a2008-04-24 19:21:10 +1000122
123#define DEF_PCI_AC_RET(name, ret, at, al, space, aa) .name = iowa_##name,
124#define DEF_PCI_AC_NORET(name, at, al, space, aa) .name = iowa_##name,
125
126#include <asm/io-defs.h>
127
128#undef DEF_PCI_AC_RET
129#undef DEF_PCI_AC_NORET
130
131};
132
Ingo Molnarb36ac9e2009-01-06 14:03:44 +0000133static void __iomem *iowa_ioremap(phys_addr_t addr, unsigned long size,
Benjamin Herrenschmidt1cdab552009-02-22 16:19:14 +0000134 unsigned long flags, void *caller)
Benjamin Herrenschmidt014da7f2006-11-11 17:25:12 +1100135{
Ishizaki Kou7cfb62a2008-04-24 19:21:10 +1000136 struct iowa_bus *bus;
Benjamin Herrenschmidt1cdab552009-02-22 16:19:14 +0000137 void __iomem *res = __ioremap_caller(addr, size, flags, caller);
Benjamin Herrenschmidt014da7f2006-11-11 17:25:12 +1100138 int busno;
139
Ingo Molnarb36ac9e2009-01-06 14:03:44 +0000140 bus = iowa_pci_find(0, (unsigned long)addr);
Benjamin Herrenschmidt014da7f2006-11-11 17:25:12 +1100141 if (bus != NULL) {
Ishizaki Kou7cfb62a2008-04-24 19:21:10 +1000142 busno = bus - iowa_busses;
Benjamin Herrenschmidt014da7f2006-11-11 17:25:12 +1100143 PCI_SET_ADDR_TOKEN(res, busno + 1);
144 }
Benjamin Herrenschmidt014da7f2006-11-11 17:25:12 +1100145 return res;
146}
147
Ishizaki Kou7cfb62a2008-04-24 19:21:10 +1000148/* Regist new bus to support workaround */
Ishizaki Kou70122552008-05-14 19:05:19 +1000149void __devinit iowa_register_bus(struct pci_controller *phb,
Ishizaki Kou7cfb62a2008-04-24 19:21:10 +1000150 struct ppc_pci_io *ops,
151 int (*initfunc)(struct iowa_bus *, void *), void *data)
Benjamin Herrenschmidt014da7f2006-11-11 17:25:12 +1100152{
Ishizaki Kou7cfb62a2008-04-24 19:21:10 +1000153 struct iowa_bus *bus;
Stephen Rothwell44ef3392007-12-10 14:33:21 +1100154 struct device_node *np = phb->dn;
Benjamin Herrenschmidt014da7f2006-11-11 17:25:12 +1100155
Ishizaki Kou7cfb62a2008-04-24 19:21:10 +1000156 if (iowa_bus_count >= IOWA_MAX_BUS) {
157 pr_err("IOWA:Too many pci bridges, "
158 "workarounds disabled for %s\n", np->full_name);
Benjamin Herrenschmidt014da7f2006-11-11 17:25:12 +1100159 return;
160 }
161
Ishizaki Kou7cfb62a2008-04-24 19:21:10 +1000162 bus = &iowa_busses[iowa_bus_count];
163 bus->phb = phb;
164 bus->ops = ops;
Benjamin Herrenschmidt014da7f2006-11-11 17:25:12 +1100165
Ishizaki Kou7cfb62a2008-04-24 19:21:10 +1000166 if (initfunc)
167 if ((*initfunc)(bus, data))
168 return;
Benjamin Herrenschmidt014da7f2006-11-11 17:25:12 +1100169
Ishizaki Kou7cfb62a2008-04-24 19:21:10 +1000170 iowa_bus_count++;
Benjamin Herrenschmidt014da7f2006-11-11 17:25:12 +1100171
Ishizaki Kou7cfb62a2008-04-24 19:21:10 +1000172 pr_debug("IOWA:[%d]Add bus, %s.\n", iowa_bus_count-1, np->full_name);
Benjamin Herrenschmidt014da7f2006-11-11 17:25:12 +1100173}
174
Ishizaki Kou7cfb62a2008-04-24 19:21:10 +1000175/* enable IO workaround */
Ishizaki Kou70122552008-05-14 19:05:19 +1000176void __devinit io_workaround_init(void)
Benjamin Herrenschmidt014da7f2006-11-11 17:25:12 +1100177{
Ishizaki Kou7cfb62a2008-04-24 19:21:10 +1000178 static int io_workaround_inited;
Benjamin Herrenschmidt014da7f2006-11-11 17:25:12 +1100179
Ishizaki Kou7cfb62a2008-04-24 19:21:10 +1000180 if (io_workaround_inited)
181 return;
182 ppc_pci_io = iowa_pci_io;
183 ppc_md.ioremap = iowa_ioremap;
184 io_workaround_inited = 1;
Benjamin Herrenschmidt014da7f2006-11-11 17:25:12 +1100185}