Commit 1f824d5c authored by Bahadir Balban's avatar Bahadir Balban Committed by Catalin Marinas

RealView/EB PCI Support (MEM only)

Signed-off-by: default avatarBahadir Balban <bahadir.balban@arm.com>
Signed-off-by: default avatarCatalin Marinas <catalin.marinas@arm.com>
parent 452624c2
......@@ -420,7 +420,7 @@ config ISA_DMA_API
bool
config PCI
bool "PCI support" if ARCH_INTEGRATOR_AP || ARCH_VERSATILE_PB || ARCH_IXP4XX
bool "PCI support" if ARCH_INTEGRATOR_AP || ARCH_VERSATILE_PB || ARCH_IXP4XX || MACH_REALVIEW_EB
help
Find out whether you have a PCI motherboard. PCI is the name of a
bus system, i.e. the way the CPU talks to the other stuff inside
......
......@@ -7,3 +7,4 @@ obj-$(CONFIG_MACH_REALVIEW_EB) += realview_eb.o
obj-$(CONFIG_SMP) += platsmp.o headsmp.o
obj-$(CONFIG_HOTPLUG_CPU) += hotplug.o
obj-$(CONFIG_LOCAL_TIMERS) += localtimer.o
obj-$(CONFIG_PCI) += pci.o
/*
* linux/arch/arm/mach-realview/pci.c
*
* (C) Copyright Koninklijke Philips Electronics NV 2004. All rights reserved.
* You can redistribute and/or modify this software under the terms of version 2
* of the GNU General Public License as published by the Free Software Foundation.
* THIS SOFTWARE IS PROVIDED "AS IS" WITHOUT ANY WARRANTY; WITHOUT EVEN THE IMPLIED
* WARRANTY OF MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* General Public License for more details.
* Koninklijke Philips Electronics nor its subsidiaries is obligated to provide any support for this software.
*
* ARM Versatile PCI driver.
*
* 14/04/2005 Initial version, colin.king@philips.com
*
*/
#include <linux/kernel.h>
#include <linux/pci.h>
#include <linux/ptrace.h>
#include <linux/slab.h>
#include <linux/ioport.h>
#include <linux/interrupt.h>
#include <linux/spinlock.h>
#include <linux/init.h>
#include <asm/hardware.h>
#include <asm/io.h>
#include <asm/irq.h>
#include <asm/system.h>
#include <asm/mach/pci.h>
/*
* these spaces are mapped using the following base registers:
*
* Usage Local Bus Memory Base/Map registers used
*
* Mem 50000000 - 5FFFFFFF LB_BASE0/LB_MAP0, non prefetch
* Mem 60000000 - 6FFFFFFF LB_BASE1/LB_MAP1, prefetch
* IO 44000000 - 4FFFFFFF LB_BASE2/LB_MAP2, IO
* Cfg 42000000 - 42FFFFFF PCI config
*
*/
#define SYS_PCICTL IO_ADDRESS(REALVIEW_SYS_PCICTL)
#define PCI_IMAP0 IO_ADDRESS(REALVIEW_EB_PCI_CORE_BASE+0x0)
#define PCI_IMAP1 IO_ADDRESS(REALVIEW_EB_PCI_CORE_BASE+0x4)
#define PCI_IMAP2 IO_ADDRESS(REALVIEW_EB_PCI_CORE_BASE+0x8)
#define PCI_SELFID IO_ADDRESS(REALVIEW_EB_PCI_CORE_BASE+0xc)
#define PCI_FLAGS IO_ADDRESS(REALVIEW_EB_PCI_CORE_BASE+0x10)
#define PCI_SMAP0 IO_ADDRESS(REALVIEW_EB_PCI_CORE_BASE+0x14)
#define PCI_SMAP1 IO_ADDRESS(REALVIEW_EB_PCI_CORE_BASE+0x18)
#define PCI_SMAP2 IO_ADDRESS(REALVIEW_EB_PCI_CORE_BASE+0x1C)
#define DEVICE_ID_OFFSET 0x00
#define CSR_OFFSET 0x04
#define CLASS_ID_OFFSET 0x08
#define VP_PCI_DEVICE_ID 0x030010ee
#define VP_PCI_CLASS_ID 0x0b400000
static unsigned long pci_slot_ignore = 0;
static int __init realview_eb_pci_slot_ignore(char *str)
{
int retval;
int slot;
while ((retval = get_option(&str,&slot))) {
if ((slot < 0) || (slot > 31)) {
printk("Illegal slot value: %d\n",slot);
} else {
pci_slot_ignore |= (1 << slot);
}
}
return 1;
}
__setup("pci_slot_ignore=", realview_eb_pci_slot_ignore);
static unsigned long __pci_addr(struct pci_bus *bus,
unsigned int devfn, int offset)
{
unsigned int busnr = bus->number;
/*
* Trap out illegal values
*/
if (offset > 255)
BUG();
if (busnr > 255)
BUG();
if (devfn > 255)
BUG();
return (REALVIEW_EB_PCI_CFG_VIRT_BASE | (busnr << 16) |
(PCI_SLOT(devfn) << 11) | (PCI_FUNC(devfn) << 8) | offset);
}
static int realview_eb_read_config(struct pci_bus *bus, unsigned int devfn, int where,
int size, u32 *val)
{
unsigned long addr = __pci_addr(bus, devfn, where);
u32 v;
int slot = PCI_SLOT(devfn);
if (pci_slot_ignore & (1 << slot)) {
/* Ignore this slot */
switch (size) {
case 1:
v = 0xff;
break;
case 2:
v = 0xffff;
break;
default:
v = 0xffffffff;
}
} else {
switch (size) {
case 1:
addr &= ~3;
v = __raw_readb(addr);
break;
case 2:
v = __raw_readl(addr & ~3);
if (addr & 2) v >>= 16;
v &= 0xffff;
break;
default:
addr &= ~3;
v = __raw_readl(addr);
break;
}
}
*val = v;
return PCIBIOS_SUCCESSFUL;
}
static int realview_eb_write_config(struct pci_bus *bus, unsigned int devfn, int where,
int size, u32 val)
{
unsigned long addr = __pci_addr(bus, devfn, where);
int slot = PCI_SLOT(devfn);
if (pci_slot_ignore & (1 << slot)) {
return PCIBIOS_SUCCESSFUL;
}
switch (size) {
case 1:
__raw_writeb((u8)val, addr);
break;
case 2:
__raw_writew((u16)val, addr);
break;
case 4:
__raw_writel(val, addr);
break;
}
return PCIBIOS_SUCCESSFUL;
}
static struct pci_ops pci_realview_eb_ops = {
.read = realview_eb_read_config,
.write = realview_eb_write_config,
};
static struct resource io_mem = {
.name = "PCI non-prefetchable I/O space",
.start = REALVIEW_EB_PCI_IO_BASE0,
.end = REALVIEW_EB_PCI_IO_BASE0+REALVIEW_EB_PCI_IO_BASE0_SIZE-1,
.flags = IORESOURCE_IO,
};
static struct resource nonpre_mem0 = {
.name = "PCI memory region 0",
.start = REALVIEW_EB_PCI_MEM_BASE0,
.end = REALVIEW_EB_PCI_MEM_BASE0+REALVIEW_EB_PCI_MEM_BASE0_SIZE-1,
.flags = IORESOURCE_MEM,
};
static struct resource pre_mem1 = {
.name = "PCI memory region 1",
.start = REALVIEW_EB_PCI_MEM_BASE1,
.end = REALVIEW_EB_PCI_MEM_BASE1+REALVIEW_EB_PCI_MEM_BASE1_SIZE-1,
.flags = IORESOURCE_MEM | IORESOURCE_PREFETCH,
};
/* Available but unused yet. To use, it must replace another region
* because only 3 available resource slots in pci_sys_data.
static struct resource pre_mem2 = {
.name = "PCI memory region 2",
.start = REALVIEW_PCI_MEM_BASE2,
.end = REALVIEW_PCI_MEM_BASE2+REALVIEW_PCI_MEM_BASE2_SIZE-1,
.flags = IORESOURCE_MEM | IORESOURCE_PREFETCH,
};
*/
static int __init pci_realview_eb_setup_resources(struct resource **resource)
{
int ret = 0;
ret = request_resource(&iomem_resource, &io_mem);
if (ret) {
printk(KERN_ERR "PCI: unable to allocate I/O "
"region 0(%d)\n", ret);
goto out;
}
ret = request_resource(&iomem_resource, &nonpre_mem0);
if (ret) {
printk(KERN_ERR "PCI: unable to allocate non-prefetcheble "
"memory region 0(%d)\n", ret);
goto release_io_mem;
}
ret = request_resource(&iomem_resource, &pre_mem1);
if (ret) {
printk(KERN_ERR "PCI: unable to allocate prefetchable "
"memory region 1(%d)\n", ret);
goto release_nonpre_mem0;
}
/*
* bus->resource[0] is the IO resource for this bus
* bus->resource[1] is the non-prefetchable mem resource 0 for this bus
* bus->resource[2] is the prefetchable mem resource 1 for this bus
*/
resource[0] = &io_mem;
resource[1] = &nonpre_mem0;
resource[2] = &pre_mem1;
/* NOTE: There are only 3 resource slots in struct pci_sys_data.
* Assigning a 4th resource would overwrite swizzle field and
* cause an oops. */
goto out;
release_nonpre_mem0:
release_resource(&nonpre_mem0);
release_io_mem:
release_resource(&io_mem);
out:
return ret;
}
int __init pci_realview_eb_setup(int nr, struct pci_sys_data *sys)
{
int ret = 0;
int i;
int myslot = -1;
unsigned long val;
void __iomem *local_pci_cfg_base;
val = __raw_readl(SYS_PCICTL);
if (!(val & 1)) {
printk("Not plugged into PCI backplane!\n");
ret = -EIO;
goto out;
}
if (nr == 0) {
sys->mem_offset = 0;
ret = pci_realview_eb_setup_resources(sys->resource);
if (ret < 0) {
printk("pci_versatile_setup: resources... oops?\n");
goto out;
}
} else {
printk("pci_versatile_setup: resources... nr == 0??\n");
goto out;
}
/*
* We need to discover the PCI core first to configure itself
* before the main PCI probing is performed
*/
for (i=0; i<32; i++)
if ((__raw_readl(REALVIEW_EB_PCI_VIRT_BASE+(i<<11)+DEVICE_ID_OFFSET) == VP_PCI_DEVICE_ID) &&
(__raw_readl(REALVIEW_EB_PCI_VIRT_BASE+(i<<11)+CLASS_ID_OFFSET) == VP_PCI_CLASS_ID)) {
myslot = i;
break;
}
if (myslot == -1) {
printk("Cannot find PCI core!\n");
ret = -EIO;
goto out;
}
printk("PCI core found (slot %d)\n",myslot);
__raw_writel(myslot, PCI_SELFID);
local_pci_cfg_base = (void *) REALVIEW_EB_PCI_CFG_VIRT_BASE + (myslot << 11);
val = __raw_readl(local_pci_cfg_base + CSR_OFFSET);
val |= PCI_COMMAND_MEMORY | PCI_COMMAND_MASTER | PCI_COMMAND_INVALIDATE;
__raw_writel(val, local_pci_cfg_base + CSR_OFFSET);
/*
* Configure the PCI inbound memory windows to be 1:1 mapped to SDRAM
*/
__raw_writel(PHYS_OFFSET, local_pci_cfg_base + PCI_BASE_ADDRESS_0);
__raw_writel(PHYS_OFFSET, local_pci_cfg_base + PCI_BASE_ADDRESS_1);
__raw_writel(PHYS_OFFSET, local_pci_cfg_base + PCI_BASE_ADDRESS_2);
/*
* Do not to map Versatile FPGA PCI device into memory space
*/
pci_slot_ignore |= (1 << myslot);
ret = 1;
out:
return ret;
}
struct pci_bus *pci_realview_eb_scan_bus(int nr, struct pci_sys_data *sys)
{
return pci_scan_bus(sys->busnr, &pci_realview_eb_ops, sys);
}
void __init pci_realview_eb_preinit(void)
{
__raw_writel(REALVIEW_EB_PCI_MEM_BASE0, PCI_IMAP0);
__raw_writel(REALVIEW_EB_PCI_MEM_BASE1, PCI_IMAP1);
__raw_writel(REALVIEW_EB_PCI_MEM_BASE2, PCI_IMAP2);
__raw_writel(PHYS_OFFSET, PCI_SMAP0);
__raw_writel(PHYS_OFFSET, PCI_SMAP1);
__raw_writel(PHYS_OFFSET, PCI_SMAP2);
__raw_writel(1, SYS_PCICTL);
}
/*
* Map the specified device/slot/pin to an IRQ.
* Different backplanes may need to modify this.
*/
static int __init realview_eb_map_irq(struct pci_dev *dev, u8 slot, u8 pin)
{
int irq;
int devslot = PCI_SLOT(dev->devfn);
/* slot, pin, irq
* 24 1 80
* 25 1 81
* 26 1 82
* 27 1 83
*/
irq = (((devslot - 29) + pin - 1) & 3) + 80;
printk("PCI map irq: slot %d, pin %d, devslot %d, irq: %d\n",
slot, pin, devslot, irq);
return irq;
}
static struct hw_pci realview_eb_pci __initdata = {
.swizzle = NULL,
.map_irq = realview_eb_map_irq,
.nr_controllers = 1,
.setup = pci_realview_eb_setup,
.scan = pci_realview_eb_scan_bus,
.preinit = pci_realview_eb_preinit,
};
static int __init realview_eb_pci_init(void)
{
pci_common_init(&realview_eb_pci);
return 0;
}
subsys_initcall(realview_eb_pci_init);
......@@ -99,7 +99,25 @@ static struct map_desc realview_eb_io_desc[] __initdata = {
.pfn = __phys_to_pfn(REALVIEW_UART0_BASE),
.length = SZ_4K,
.type = MT_DEVICE,
}
},
#endif
#ifdef CONFIG_PCI
{
.virtual = IO_ADDRESS(REALVIEW_EB_PCI_CORE_BASE),
.pfn = __phys_to_pfn(REALVIEW_EB_PCI_CORE_BASE),
.length = SZ_4K,
.type = MT_DEVICE
}, {
.virtual = REALVIEW_EB_PCI_VIRT_BASE,
.pfn = __phys_to_pfn(REALVIEW_EB_PCI_BASE),
.length = REALVIEW_EB_PCI_BASE_SIZE,
.type = MT_DEVICE
}, {
.virtual = REALVIEW_EB_PCI_CFG_VIRT_BASE,
.pfn = __phys_to_pfn(REALVIEW_EB_PCI_CFG_BASE),
.length = REALVIEW_EB_PCI_CFG_BASE_SIZE,
.type = MT_DEVICE
},
#endif
};
......
......@@ -26,6 +26,19 @@
#include <asm/arch/platform.h>
#ifdef CONFIG_MMU
/*
* PCI space virtual addresses
*/
#define REALVIEW_EB_PCI_VIRT_BASE 0xe8000000
#define REALVIEW_EB_PCI_CFG_VIRT_BASE 0xe9000000
/* CIK guesswork */
#define PCIBIOS_MIN_IO 0x62000000
#define PCIBIOS_MIN_MEM 0x63000000
#define pcibios_assign_all_busses() 1
/* macro to get at IO space when running virtually */
#define IO_ADDRESS(x) ((((x) & 0x0effffff) | (((x) >> 4) & 0x0f000000)) + 0xf0000000)
#else
......
......@@ -199,7 +199,6 @@
/* Reserved 0x10016000 */
#define REALVIEW_RTC_BASE 0x10017000 /* Real Time Clock */
#define REALVIEW_DMC_BASE 0x10018000 /* DMC configuration */
#define REALVIEW_PCI_CORE_BASE 0x10019000 /* PCI configuration */
/* Reserved 0x1001A000 - 0x1001FFFF */
#define REALVIEW_CLCD_BASE 0x10020000 /* CLCD */
#define REALVIEW_DMAC_BASE 0x10030000 /* DMA controller */
......@@ -231,19 +230,23 @@
/* Reserved 0x10090000 - 0x100EFFFF */
#define REALVIEW_ETH_BASE 0x4E000000 /* Ethernet */
/* PCI space */
#define REALVIEW_PCI_BASE 0x41000000 /* PCI Interface */
#define REALVIEW_PCI_CFG_BASE 0x42000000
#define REALVIEW_PCI_MEM_BASE0 0x44000000
#define REALVIEW_PCI_MEM_BASE1 0x50000000
#define REALVIEW_PCI_MEM_BASE2 0x60000000
/* Sizes of above maps */
#define REALVIEW_PCI_BASE_SIZE 0x01000000
#define REALVIEW_PCI_CFG_BASE_SIZE 0x02000000
#define REALVIEW_PCI_MEM_BASE0_SIZE 0x0c000000 /* 32Mb */
#define REALVIEW_PCI_MEM_BASE1_SIZE 0x10000000 /* 256Mb */
#define REALVIEW_PCI_MEM_BASE2_SIZE 0x10000000 /* 256Mb */
#define REALVIEW_EB_PCI_CORE_BASE 0x10019000 /* PCI configuration */
/* EB PCI regions */
#define REALVIEW_EB_PCI_BASE 0x60000000 /* PCI self config */
#define REALVIEW_EB_PCI_CFG_BASE 0x61000000 /* PCI config */
#define REALVIEW_EB_PCI_IO_BASE0 0x62000000 /* PCI IO region */
#define REALVIEW_EB_PCI_MEM_BASE0 0x63000000 /* Memory region 1 */
#define REALVIEW_EB_PCI_MEM_BASE1 0x64000000 /* Memory region 2 */
#define REALVIEW_EB_PCI_MEM_BASE2 0x68000000 /* Memory region 3 */
/* EB PCI region sizes */
#define REALVIEW_EB_PCI_BASE_SIZE 0x01000000 /* 16MB */
#define REALVIEW_EB_PCI_CFG_BASE_SIZE 0x01000000 /* 16MB */
#define REALVIEW_EB_PCI_IO_BASE0_SIZE 0x01000000 /* 16MB */
#define REALVIEW_EB_PCI_MEM_BASE0_SIZE 0x01000000 /* 16MB */
#define REALVIEW_EB_PCI_MEM_BASE1_SIZE 0x04000000 /* 64MB */
#define REALVIEW_EB_PCI_MEM_BASE2_SIZE 0x08000000 /* 128MB */
#define REALVIEW_SDRAM67_BASE 0x70000000 /* SDRAM banks 6 and 7 */
#define REALVIEW_LT_BASE 0x80000000 /* Logic Tile expansion */
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment