2 * This file is subject to the terms and conditions of the GNU General Public
3 * License. See the file "COPYING" in the main directory of this archive
6 * Copyright (C) 2004 by Ralf Baechle (ralf@linux-mips.org)
9 #include <linux/types.h>
10 #include <linux/pci.h>
11 #include <asm/mv64340.h>
13 #include <linux/init.h>
16 * We assume the address ranges have already been setup appropriately by
17 * the firmware. PMON in case of the Ocelot C does that.
19 static struct resource mv_pci_io_mem0_resource
= {
20 .name
= "MV64340 PCI0 IO MEM",
21 .flags
= IORESOURCE_IO
24 static struct resource mv_pci_mem0_resource
= {
25 .name
= "MV64340 PCI0 MEM",
26 .flags
= IORESOURCE_MEM
29 static struct mv_pci_controller mv_bus0_controller
= {
31 .pci_ops
= &mv_pci_ops
,
32 .mem_resource
= &mv_pci_mem0_resource
,
33 .io_resource
= &mv_pci_io_mem0_resource
,
35 .config_addr
= MV64340_PCI_0_CONFIG_ADDR
,
36 .config_vreg
= MV64340_PCI_0_CONFIG_DATA_VIRTUAL_REG
,
39 static uint32_t mv_io_base
, mv_io_size
;
41 static void mv64340_pci0_init(void)
43 uint32_t mem0_base
, mem0_size
;
44 uint32_t io_base
, io_size
;
46 io_base
= MV_READ(MV64340_PCI_0_IO_BASE_ADDR
) << 16;
47 io_size
= (MV_READ(MV64340_PCI_0_IO_SIZE
) + 1) << 16;
48 mem0_base
= MV_READ(MV64340_PCI_0_MEMORY0_BASE_ADDR
) << 16;
49 mem0_size
= (MV_READ(MV64340_PCI_0_MEMORY0_SIZE
) + 1) << 16;
51 mv_pci_io_mem0_resource
.start
= 0;
52 mv_pci_io_mem0_resource
.end
= io_size
- 1;
53 mv_pci_mem0_resource
.start
= mem0_base
;
54 mv_pci_mem0_resource
.end
= mem0_base
+ mem0_size
- 1;
55 mv_bus0_controller
.pcic
.mem_offset
= mem0_base
;
56 mv_bus0_controller
.pcic
.io_offset
= 0;
58 ioport_resource
.end
= io_size
- 1;
60 register_pci_controller(&mv_bus0_controller
.pcic
);
66 static struct resource mv_pci_io_mem1_resource
= {
67 .name
= "MV64340 PCI1 IO MEM",
68 .flags
= IORESOURCE_IO
71 static struct resource mv_pci_mem1_resource
= {
72 .name
= "MV64340 PCI1 MEM",
73 .flags
= IORESOURCE_MEM
76 static struct mv_pci_controller mv_bus1_controller
= {
78 .pci_ops
= &mv_pci_ops
,
79 .mem_resource
= &mv_pci_mem1_resource
,
80 .io_resource
= &mv_pci_io_mem1_resource
,
82 .config_addr
= MV64340_PCI_1_CONFIG_ADDR
,
83 .config_vreg
= MV64340_PCI_1_CONFIG_DATA_VIRTUAL_REG
,
86 static __init
void mv64340_pci1_init(void)
88 uint32_t mem0_base
, mem0_size
;
89 uint32_t io_base
, io_size
;
91 io_base
= MV_READ(MV64340_PCI_1_IO_BASE_ADDR
) << 16;
92 io_size
= (MV_READ(MV64340_PCI_1_IO_SIZE
) + 1) << 16;
93 mem0_base
= MV_READ(MV64340_PCI_1_MEMORY0_BASE_ADDR
) << 16;
94 mem0_size
= (MV_READ(MV64340_PCI_1_MEMORY0_SIZE
) + 1) << 16;
97 * Here we assume the I/O window of second bus to be contiguous with
98 * the first. A gap is no problem but would waste address space for
99 * remapping the port space.
101 mv_pci_io_mem1_resource
.start
= mv_io_size
;
102 mv_pci_io_mem1_resource
.end
= mv_io_size
+ io_size
- 1;
103 mv_pci_mem1_resource
.start
= mem0_base
;
104 mv_pci_mem1_resource
.end
= mem0_base
+ mem0_size
- 1;
105 mv_bus1_controller
.pcic
.mem_offset
= mem0_base
;
106 mv_bus1_controller
.pcic
.io_offset
= 0;
108 ioport_resource
.end
= io_base
+ io_size
-mv_io_base
- 1;
110 register_pci_controller(&mv_bus1_controller
.pcic
);
112 mv_io_size
= io_base
+ io_size
- mv_io_base
;
115 static __init
int __init
ocelot_c_pci_init(void)
117 unsigned long io_v_base
;
120 enable
= ~MV_READ(MV64340_BASE_ADDR_ENABLE
);
123 * We require at least one enabled I/O or PCI memory window or we
124 * will ignore this PCI bus. We ignore PCI windows 1, 2 and 3.
126 if (enable
& (0x01 << 9) || enable
& (0x01 << 10))
129 if (enable
& (0x01 << 14) || enable
& (0x01 << 15))
133 io_v_base
= (unsigned long) ioremap(mv_io_base
, mv_io_size
);
135 panic("Could not ioremap I/O port range");
137 set_io_port_base(io_v_base
);
143 arch_initcall(ocelot_c_pci_init
);