|
@@ -18,29 +18,43 @@
|
|
|
#include <mach/bridge-regs.h>
|
|
|
#include "common.h"
|
|
|
|
|
|
+void __init kirkwood_pcie_id(u32 *dev, u32 *rev)
|
|
|
+{
|
|
|
+ *dev = orion_pcie_dev_id((void __iomem *)PCIE_VIRT_BASE);
|
|
|
+ *rev = orion_pcie_rev((void __iomem *)PCIE_VIRT_BASE);
|
|
|
+}
|
|
|
|
|
|
-#define PCIE_BASE ((void __iomem *)PCIE_VIRT_BASE)
|
|
|
+struct pcie_port {
|
|
|
+ u8 root_bus_nr;
|
|
|
+ void __iomem *base;
|
|
|
+ spinlock_t conf_lock;
|
|
|
+ int irq;
|
|
|
+ struct resource res[2];
|
|
|
+};
|
|
|
|
|
|
-void __init kirkwood_pcie_id(u32 *dev, u32 *rev)
|
|
|
+static int pcie_port_map[2];
|
|
|
+static int num_pcie_ports;
|
|
|
+
|
|
|
+static inline struct pcie_port *bus_to_port(struct pci_bus *bus)
|
|
|
{
|
|
|
- *dev = orion_pcie_dev_id(PCIE_BASE);
|
|
|
- *rev = orion_pcie_rev(PCIE_BASE);
|
|
|
+ struct pci_sys_data *sys = bus->sysdata;
|
|
|
+ return sys->private_data;
|
|
|
}
|
|
|
|
|
|
-static int pcie_valid_config(int bus, int dev)
|
|
|
+static int pcie_valid_config(struct pcie_port *pp, int bus, int dev)
|
|
|
{
|
|
|
/*
|
|
|
* Don't go out when trying to access --
|
|
|
* 1. nonexisting device on local bus
|
|
|
* 2. where there's no device connected (no link)
|
|
|
*/
|
|
|
- if (bus == 0 && dev == 0)
|
|
|
+ if (bus == pp->root_bus_nr && dev == 0)
|
|
|
return 1;
|
|
|
|
|
|
- if (!orion_pcie_link_up(PCIE_BASE))
|
|
|
+ if (!orion_pcie_link_up(pp->base))
|
|
|
return 0;
|
|
|
|
|
|
- if (bus == 0 && dev != 1)
|
|
|
+ if (bus == pp->root_bus_nr && dev != 1)
|
|
|
return 0;
|
|
|
|
|
|
return 1;
|
|
@@ -52,22 +66,22 @@ static int pcie_valid_config(int bus, int dev)
|
|
|
* and then reading the PCIE_CONF_DATA register. Need to make sure these
|
|
|
* transactions are atomic.
|
|
|
*/
|
|
|
-static DEFINE_SPINLOCK(kirkwood_pcie_lock);
|
|
|
|
|
|
static int pcie_rd_conf(struct pci_bus *bus, u32 devfn, int where,
|
|
|
int size, u32 *val)
|
|
|
{
|
|
|
+ struct pcie_port *pp = bus_to_port(bus);
|
|
|
unsigned long flags;
|
|
|
int ret;
|
|
|
|
|
|
- if (pcie_valid_config(bus->number, PCI_SLOT(devfn)) == 0) {
|
|
|
+ if (pcie_valid_config(pp, bus->number, PCI_SLOT(devfn)) == 0) {
|
|
|
*val = 0xffffffff;
|
|
|
return PCIBIOS_DEVICE_NOT_FOUND;
|
|
|
}
|
|
|
|
|
|
- spin_lock_irqsave(&kirkwood_pcie_lock, flags);
|
|
|
- ret = orion_pcie_rd_conf(PCIE_BASE, bus, devfn, where, size, val);
|
|
|
- spin_unlock_irqrestore(&kirkwood_pcie_lock, flags);
|
|
|
+ spin_lock_irqsave(&pp->conf_lock, flags);
|
|
|
+ ret = orion_pcie_rd_conf(pp->base, bus, devfn, where, size, val);
|
|
|
+ spin_unlock_irqrestore(&pp->conf_lock, flags);
|
|
|
|
|
|
return ret;
|
|
|
}
|
|
@@ -75,15 +89,16 @@ static int pcie_rd_conf(struct pci_bus *bus, u32 devfn, int where,
|
|
|
static int pcie_wr_conf(struct pci_bus *bus, u32 devfn,
|
|
|
int where, int size, u32 val)
|
|
|
{
|
|
|
+ struct pcie_port *pp = bus_to_port(bus);
|
|
|
unsigned long flags;
|
|
|
int ret;
|
|
|
|
|
|
- if (pcie_valid_config(bus->number, PCI_SLOT(devfn)) == 0)
|
|
|
+ if (pcie_valid_config(pp, bus->number, PCI_SLOT(devfn)) == 0)
|
|
|
return PCIBIOS_DEVICE_NOT_FOUND;
|
|
|
|
|
|
- spin_lock_irqsave(&kirkwood_pcie_lock, flags);
|
|
|
- ret = orion_pcie_wr_conf(PCIE_BASE, bus, devfn, where, size, val);
|
|
|
- spin_unlock_irqrestore(&kirkwood_pcie_lock, flags);
|
|
|
+ spin_lock_irqsave(&pp->conf_lock, flags);
|
|
|
+ ret = orion_pcie_wr_conf(pp->base, bus, devfn, where, size, val);
|
|
|
+ spin_unlock_irqrestore(&pp->conf_lock, flags);
|
|
|
|
|
|
return ret;
|
|
|
}
|
|
@@ -93,50 +108,112 @@ static struct pci_ops pcie_ops = {
|
|
|
.write = pcie_wr_conf,
|
|
|
};
|
|
|
|
|
|
-
|
|
|
-static int __init kirkwood_pcie_setup(int nr, struct pci_sys_data *sys)
|
|
|
+static int __init pcie0_ioresources_setup(struct pci_sys_data *sys)
|
|
|
{
|
|
|
- struct resource *res;
|
|
|
- extern unsigned int kirkwood_clk_ctrl;
|
|
|
+ struct pcie_port *pp = (struct pcie_port *)sys->private_data;
|
|
|
|
|
|
/*
|
|
|
- * Generic PCIe unit setup.
|
|
|
+ * IORESOURCE_IO
|
|
|
*/
|
|
|
- orion_pcie_setup(PCIE_BASE, &kirkwood_mbus_dram_info);
|
|
|
+ pp->res[0].name = "PCIe 0 I/O Space";
|
|
|
+ pp->res[0].start = KIRKWOOD_PCIE_IO_PHYS_BASE;
|
|
|
+ pp->res[0].end = pp->res[0].start + KIRKWOOD_PCIE_IO_SIZE - 1;
|
|
|
+ pp->res[0].flags = IORESOURCE_IO;
|
|
|
+ if (request_resource(&ioport_resource, &pp->res[0]))
|
|
|
+ panic("Request PCIe 0 IO resource failed\n");
|
|
|
+ sys->resource[0] = &pp->res[0];
|
|
|
|
|
|
/*
|
|
|
- * Request resources.
|
|
|
+ * IORESOURCE_MEM
|
|
|
*/
|
|
|
- res = kzalloc(sizeof(struct resource) * 2, GFP_KERNEL);
|
|
|
- if (!res)
|
|
|
- panic("pcie_setup unable to alloc resources");
|
|
|
+ pp->res[1].name = "PCIe 0 MEM";
|
|
|
+ pp->res[1].start = KIRKWOOD_PCIE_MEM_PHYS_BASE;
|
|
|
+ pp->res[1].end = pp->res[1].start + KIRKWOOD_PCIE_MEM_SIZE - 1;
|
|
|
+ pp->res[1].flags = IORESOURCE_MEM;
|
|
|
+ if (request_resource(&iomem_resource, &pp->res[1]))
|
|
|
+ panic("Request PCIe 0 Memory resource failed\n");
|
|
|
+ sys->resource[1] = &pp->res[1];
|
|
|
+
|
|
|
+ sys->resource[2] = NULL;
|
|
|
+ sys->io_offset = 0;
|
|
|
+
|
|
|
+ return 1;
|
|
|
+}
|
|
|
+
|
|
|
+static int __init pcie1_ioresources_setup(struct pci_sys_data *sys)
|
|
|
+{
|
|
|
+ struct pcie_port *pp = (struct pcie_port *)sys->private_data;
|
|
|
|
|
|
/*
|
|
|
* IORESOURCE_IO
|
|
|
*/
|
|
|
- res[0].name = "PCIe I/O Space";
|
|
|
- res[0].flags = IORESOURCE_IO;
|
|
|
- res[0].start = KIRKWOOD_PCIE_IO_BUS_BASE;
|
|
|
- res[0].end = res[0].start + KIRKWOOD_PCIE_IO_SIZE - 1;
|
|
|
- if (request_resource(&ioport_resource, &res[0]))
|
|
|
- panic("Request PCIe IO resource failed\n");
|
|
|
- sys->resource[0] = &res[0];
|
|
|
+ pp->res[0].name = "PCIe 1 I/O Space";
|
|
|
+ pp->res[0].start = KIRKWOOD_PCIE1_IO_PHYS_BASE;
|
|
|
+ pp->res[0].end = pp->res[0].start + KIRKWOOD_PCIE1_IO_SIZE - 1;
|
|
|
+ pp->res[0].flags = IORESOURCE_IO;
|
|
|
+ if (request_resource(&ioport_resource, &pp->res[0]))
|
|
|
+ panic("Request PCIe 1 IO resource failed\n");
|
|
|
+ sys->resource[0] = &pp->res[0];
|
|
|
|
|
|
/*
|
|
|
* IORESOURCE_MEM
|
|
|
*/
|
|
|
- res[1].name = "PCIe Memory Space";
|
|
|
- res[1].flags = IORESOURCE_MEM;
|
|
|
- res[1].start = KIRKWOOD_PCIE_MEM_BUS_BASE;
|
|
|
- res[1].end = res[1].start + KIRKWOOD_PCIE_MEM_SIZE - 1;
|
|
|
- if (request_resource(&iomem_resource, &res[1]))
|
|
|
- panic("Request PCIe Memory resource failed\n");
|
|
|
- sys->resource[1] = &res[1];
|
|
|
+ pp->res[1].name = "PCIe 1 MEM";
|
|
|
+ pp->res[1].start = KIRKWOOD_PCIE1_MEM_PHYS_BASE;
|
|
|
+ pp->res[1].end = pp->res[1].start + KIRKWOOD_PCIE1_MEM_SIZE - 1;
|
|
|
+ pp->res[1].flags = IORESOURCE_MEM;
|
|
|
+ if (request_resource(&iomem_resource, &pp->res[1]))
|
|
|
+ panic("Request PCIe 1 Memory resource failed\n");
|
|
|
+ sys->resource[1] = &pp->res[1];
|
|
|
|
|
|
sys->resource[2] = NULL;
|
|
|
sys->io_offset = 0;
|
|
|
|
|
|
- kirkwood_clk_ctrl |= CGC_PEX0;
|
|
|
+ return 1;
|
|
|
+}
|
|
|
+
|
|
|
+static int __init kirkwood_pcie_setup(int nr, struct pci_sys_data *sys)
|
|
|
+{
|
|
|
+ extern unsigned int kirkwood_clk_ctrl;
|
|
|
+ struct pcie_port *pp;
|
|
|
+ int index;
|
|
|
+
|
|
|
+ if (nr >= num_pcie_ports)
|
|
|
+ return 0;
|
|
|
+
|
|
|
+ index = pcie_port_map[nr];
|
|
|
+ printk(KERN_INFO "PCI: bus%d uses PCIe port %d\n", sys->busnr, index);
|
|
|
+
|
|
|
+ pp = kzalloc(sizeof(*pp), GFP_KERNEL);
|
|
|
+ if (!pp)
|
|
|
+ panic("PCIe: failed to allocate pcie_port data");
|
|
|
+ sys->private_data = pp;
|
|
|
+ pp->root_bus_nr = sys->busnr;
|
|
|
+ spin_lock_init(&pp->conf_lock);
|
|
|
+
|
|
|
+ switch (index) {
|
|
|
+ case 0:
|
|
|
+ pp->base = (void __iomem *)PCIE_VIRT_BASE;
|
|
|
+ pp->irq = IRQ_KIRKWOOD_PCIE;
|
|
|
+ kirkwood_clk_ctrl |= CGC_PEX0;
|
|
|
+ pcie0_ioresources_setup(sys);
|
|
|
+ break;
|
|
|
+ case 1:
|
|
|
+ pp->base = (void __iomem *)PCIE1_VIRT_BASE;
|
|
|
+ pp->irq = IRQ_KIRKWOOD_PCIE1;
|
|
|
+ kirkwood_clk_ctrl |= CGC_PEX1;
|
|
|
+ pcie1_ioresources_setup(sys);
|
|
|
+ break;
|
|
|
+ default:
|
|
|
+ panic("PCIe setup: invalid controller");
|
|
|
+ }
|
|
|
+
|
|
|
+ /*
|
|
|
+ * Generic PCIe unit setup.
|
|
|
+ */
|
|
|
+ orion_pcie_set_local_bus_nr(pp->base, sys->busnr);
|
|
|
+
|
|
|
+ orion_pcie_setup(pp->base, &kirkwood_mbus_dram_info);
|
|
|
|
|
|
return 1;
|
|
|
}
|
|
@@ -163,7 +240,7 @@ kirkwood_pcie_scan_bus(int nr, struct pci_sys_data *sys)
|
|
|
{
|
|
|
struct pci_bus *bus;
|
|
|
|
|
|
- if (nr == 0) {
|
|
|
+ if (nr < num_pcie_ports) {
|
|
|
bus = pci_scan_bus(sys->busnr, &pcie_ops, sys);
|
|
|
} else {
|
|
|
bus = NULL;
|
|
@@ -175,18 +252,37 @@ kirkwood_pcie_scan_bus(int nr, struct pci_sys_data *sys)
|
|
|
|
|
|
static int __init kirkwood_pcie_map_irq(struct pci_dev *dev, u8 slot, u8 pin)
|
|
|
{
|
|
|
- return IRQ_KIRKWOOD_PCIE;
|
|
|
+ struct pcie_port *pp = bus_to_port(dev->bus);
|
|
|
+
|
|
|
+ return pp->irq;
|
|
|
}
|
|
|
|
|
|
static struct hw_pci kirkwood_pci __initdata = {
|
|
|
- .nr_controllers = 1,
|
|
|
.swizzle = pci_std_swizzle,
|
|
|
.setup = kirkwood_pcie_setup,
|
|
|
.scan = kirkwood_pcie_scan_bus,
|
|
|
.map_irq = kirkwood_pcie_map_irq,
|
|
|
};
|
|
|
|
|
|
-void __init kirkwood_pcie_init(void)
|
|
|
+static void __init add_pcie_port(int index, unsigned long base)
|
|
|
+{
|
|
|
+ printk(KERN_INFO "Kirkwood PCIe port %d: ", index);
|
|
|
+
|
|
|
+ if (orion_pcie_link_up((void __iomem *)base)) {
|
|
|
+ printk(KERN_INFO "link up\n");
|
|
|
+ pcie_port_map[num_pcie_ports++] = index;
|
|
|
+ } else
|
|
|
+ printk(KERN_INFO "link down, ignoring\n");
|
|
|
+}
|
|
|
+
|
|
|
+void __init kirkwood_pcie_init(unsigned int portmask)
|
|
|
{
|
|
|
+ if (portmask & KW_PCIE0)
|
|
|
+ add_pcie_port(0, PCIE_VIRT_BASE);
|
|
|
+
|
|
|
+ if (portmask & KW_PCIE1)
|
|
|
+ add_pcie_port(1, PCIE1_VIRT_BASE);
|
|
|
+
|
|
|
+ kirkwood_pci.nr_controllers = num_pcie_ports;
|
|
|
pci_common_init(&kirkwood_pci);
|
|
|
}
|