diff options
author | Bjorn Helgaas <bhelgaas@google.com> | 2012-02-23 22:19:02 -0500 |
---|---|---|
committer | Bjorn Helgaas <bhelgaas@google.com> | 2012-02-23 22:19:02 -0500 |
commit | 96a6b9ad05a2d5654f168d3ad9d9696d3b03a7c2 (patch) | |
tree | 8ca03136d6ab471f757da885cdabd485649dce7a /arch/mips/pci/pci.c | |
parent | aa23bdc0a8a29029ac4c89163a4b63a029580720 (diff) |
mips/PCI: get rid of device resource fixups
Tell the PCI core about host bridge address translation so it can take
care of bus-to-resource conversion for us.
Here's the wrinkle on Cobalt: we can't generate normal I/O port addresses
on PCI because the GT-64111 doesn't do any address translation, so we have
this:
CPU I/O port addresses [io 0x0000-0xffffff]
PCI bus I/O port addresses [io 0x10000000-0x10ffffff]
Legacy-mode IDE controllers start out with the legacy bus addresses, e.g.,
0x1f0, assigned by pci_setup_device(). These are outside the range of
addresses GT-64111 can generate on PCI, but pcibios_fixup_device_resources()
converted them to CPU addresses anyway by adding io_offset. Therefore, we
had to pre-adjust them in cobalt_legacy_ide_fixup().
With io_offset = 0xf0000000, we had this:
res->start = 0x1f0 initialized in pci_setup_device()
res->start = 0x100001f0 -= io_offset in cobalt_legacy_ide_fixup()
res->start = 0x1f0 += io_offset in pcibios_fixup_device_resources()
The difference after this patch is that the generic pci_bus_to_resource()
only adds the offset if the bus address is inside a host bridge window.
Since 0x1f0 is not a valid bus address and is not inside any windows, it is
unaffected, so we now have this:
region->start = 0x1f0 initialized in pci_setup_device()
res->start = 0x1f0 no offset by pci_bus_to_resource()
That means we can remove both pcibios_fixup_device_resources() and
cobalt_legacy_ide_fixup().
I would *rather* set the host bridge offset to zero (which corresponds
to what the GT-64111 actually does), and have both CPU and PCI addresses
of [io 0x10000000-0x10ffffff]. However, that would require changes to
generic code that assumes legacy I/O addresses, such as pic1_io_resource
([io 0x0020-0x00021]), and we'd have to keep a Cobalt IDE fixup.
Of course, none of this changes the fact that references to I/O port
0x1f0 actually go to port 0x100001f0, not 0x1f0, on the Cobalt PCI bus.
Fortunately the VT82C586 IDE controller only decodes the low 24 address
bits, so it does work.
CC: Ralf Baechle <ralf@linux-mips.org>
CC: Yoichi Yuasa <yuasa@linux-mips.org>
Signed-off-by: Bjorn Helgaas <bhelgaas@google.com>
Diffstat (limited to 'arch/mips/pci/pci.c')
-rw-r--r-- | arch/mips/pci/pci.c | 70 |
1 files changed, 3 insertions, 67 deletions
diff --git a/arch/mips/pci/pci.c b/arch/mips/pci/pci.c index 19f6d194a568..0514866fa925 100644 --- a/arch/mips/pci/pci.c +++ b/arch/mips/pci/pci.c | |||
@@ -88,8 +88,9 @@ static void __devinit pcibios_scanbus(struct pci_controller *hose) | |||
88 | if (hose->get_busno && pci_has_flag(PCI_PROBE_ONLY)) | 88 | if (hose->get_busno && pci_has_flag(PCI_PROBE_ONLY)) |
89 | next_busno = (*hose->get_busno)(); | 89 | next_busno = (*hose->get_busno)(); |
90 | 90 | ||
91 | pci_add_resource(&resources, hose->mem_resource); | 91 | pci_add_resource_offset(&resources, |
92 | pci_add_resource(&resources, hose->io_resource); | 92 | hose->mem_resource, hose->mem_offset); |
93 | pci_add_resource_offset(&resources, hose->io_resource, hose->io_offset); | ||
93 | bus = pci_scan_root_bus(NULL, next_busno, hose->pci_ops, hose, | 94 | bus = pci_scan_root_bus(NULL, next_busno, hose->pci_ops, hose, |
94 | &resources); | 95 | &resources); |
95 | if (!bus) | 96 | if (!bus) |
@@ -247,45 +248,13 @@ int pcibios_enable_device(struct pci_dev *dev, int mask) | |||
247 | return pcibios_plat_dev_init(dev); | 248 | return pcibios_plat_dev_init(dev); |
248 | } | 249 | } |
249 | 250 | ||
250 | static void pcibios_fixup_device_resources(struct pci_dev *dev, | ||
251 | struct pci_bus *bus) | ||
252 | { | ||
253 | /* Update device resources. */ | ||
254 | struct pci_controller *hose = (struct pci_controller *)bus->sysdata; | ||
255 | unsigned long offset = 0; | ||
256 | int i; | ||
257 | |||
258 | for (i = 0; i < PCI_NUM_RESOURCES; i++) { | ||
259 | if (!dev->resource[i].start) | ||
260 | continue; | ||
261 | if (dev->resource[i].flags & IORESOURCE_IO) | ||
262 | offset = hose->io_offset; | ||
263 | else if (dev->resource[i].flags & IORESOURCE_MEM) | ||
264 | offset = hose->mem_offset; | ||
265 | |||
266 | dev->resource[i].start += offset; | ||
267 | dev->resource[i].end += offset; | ||
268 | } | ||
269 | } | ||
270 | |||
271 | void __devinit pcibios_fixup_bus(struct pci_bus *bus) | 251 | void __devinit pcibios_fixup_bus(struct pci_bus *bus) |
272 | { | 252 | { |
273 | /* Propagate hose info into the subordinate devices. */ | ||
274 | |||
275 | struct list_head *ln; | ||
276 | struct pci_dev *dev = bus->self; | 253 | struct pci_dev *dev = bus->self; |
277 | 254 | ||
278 | if (pci_has_flag(PCI_PROBE_ONLY) && dev && | 255 | if (pci_has_flag(PCI_PROBE_ONLY) && dev && |
279 | (dev->class >> 8) == PCI_CLASS_BRIDGE_PCI) { | 256 | (dev->class >> 8) == PCI_CLASS_BRIDGE_PCI) { |
280 | pci_read_bridge_bases(bus); | 257 | pci_read_bridge_bases(bus); |
281 | pcibios_fixup_device_resources(dev, bus); | ||
282 | } | ||
283 | |||
284 | for (ln = bus->devices.next; ln != &bus->devices; ln = ln->next) { | ||
285 | dev = pci_dev_b(ln); | ||
286 | |||
287 | if ((dev->class >> 8) != PCI_CLASS_BRIDGE_PCI) | ||
288 | pcibios_fixup_device_resources(dev, bus); | ||
289 | } | 258 | } |
290 | } | 259 | } |
291 | 260 | ||
@@ -295,40 +264,7 @@ pcibios_update_irq(struct pci_dev *dev, int irq) | |||
295 | pci_write_config_byte(dev, PCI_INTERRUPT_LINE, irq); | 264 | pci_write_config_byte(dev, PCI_INTERRUPT_LINE, irq); |
296 | } | 265 | } |
297 | 266 | ||
298 | void pcibios_resource_to_bus(struct pci_dev *dev, struct pci_bus_region *region, | ||
299 | struct resource *res) | ||
300 | { | ||
301 | struct pci_controller *hose = (struct pci_controller *)dev->sysdata; | ||
302 | unsigned long offset = 0; | ||
303 | |||
304 | if (res->flags & IORESOURCE_IO) | ||
305 | offset = hose->io_offset; | ||
306 | else if (res->flags & IORESOURCE_MEM) | ||
307 | offset = hose->mem_offset; | ||
308 | |||
309 | region->start = res->start - offset; | ||
310 | region->end = res->end - offset; | ||
311 | } | ||
312 | |||
313 | void __devinit | ||
314 | pcibios_bus_to_resource(struct pci_dev *dev, struct resource *res, | ||
315 | struct pci_bus_region *region) | ||
316 | { | ||
317 | struct pci_controller *hose = (struct pci_controller *)dev->sysdata; | ||
318 | unsigned long offset = 0; | ||
319 | |||
320 | if (res->flags & IORESOURCE_IO) | ||
321 | offset = hose->io_offset; | ||
322 | else if (res->flags & IORESOURCE_MEM) | ||
323 | offset = hose->mem_offset; | ||
324 | |||
325 | res->start = region->start + offset; | ||
326 | res->end = region->end + offset; | ||
327 | } | ||
328 | |||
329 | #ifdef CONFIG_HOTPLUG | 267 | #ifdef CONFIG_HOTPLUG |
330 | EXPORT_SYMBOL(pcibios_resource_to_bus); | ||
331 | EXPORT_SYMBOL(pcibios_bus_to_resource); | ||
332 | EXPORT_SYMBOL(PCIBIOS_MIN_IO); | 268 | EXPORT_SYMBOL(PCIBIOS_MIN_IO); |
333 | EXPORT_SYMBOL(PCIBIOS_MIN_MEM); | 269 | EXPORT_SYMBOL(PCIBIOS_MIN_MEM); |
334 | #endif | 270 | #endif |