mirror of
https://github.com/edk2-porting/linux-next.git
synced 2025-01-04 11:43:54 +08:00
5ce4b59653
The 440EPx/GRx chips don't support PCI MRM commands. Drivers determine this by looking for a zero value in the PCI cache line size register. However, some drivers write to this register upon initialization. This can cause MRMs to be used on these chips, which may cause deadlocks on PLB4. The workaround implemented here introduces a new indirect_type flag, called PPC_INDIRECT_TYPE_BROKEN_MRM. This is set in the pci_controller structure in the pci fixup function for 4xx PCI bridges by determining if the bridge is compatible with 440EPx/GRx. The flag is checked in the indirect_write_config function, and forces any writes to the PCI_CACHE_LINE_SIZE register to be zero, which will disable MRMs for these chips. A similar workaround has been tested by AMCC on various PCI cards, such as the Silicon Image ATA card and Intel E1000 GIGE card. Hangs were seen with the Silicon Image card, and MRMs were seen on the bus with a PCI analyzer. With the workaround in place, the card functioned properly and only Memory Reads were seen on the bus with the analyzer. Acked-by: Stefan Roese <sr@denx.de> Signed-off-by: Josh Boyer <jwboyer@linux.vnet.ibm.com>
173 lines
4.4 KiB
C
173 lines
4.4 KiB
C
/*
|
|
* Support for indirect PCI bridges.
|
|
*
|
|
* Copyright (C) 1998 Gabriel Paubert.
|
|
*
|
|
* 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.
|
|
*/
|
|
|
|
#include <linux/kernel.h>
|
|
#include <linux/pci.h>
|
|
#include <linux/delay.h>
|
|
#include <linux/string.h>
|
|
#include <linux/init.h>
|
|
|
|
#include <asm/io.h>
|
|
#include <asm/prom.h>
|
|
#include <asm/pci-bridge.h>
|
|
#include <asm/machdep.h>
|
|
|
|
static int
|
|
indirect_read_config(struct pci_bus *bus, unsigned int devfn, int offset,
|
|
int len, u32 *val)
|
|
{
|
|
struct pci_controller *hose = bus->sysdata;
|
|
volatile void __iomem *cfg_data;
|
|
u8 cfg_type = 0;
|
|
u32 bus_no, reg;
|
|
|
|
if (hose->indirect_type & PPC_INDIRECT_TYPE_NO_PCIE_LINK) {
|
|
if (bus->number != hose->first_busno)
|
|
return PCIBIOS_DEVICE_NOT_FOUND;
|
|
if (devfn != 0)
|
|
return PCIBIOS_DEVICE_NOT_FOUND;
|
|
}
|
|
|
|
if (ppc_md.pci_exclude_device)
|
|
if (ppc_md.pci_exclude_device(hose, bus->number, devfn))
|
|
return PCIBIOS_DEVICE_NOT_FOUND;
|
|
|
|
if (hose->indirect_type & PPC_INDIRECT_TYPE_SET_CFG_TYPE)
|
|
if (bus->number != hose->first_busno)
|
|
cfg_type = 1;
|
|
|
|
bus_no = (bus->number == hose->first_busno) ?
|
|
hose->self_busno : bus->number;
|
|
|
|
if (hose->indirect_type & PPC_INDIRECT_TYPE_EXT_REG)
|
|
reg = ((offset & 0xf00) << 16) | (offset & 0xfc);
|
|
else
|
|
reg = offset & 0xfc;
|
|
|
|
if (hose->indirect_type & PPC_INDIRECT_TYPE_BIG_ENDIAN)
|
|
out_be32(hose->cfg_addr, (0x80000000 | (bus_no << 16) |
|
|
(devfn << 8) | reg | cfg_type));
|
|
else
|
|
out_le32(hose->cfg_addr, (0x80000000 | (bus_no << 16) |
|
|
(devfn << 8) | reg | cfg_type));
|
|
|
|
/*
|
|
* Note: the caller has already checked that offset is
|
|
* suitably aligned and that len is 1, 2 or 4.
|
|
*/
|
|
cfg_data = hose->cfg_data + (offset & 3);
|
|
switch (len) {
|
|
case 1:
|
|
*val = in_8(cfg_data);
|
|
break;
|
|
case 2:
|
|
*val = in_le16(cfg_data);
|
|
break;
|
|
default:
|
|
*val = in_le32(cfg_data);
|
|
break;
|
|
}
|
|
return PCIBIOS_SUCCESSFUL;
|
|
}
|
|
|
|
static int
|
|
indirect_write_config(struct pci_bus *bus, unsigned int devfn, int offset,
|
|
int len, u32 val)
|
|
{
|
|
struct pci_controller *hose = bus->sysdata;
|
|
volatile void __iomem *cfg_data;
|
|
u8 cfg_type = 0;
|
|
u32 bus_no, reg;
|
|
|
|
if (hose->indirect_type & PPC_INDIRECT_TYPE_NO_PCIE_LINK) {
|
|
if (bus->number != hose->first_busno)
|
|
return PCIBIOS_DEVICE_NOT_FOUND;
|
|
if (devfn != 0)
|
|
return PCIBIOS_DEVICE_NOT_FOUND;
|
|
}
|
|
|
|
if (ppc_md.pci_exclude_device)
|
|
if (ppc_md.pci_exclude_device(hose, bus->number, devfn))
|
|
return PCIBIOS_DEVICE_NOT_FOUND;
|
|
|
|
if (hose->indirect_type & PPC_INDIRECT_TYPE_SET_CFG_TYPE)
|
|
if (bus->number != hose->first_busno)
|
|
cfg_type = 1;
|
|
|
|
bus_no = (bus->number == hose->first_busno) ?
|
|
hose->self_busno : bus->number;
|
|
|
|
if (hose->indirect_type & PPC_INDIRECT_TYPE_EXT_REG)
|
|
reg = ((offset & 0xf00) << 16) | (offset & 0xfc);
|
|
else
|
|
reg = offset & 0xfc;
|
|
|
|
if (hose->indirect_type & PPC_INDIRECT_TYPE_BIG_ENDIAN)
|
|
out_be32(hose->cfg_addr, (0x80000000 | (bus_no << 16) |
|
|
(devfn << 8) | reg | cfg_type));
|
|
else
|
|
out_le32(hose->cfg_addr, (0x80000000 | (bus_no << 16) |
|
|
(devfn << 8) | reg | cfg_type));
|
|
|
|
/* surpress setting of PCI_PRIMARY_BUS */
|
|
if (hose->indirect_type & PPC_INDIRECT_TYPE_SURPRESS_PRIMARY_BUS)
|
|
if ((offset == PCI_PRIMARY_BUS) &&
|
|
(bus->number == hose->first_busno))
|
|
val &= 0xffffff00;
|
|
|
|
/* Workaround for PCI_28 Errata in 440EPx/GRx */
|
|
if ((hose->indirect_type & PPC_INDIRECT_TYPE_BROKEN_MRM) &&
|
|
offset == PCI_CACHE_LINE_SIZE) {
|
|
val = 0;
|
|
}
|
|
|
|
/*
|
|
* Note: the caller has already checked that offset is
|
|
* suitably aligned and that len is 1, 2 or 4.
|
|
*/
|
|
cfg_data = hose->cfg_data + (offset & 3);
|
|
switch (len) {
|
|
case 1:
|
|
out_8(cfg_data, val);
|
|
break;
|
|
case 2:
|
|
out_le16(cfg_data, val);
|
|
break;
|
|
default:
|
|
out_le32(cfg_data, val);
|
|
break;
|
|
}
|
|
return PCIBIOS_SUCCESSFUL;
|
|
}
|
|
|
|
static struct pci_ops indirect_pci_ops =
|
|
{
|
|
.read = indirect_read_config,
|
|
.write = indirect_write_config,
|
|
};
|
|
|
|
void __init
|
|
setup_indirect_pci(struct pci_controller* hose,
|
|
resource_size_t cfg_addr,
|
|
resource_size_t cfg_data, u32 flags)
|
|
{
|
|
resource_size_t base = cfg_addr & PAGE_MASK;
|
|
void __iomem *mbase;
|
|
|
|
mbase = ioremap(base, PAGE_SIZE);
|
|
hose->cfg_addr = mbase + (cfg_addr & ~PAGE_MASK);
|
|
if ((cfg_data & PAGE_MASK) != base)
|
|
mbase = ioremap(cfg_data & PAGE_MASK, PAGE_SIZE);
|
|
hose->cfg_data = mbase + (cfg_data & ~PAGE_MASK);
|
|
hose->ops = &indirect_pci_ops;
|
|
hose->indirect_type = flags;
|
|
}
|