diff options
24 files changed, 1856 insertions, 2 deletions
diff --git a/arch/arm/Kconfig b/arch/arm/Kconfig index 3535cae73d24..5bf60d5897d3 100644 --- a/arch/arm/Kconfig +++ b/arch/arm/Kconfig | |||
@@ -388,6 +388,16 @@ config ARCH_LOKI | |||
388 | help | 388 | help |
389 | Support for the Marvell Loki (88RC8480) SoC. | 389 | Support for the Marvell Loki (88RC8480) SoC. |
390 | 390 | ||
391 | config ARCH_MV78XX0 | ||
392 | bool "Marvell MV78xx0" | ||
393 | select PCI | ||
394 | select GENERIC_TIME | ||
395 | select GENERIC_CLOCKEVENTS | ||
396 | select PLAT_ORION | ||
397 | help | ||
398 | Support for the following Marvell MV78xx0 series SoCs: | ||
399 | MV781x0, MV782x0. | ||
400 | |||
391 | config ARCH_MXC | 401 | config ARCH_MXC |
392 | bool "Freescale MXC/iMX-based" | 402 | bool "Freescale MXC/iMX-based" |
393 | select ARCH_MTD_XIP | 403 | select ARCH_MTD_XIP |
@@ -528,6 +538,8 @@ source "arch/arm/mach-ixp23xx/Kconfig" | |||
528 | 538 | ||
529 | source "arch/arm/mach-loki/Kconfig" | 539 | source "arch/arm/mach-loki/Kconfig" |
530 | 540 | ||
541 | source "arch/arm/mach-mv78xx0/Kconfig" | ||
542 | |||
531 | source "arch/arm/mach-pxa/Kconfig" | 543 | source "arch/arm/mach-pxa/Kconfig" |
532 | 544 | ||
533 | source "arch/arm/mach-sa1100/Kconfig" | 545 | source "arch/arm/mach-sa1100/Kconfig" |
diff --git a/arch/arm/Makefile b/arch/arm/Makefile index 1459b3849c8c..b53237bb6f13 100644 --- a/arch/arm/Makefile +++ b/arch/arm/Makefile | |||
@@ -142,6 +142,7 @@ endif | |||
142 | machine-$(CONFIG_ARCH_ORION5X) := orion5x | 142 | machine-$(CONFIG_ARCH_ORION5X) := orion5x |
143 | machine-$(CONFIG_ARCH_MSM7X00A) := msm | 143 | machine-$(CONFIG_ARCH_MSM7X00A) := msm |
144 | machine-$(CONFIG_ARCH_LOKI) := loki | 144 | machine-$(CONFIG_ARCH_LOKI) := loki |
145 | machine-$(CONFIG_ARCH_MV78XX0) := mv78xx0 | ||
145 | 146 | ||
146 | ifeq ($(CONFIG_ARCH_EBSA110),y) | 147 | ifeq ($(CONFIG_ARCH_EBSA110),y) |
147 | # This is what happens if you forget the IOCS16 line. | 148 | # This is what happens if you forget the IOCS16 line. |
diff --git a/arch/arm/mach-mv78xx0/Kconfig b/arch/arm/mach-mv78xx0/Kconfig new file mode 100644 index 000000000000..d83cb86837db --- /dev/null +++ b/arch/arm/mach-mv78xx0/Kconfig | |||
@@ -0,0 +1,13 @@ | |||
1 | if ARCH_MV78XX0 | ||
2 | |||
3 | menu "Marvell MV78xx0 Implementations" | ||
4 | |||
5 | config MACH_DB78X00_BP | ||
6 | bool "Marvell DB-78x00-BP Development Board" | ||
7 | help | ||
8 | Say 'Y' here if you want your kernel to support the | ||
9 | Marvell DB-78x00-BP Development Board. | ||
10 | |||
11 | endmenu | ||
12 | |||
13 | endif | ||
diff --git a/arch/arm/mach-mv78xx0/Makefile b/arch/arm/mach-mv78xx0/Makefile new file mode 100644 index 000000000000..ec16c05c3b1b --- /dev/null +++ b/arch/arm/mach-mv78xx0/Makefile | |||
@@ -0,0 +1,2 @@ | |||
1 | obj-y += common.o addr-map.o irq.o pcie.o | ||
2 | obj-$(CONFIG_MACH_DB78X00_BP) += db78x00-bp-setup.o | ||
diff --git a/arch/arm/mach-mv78xx0/Makefile.boot b/arch/arm/mach-mv78xx0/Makefile.boot new file mode 100644 index 000000000000..67039c3e0c48 --- /dev/null +++ b/arch/arm/mach-mv78xx0/Makefile.boot | |||
@@ -0,0 +1,3 @@ | |||
1 | zreladdr-y := 0x00008000 | ||
2 | params_phys-y := 0x00000100 | ||
3 | initrd_phys-y := 0x00800000 | ||
diff --git a/arch/arm/mach-mv78xx0/addr-map.c b/arch/arm/mach-mv78xx0/addr-map.c new file mode 100644 index 000000000000..4004b672a2eb --- /dev/null +++ b/arch/arm/mach-mv78xx0/addr-map.c | |||
@@ -0,0 +1,156 @@ | |||
1 | /* | ||
2 | * arch/arm/mach-mv78xx0/addr-map.c | ||
3 | * | ||
4 | * Address map functions for Marvell MV78xx0 SoCs | ||
5 | * | ||
6 | * This file is licensed under the terms of the GNU General Public | ||
7 | * License version 2. This program is licensed "as is" without any | ||
8 | * warranty of any kind, whether express or implied. | ||
9 | */ | ||
10 | |||
11 | #include <linux/kernel.h> | ||
12 | #include <linux/init.h> | ||
13 | #include <linux/mbus.h> | ||
14 | #include <asm/io.h> | ||
15 | #include "common.h" | ||
16 | |||
17 | /* | ||
18 | * Generic Address Decode Windows bit settings | ||
19 | */ | ||
20 | #define TARGET_DDR 0 | ||
21 | #define TARGET_DEV_BUS 1 | ||
22 | #define TARGET_PCIE0 4 | ||
23 | #define TARGET_PCIE1 8 | ||
24 | #define TARGET_PCIE(i) ((i) ? TARGET_PCIE1 : TARGET_PCIE0) | ||
25 | #define ATTR_DEV_SPI_ROM 0x1f | ||
26 | #define ATTR_DEV_BOOT 0x2f | ||
27 | #define ATTR_DEV_CS3 0x37 | ||
28 | #define ATTR_DEV_CS2 0x3b | ||
29 | #define ATTR_DEV_CS1 0x3d | ||
30 | #define ATTR_DEV_CS0 0x3e | ||
31 | #define ATTR_PCIE_IO(l) (0xf0 & ~(0x10 << (l))) | ||
32 | #define ATTR_PCIE_MEM(l) (0xf8 & ~(0x10 << (l))) | ||
33 | |||
34 | /* | ||
35 | * Helpers to get DDR bank info | ||
36 | */ | ||
37 | #define DDR_BASE_CS_OFF(n) (0x0000 + ((n) << 3)) | ||
38 | #define DDR_SIZE_CS_OFF(n) (0x0004 + ((n) << 3)) | ||
39 | |||
40 | /* | ||
41 | * CPU Address Decode Windows registers | ||
42 | */ | ||
43 | #define WIN0_OFF(n) (BRIDGE_VIRT_BASE + 0x0000 + ((n) << 4)) | ||
44 | #define WIN8_OFF(n) (BRIDGE_VIRT_BASE + 0x0900 + (((n) - 8) << 4)) | ||
45 | #define WIN_CTRL_OFF 0x0000 | ||
46 | #define WIN_BASE_OFF 0x0004 | ||
47 | #define WIN_REMAP_LO_OFF 0x0008 | ||
48 | #define WIN_REMAP_HI_OFF 0x000c | ||
49 | |||
50 | |||
51 | struct mbus_dram_target_info mv78xx0_mbus_dram_info; | ||
52 | |||
53 | static void __init __iomem *win_cfg_base(int win) | ||
54 | { | ||
55 | /* | ||
56 | * Find the control register base address for this window. | ||
57 | * | ||
58 | * BRIDGE_VIRT_BASE points to the right (CPU0's or CPU1's) | ||
59 | * MBUS bridge depending on which CPU core we're running on, | ||
60 | * so we don't need to take that into account here. | ||
61 | */ | ||
62 | |||
63 | return (void __iomem *)((win < 8) ? WIN0_OFF(win) : WIN8_OFF(win)); | ||
64 | } | ||
65 | |||
66 | static int __init cpu_win_can_remap(int win) | ||
67 | { | ||
68 | if (win < 8) | ||
69 | return 1; | ||
70 | |||
71 | return 0; | ||
72 | } | ||
73 | |||
74 | static void __init setup_cpu_win(int win, u32 base, u32 size, | ||
75 | u8 target, u8 attr, int remap) | ||
76 | { | ||
77 | void __iomem *addr = win_cfg_base(win); | ||
78 | u32 ctrl; | ||
79 | |||
80 | base &= 0xffff0000; | ||
81 | ctrl = ((size - 1) & 0xffff0000) | (attr << 8) | (target << 4) | 1; | ||
82 | |||
83 | writel(base, addr + WIN_BASE_OFF); | ||
84 | writel(ctrl, addr + WIN_CTRL_OFF); | ||
85 | if (cpu_win_can_remap(win)) { | ||
86 | if (remap < 0) | ||
87 | remap = base; | ||
88 | |||
89 | writel(remap & 0xffff0000, addr + WIN_REMAP_LO_OFF); | ||
90 | writel(0, addr + WIN_REMAP_HI_OFF); | ||
91 | } | ||
92 | } | ||
93 | |||
94 | void __init mv78xx0_setup_cpu_mbus(void) | ||
95 | { | ||
96 | void __iomem *addr; | ||
97 | int i; | ||
98 | int cs; | ||
99 | |||
100 | /* | ||
101 | * First, disable and clear windows. | ||
102 | */ | ||
103 | for (i = 0; i < 14; i++) { | ||
104 | addr = win_cfg_base(i); | ||
105 | |||
106 | writel(0, addr + WIN_BASE_OFF); | ||
107 | writel(0, addr + WIN_CTRL_OFF); | ||
108 | if (cpu_win_can_remap(i)) { | ||
109 | writel(0, addr + WIN_REMAP_LO_OFF); | ||
110 | writel(0, addr + WIN_REMAP_HI_OFF); | ||
111 | } | ||
112 | } | ||
113 | |||
114 | /* | ||
115 | * Setup MBUS dram target info. | ||
116 | */ | ||
117 | mv78xx0_mbus_dram_info.mbus_dram_target_id = TARGET_DDR; | ||
118 | |||
119 | if (mv78xx0_core_index() == 0) | ||
120 | addr = (void __iomem *)DDR_WINDOW_CPU0_BASE; | ||
121 | else | ||
122 | addr = (void __iomem *)DDR_WINDOW_CPU1_BASE; | ||
123 | |||
124 | for (i = 0, cs = 0; i < 4; i++) { | ||
125 | u32 base = readl(addr + DDR_BASE_CS_OFF(i)); | ||
126 | u32 size = readl(addr + DDR_SIZE_CS_OFF(i)); | ||
127 | |||
128 | /* | ||
129 | * Chip select enabled? | ||
130 | */ | ||
131 | if (size & 1) { | ||
132 | struct mbus_dram_window *w; | ||
133 | |||
134 | w = &mv78xx0_mbus_dram_info.cs[cs++]; | ||
135 | w->cs_index = i; | ||
136 | w->mbus_attr = 0xf & ~(1 << i); | ||
137 | w->base = base & 0xffff0000; | ||
138 | w->size = (size | 0x0000ffff) + 1; | ||
139 | } | ||
140 | } | ||
141 | mv78xx0_mbus_dram_info.num_cs = cs; | ||
142 | } | ||
143 | |||
144 | void __init mv78xx0_setup_pcie_io_win(int window, u32 base, u32 size, | ||
145 | int maj, int min) | ||
146 | { | ||
147 | setup_cpu_win(window, base, size, TARGET_PCIE(maj), | ||
148 | ATTR_PCIE_IO(min), -1); | ||
149 | } | ||
150 | |||
151 | void __init mv78xx0_setup_pcie_mem_win(int window, u32 base, u32 size, | ||
152 | int maj, int min) | ||
153 | { | ||
154 | setup_cpu_win(window, base, size, TARGET_PCIE(maj), | ||
155 | ATTR_PCIE_MEM(min), -1); | ||
156 | } | ||
diff --git a/arch/arm/mach-mv78xx0/common.c b/arch/arm/mach-mv78xx0/common.c new file mode 100644 index 000000000000..d27b83b7bf62 --- /dev/null +++ b/arch/arm/mach-mv78xx0/common.c | |||
@@ -0,0 +1,754 @@ | |||
1 | /* | ||
2 | * arch/arm/mach-mv78xx0/common.c | ||
3 | * | ||
4 | * Core functions for Marvell MV78xx0 SoCs | ||
5 | * | ||
6 | * This file is licensed under the terms of the GNU General Public | ||
7 | * License version 2. This program is licensed "as is" without any | ||
8 | * warranty of any kind, whether express or implied. | ||
9 | */ | ||
10 | |||
11 | #include <linux/kernel.h> | ||
12 | #include <linux/init.h> | ||
13 | #include <linux/platform_device.h> | ||
14 | #include <linux/serial_8250.h> | ||
15 | #include <linux/mbus.h> | ||
16 | #include <linux/mv643xx_eth.h> | ||
17 | #include <linux/ata_platform.h> | ||
18 | #include <asm/mach/map.h> | ||
19 | #include <asm/mach/time.h> | ||
20 | #include <asm/arch/mv78xx0.h> | ||
21 | #include <asm/plat-orion/cache-feroceon-l2.h> | ||
22 | #include <asm/plat-orion/ehci-orion.h> | ||
23 | #include <asm/plat-orion/orion_nand.h> | ||
24 | #include <asm/plat-orion/time.h> | ||
25 | #include "common.h" | ||
26 | |||
27 | |||
28 | /***************************************************************************** | ||
29 | * Common bits | ||
30 | ****************************************************************************/ | ||
31 | int mv78xx0_core_index(void) | ||
32 | { | ||
33 | u32 extra; | ||
34 | |||
35 | /* | ||
36 | * Read Extra Features register. | ||
37 | */ | ||
38 | __asm__("mrc p15, 1, %0, c15, c1, 0" : "=r" (extra)); | ||
39 | |||
40 | return !!(extra & 0x00004000); | ||
41 | } | ||
42 | |||
43 | static int get_hclk(void) | ||
44 | { | ||
45 | int hclk; | ||
46 | |||
47 | /* | ||
48 | * HCLK tick rate is configured by DEV_D[7:5] pins. | ||
49 | */ | ||
50 | switch ((readl(SAMPLE_AT_RESET_LOW) >> 5) & 7) { | ||
51 | case 0: | ||
52 | hclk = 166666667; | ||
53 | break; | ||
54 | case 1: | ||
55 | hclk = 200000000; | ||
56 | break; | ||
57 | case 2: | ||
58 | hclk = 266666667; | ||
59 | break; | ||
60 | case 3: | ||
61 | hclk = 333333333; | ||
62 | break; | ||
63 | case 4: | ||
64 | hclk = 400000000; | ||
65 | break; | ||
66 | default: | ||
67 | panic("unknown HCLK PLL setting: %.8x\n", | ||
68 | readl(SAMPLE_AT_RESET_LOW)); | ||
69 | } | ||
70 | |||
71 | return hclk; | ||
72 | } | ||
73 | |||
74 | static void get_pclk_l2clk(int hclk, int core_index, int *pclk, int *l2clk) | ||
75 | { | ||
76 | u32 cfg; | ||
77 | |||
78 | /* | ||
79 | * Core #0 PCLK/L2CLK is configured by bits [13:8], core #1 | ||
80 | * PCLK/L2CLK by bits [19:14]. | ||
81 | */ | ||
82 | if (core_index == 0) { | ||
83 | cfg = (readl(SAMPLE_AT_RESET_LOW) >> 8) & 0x3f; | ||
84 | } else { | ||
85 | cfg = (readl(SAMPLE_AT_RESET_LOW) >> 14) & 0x3f; | ||
86 | } | ||
87 | |||
88 | /* | ||
89 | * Bits [11:8] ([17:14] for core #1) configure the PCLK:HCLK | ||
90 | * ratio (1, 1.5, 2, 2.5, 3, 3.5, 4, 4.5, 5, 5.5, 6). | ||
91 | */ | ||
92 | *pclk = ((u64)hclk * (2 + (cfg & 0xf))) >> 1; | ||
93 | |||
94 | /* | ||
95 | * Bits [13:12] ([19:18] for core #1) configure the PCLK:L2CLK | ||
96 | * ratio (1, 2, 3). | ||
97 | */ | ||
98 | *l2clk = *pclk / (((cfg >> 4) & 3) + 1); | ||
99 | } | ||
100 | |||
101 | static int get_tclk(void) | ||
102 | { | ||
103 | int tclk; | ||
104 | |||
105 | /* | ||
106 | * TCLK tick rate is configured by DEV_A[2:0] strap pins. | ||
107 | */ | ||
108 | switch ((readl(SAMPLE_AT_RESET_HIGH) >> 6) & 7) { | ||
109 | case 1: | ||
110 | tclk = 166666667; | ||
111 | break; | ||
112 | case 3: | ||
113 | tclk = 200000000; | ||
114 | break; | ||
115 | default: | ||
116 | panic("unknown TCLK PLL setting: %.8x\n", | ||
117 | readl(SAMPLE_AT_RESET_HIGH)); | ||
118 | } | ||
119 | |||
120 | return tclk; | ||
121 | } | ||
122 | |||
123 | |||
124 | /***************************************************************************** | ||
125 | * I/O Address Mapping | ||
126 | ****************************************************************************/ | ||
127 | static struct map_desc mv78xx0_io_desc[] __initdata = { | ||
128 | { | ||
129 | .virtual = MV78XX0_CORE_REGS_VIRT_BASE, | ||
130 | .pfn = 0, | ||
131 | .length = MV78XX0_CORE_REGS_SIZE, | ||
132 | .type = MT_DEVICE, | ||
133 | }, { | ||
134 | .virtual = MV78XX0_PCIE_IO_VIRT_BASE(0), | ||
135 | .pfn = __phys_to_pfn(MV78XX0_PCIE_IO_PHYS_BASE(0)), | ||
136 | .length = MV78XX0_PCIE_IO_SIZE * 8, | ||
137 | .type = MT_DEVICE, | ||
138 | }, { | ||
139 | .virtual = MV78XX0_REGS_VIRT_BASE, | ||
140 | .pfn = __phys_to_pfn(MV78XX0_REGS_PHYS_BASE), | ||
141 | .length = MV78XX0_REGS_SIZE, | ||
142 | .type = MT_DEVICE, | ||
143 | }, | ||
144 | }; | ||
145 | |||
146 | void __init mv78xx0_map_io(void) | ||
147 | { | ||
148 | unsigned long phys; | ||
149 | |||
150 | /* | ||
151 | * Map the right set of per-core registers depending on | ||
152 | * which core we are running on. | ||
153 | */ | ||
154 | if (mv78xx0_core_index() == 0) { | ||
155 | phys = MV78XX0_CORE0_REGS_PHYS_BASE; | ||
156 | } else { | ||
157 | phys = MV78XX0_CORE1_REGS_PHYS_BASE; | ||
158 | } | ||
159 | mv78xx0_io_desc[0].pfn = __phys_to_pfn(phys); | ||
160 | |||
161 | iotable_init(mv78xx0_io_desc, ARRAY_SIZE(mv78xx0_io_desc)); | ||
162 | } | ||
163 | |||
164 | |||
165 | /***************************************************************************** | ||
166 | * EHCI | ||
167 | ****************************************************************************/ | ||
168 | static struct orion_ehci_data mv78xx0_ehci_data = { | ||
169 | .dram = &mv78xx0_mbus_dram_info, | ||
170 | }; | ||
171 | |||
172 | static u64 ehci_dmamask = 0xffffffffUL; | ||
173 | |||
174 | |||
175 | /***************************************************************************** | ||
176 | * EHCI0 | ||
177 | ****************************************************************************/ | ||
178 | static struct resource mv78xx0_ehci0_resources[] = { | ||
179 | { | ||
180 | .start = USB0_PHYS_BASE, | ||
181 | .end = USB0_PHYS_BASE + 0x0fff, | ||
182 | .flags = IORESOURCE_MEM, | ||
183 | }, { | ||
184 | .start = IRQ_MV78XX0_USB_0, | ||
185 | .end = IRQ_MV78XX0_USB_0, | ||
186 | .flags = IORESOURCE_IRQ, | ||
187 | }, | ||
188 | }; | ||
189 | |||
190 | static struct platform_device mv78xx0_ehci0 = { | ||
191 | .name = "orion-ehci", | ||
192 | .id = 0, | ||
193 | .dev = { | ||
194 | .dma_mask = &ehci_dmamask, | ||
195 | .coherent_dma_mask = 0xffffffff, | ||
196 | .platform_data = &mv78xx0_ehci_data, | ||
197 | }, | ||
198 | .resource = mv78xx0_ehci0_resources, | ||
199 | .num_resources = ARRAY_SIZE(mv78xx0_ehci0_resources), | ||
200 | }; | ||
201 | |||
202 | void __init mv78xx0_ehci0_init(void) | ||
203 | { | ||
204 | platform_device_register(&mv78xx0_ehci0); | ||
205 | } | ||
206 | |||
207 | |||
208 | /***************************************************************************** | ||
209 | * EHCI1 | ||
210 | ****************************************************************************/ | ||
211 | static struct resource mv78xx0_ehci1_resources[] = { | ||
212 | { | ||
213 | .start = USB1_PHYS_BASE, | ||
214 | .end = USB1_PHYS_BASE + 0x0fff, | ||
215 | .flags = IORESOURCE_MEM, | ||
216 | }, { | ||
217 | .start = IRQ_MV78XX0_USB_1, | ||
218 | .end = IRQ_MV78XX0_USB_1, | ||
219 | .flags = IORESOURCE_IRQ, | ||
220 | }, | ||
221 | }; | ||
222 | |||
223 | static struct platform_device mv78xx0_ehci1 = { | ||
224 | .name = "orion-ehci", | ||
225 | .id = 1, | ||
226 | .dev = { | ||
227 | .dma_mask = &ehci_dmamask, | ||
228 | .coherent_dma_mask = 0xffffffff, | ||
229 | .platform_data = &mv78xx0_ehci_data, | ||
230 | }, | ||
231 | .resource = mv78xx0_ehci1_resources, | ||
232 | .num_resources = ARRAY_SIZE(mv78xx0_ehci1_resources), | ||
233 | }; | ||
234 | |||
235 | void __init mv78xx0_ehci1_init(void) | ||
236 | { | ||
237 | platform_device_register(&mv78xx0_ehci1); | ||
238 | } | ||
239 | |||
240 | |||
241 | /***************************************************************************** | ||
242 | * EHCI2 | ||
243 | ****************************************************************************/ | ||
244 | static struct resource mv78xx0_ehci2_resources[] = { | ||
245 | { | ||
246 | .start = USB2_PHYS_BASE, | ||
247 | .end = USB2_PHYS_BASE + 0x0fff, | ||
248 | .flags = IORESOURCE_MEM, | ||
249 | }, { | ||
250 | .start = IRQ_MV78XX0_USB_2, | ||
251 | .end = IRQ_MV78XX0_USB_2, | ||
252 | .flags = IORESOURCE_IRQ, | ||
253 | }, | ||
254 | }; | ||
255 | |||
256 | static struct platform_device mv78xx0_ehci2 = { | ||
257 | .name = "orion-ehci", | ||
258 | .id = 2, | ||
259 | .dev = { | ||
260 | .dma_mask = &ehci_dmamask, | ||
261 | .coherent_dma_mask = 0xffffffff, | ||
262 | .platform_data = &mv78xx0_ehci_data, | ||
263 | }, | ||
264 | .resource = mv78xx0_ehci2_resources, | ||
265 | .num_resources = ARRAY_SIZE(mv78xx0_ehci2_resources), | ||
266 | }; | ||
267 | |||
268 | void __init mv78xx0_ehci2_init(void) | ||
269 | { | ||
270 | platform_device_register(&mv78xx0_ehci2); | ||
271 | } | ||
272 | |||
273 | |||
274 | /***************************************************************************** | ||
275 | * GE00 | ||
276 | ****************************************************************************/ | ||
277 | struct mv643xx_eth_shared_platform_data mv78xx0_ge00_shared_data = { | ||
278 | .t_clk = 0, | ||
279 | .dram = &mv78xx0_mbus_dram_info, | ||
280 | }; | ||
281 | |||
282 | static struct resource mv78xx0_ge00_shared_resources[] = { | ||
283 | { | ||
284 | .name = "ge00 base", | ||
285 | .start = GE00_PHYS_BASE + 0x2000, | ||
286 | .end = GE00_PHYS_BASE + 0x3fff, | ||
287 | .flags = IORESOURCE_MEM, | ||
288 | }, | ||
289 | }; | ||
290 | |||
291 | static struct platform_device mv78xx0_ge00_shared = { | ||
292 | .name = MV643XX_ETH_SHARED_NAME, | ||
293 | .id = 0, | ||
294 | .dev = { | ||
295 | .platform_data = &mv78xx0_ge00_shared_data, | ||
296 | }, | ||
297 | .num_resources = 1, | ||
298 | .resource = mv78xx0_ge00_shared_resources, | ||
299 | }; | ||
300 | |||
301 | static struct resource mv78xx0_ge00_resources[] = { | ||
302 | { | ||
303 | .name = "ge00 irq", | ||
304 | .start = IRQ_MV78XX0_GE00_SUM, | ||
305 | .end = IRQ_MV78XX0_GE00_SUM, | ||
306 | .flags = IORESOURCE_IRQ, | ||
307 | }, | ||
308 | }; | ||
309 | |||
310 | static struct platform_device mv78xx0_ge00 = { | ||
311 | .name = MV643XX_ETH_NAME, | ||
312 | .id = 0, | ||
313 | .num_resources = 1, | ||
314 | .resource = mv78xx0_ge00_resources, | ||
315 | }; | ||
316 | |||
317 | void __init mv78xx0_ge00_init(struct mv643xx_eth_platform_data *eth_data) | ||
318 | { | ||
319 | eth_data->shared = &mv78xx0_ge00_shared; | ||
320 | mv78xx0_ge00.dev.platform_data = eth_data; | ||
321 | |||
322 | platform_device_register(&mv78xx0_ge00_shared); | ||
323 | platform_device_register(&mv78xx0_ge00); | ||
324 | } | ||
325 | |||
326 | |||
327 | /***************************************************************************** | ||
328 | * GE01 | ||
329 | ****************************************************************************/ | ||
330 | struct mv643xx_eth_shared_platform_data mv78xx0_ge01_shared_data = { | ||
331 | .t_clk = 0, | ||
332 | .dram = &mv78xx0_mbus_dram_info, | ||
333 | }; | ||
334 | |||
335 | static struct resource mv78xx0_ge01_shared_resources[] = { | ||
336 | { | ||
337 | .name = "ge01 base", | ||
338 | .start = GE01_PHYS_BASE + 0x2000, | ||
339 | .end = GE01_PHYS_BASE + 0x3fff, | ||
340 | .flags = IORESOURCE_MEM, | ||
341 | }, | ||
342 | }; | ||
343 | |||
344 | static struct platform_device mv78xx0_ge01_shared = { | ||
345 | .name = MV643XX_ETH_SHARED_NAME, | ||
346 | .id = 1, | ||
347 | .dev = { | ||
348 | .platform_data = &mv78xx0_ge01_shared_data, | ||
349 | }, | ||
350 | .num_resources = 1, | ||
351 | .resource = mv78xx0_ge01_shared_resources, | ||
352 | }; | ||
353 | |||
354 | static struct resource mv78xx0_ge01_resources[] = { | ||
355 | { | ||
356 | .name = "ge01 irq", | ||
357 | .start = IRQ_MV78XX0_GE01_SUM, | ||
358 | .end = IRQ_MV78XX0_GE01_SUM, | ||
359 | .flags = IORESOURCE_IRQ, | ||
360 | }, | ||
361 | }; | ||
362 | |||
363 | static struct platform_device mv78xx0_ge01 = { | ||
364 | .name = MV643XX_ETH_NAME, | ||
365 | .id = 1, | ||
366 | .num_resources = 1, | ||
367 | .resource = mv78xx0_ge01_resources, | ||
368 | }; | ||
369 | |||
370 | void __init mv78xx0_ge01_init(struct mv643xx_eth_platform_data *eth_data) | ||
371 | { | ||
372 | eth_data->shared = &mv78xx0_ge01_shared; | ||
373 | eth_data->shared_smi = &mv78xx0_ge00_shared; | ||
374 | mv78xx0_ge01.dev.platform_data = eth_data; | ||
375 | |||
376 | platform_device_register(&mv78xx0_ge01_shared); | ||
377 | platform_device_register(&mv78xx0_ge01); | ||
378 | } | ||
379 | |||
380 | |||
381 | /***************************************************************************** | ||
382 | * GE10 | ||
383 | ****************************************************************************/ | ||
384 | struct mv643xx_eth_shared_platform_data mv78xx0_ge10_shared_data = { | ||
385 | .t_clk = 0, | ||
386 | .dram = &mv78xx0_mbus_dram_info, | ||
387 | }; | ||
388 | |||
389 | static struct resource mv78xx0_ge10_shared_resources[] = { | ||
390 | { | ||
391 | .name = "ge10 base", | ||
392 | .start = GE10_PHYS_BASE + 0x2000, | ||
393 | .end = GE10_PHYS_BASE + 0x3fff, | ||
394 | .flags = IORESOURCE_MEM, | ||
395 | }, | ||
396 | }; | ||
397 | |||
398 | static struct platform_device mv78xx0_ge10_shared = { | ||
399 | .name = MV643XX_ETH_SHARED_NAME, | ||
400 | .id = 2, | ||
401 | .dev = { | ||
402 | .platform_data = &mv78xx0_ge10_shared_data, | ||
403 | }, | ||
404 | .num_resources = 1, | ||
405 | .resource = mv78xx0_ge10_shared_resources, | ||
406 | }; | ||
407 | |||
408 | static struct resource mv78xx0_ge10_resources[] = { | ||
409 | { | ||
410 | .name = "ge10 irq", | ||
411 | .start = IRQ_MV78XX0_GE10_SUM, | ||
412 | .end = IRQ_MV78XX0_GE10_SUM, | ||
413 | .flags = IORESOURCE_IRQ, | ||
414 | }, | ||
415 | }; | ||
416 | |||
417 | static struct platform_device mv78xx0_ge10 = { | ||
418 | .name = MV643XX_ETH_NAME, | ||
419 | .id = 2, | ||
420 | .num_resources = 1, | ||
421 | .resource = mv78xx0_ge10_resources, | ||
422 | }; | ||
423 | |||
424 | void __init mv78xx0_ge10_init(struct mv643xx_eth_platform_data *eth_data) | ||
425 | { | ||
426 | eth_data->shared = &mv78xx0_ge10_shared; | ||
427 | eth_data->shared_smi = &mv78xx0_ge00_shared; | ||
428 | mv78xx0_ge10.dev.platform_data = eth_data; | ||
429 | |||
430 | platform_device_register(&mv78xx0_ge10_shared); | ||
431 | platform_device_register(&mv78xx0_ge10); | ||
432 | } | ||
433 | |||
434 | |||
435 | /***************************************************************************** | ||
436 | * GE11 | ||
437 | ****************************************************************************/ | ||
438 | struct mv643xx_eth_shared_platform_data mv78xx0_ge11_shared_data = { | ||
439 | .t_clk = 0, | ||
440 | .dram = &mv78xx0_mbus_dram_info, | ||
441 | }; | ||
442 | |||
443 | static struct resource mv78xx0_ge11_shared_resources[] = { | ||
444 | { | ||
445 | .name = "ge11 base", | ||
446 | .start = GE11_PHYS_BASE + 0x2000, | ||
447 | .end = GE11_PHYS_BASE + 0x3fff, | ||
448 | .flags = IORESOURCE_MEM, | ||
449 | }, | ||
450 | }; | ||
451 | |||
452 | static struct platform_device mv78xx0_ge11_shared = { | ||
453 | .name = MV643XX_ETH_SHARED_NAME, | ||
454 | .id = 3, | ||
455 | .dev = { | ||
456 | .platform_data = &mv78xx0_ge11_shared_data, | ||
457 | }, | ||
458 | .num_resources = 1, | ||
459 | .resource = mv78xx0_ge11_shared_resources, | ||
460 | }; | ||
461 | |||
462 | static struct resource mv78xx0_ge11_resources[] = { | ||
463 | { | ||
464 | .name = "ge11 irq", | ||
465 | .start = IRQ_MV78XX0_GE11_SUM, | ||
466 | .end = IRQ_MV78XX0_GE11_SUM, | ||
467 | .flags = IORESOURCE_IRQ, | ||
468 | }, | ||
469 | }; | ||
470 | |||
471 | static struct platform_device mv78xx0_ge11 = { | ||
472 | .name = MV643XX_ETH_NAME, | ||
473 | .id = 3, | ||
474 | .num_resources = 1, | ||
475 | .resource = mv78xx0_ge11_resources, | ||
476 | }; | ||
477 | |||
478 | void __init mv78xx0_ge11_init(struct mv643xx_eth_platform_data *eth_data) | ||
479 | { | ||
480 | eth_data->shared = &mv78xx0_ge11_shared; | ||
481 | eth_data->shared_smi = &mv78xx0_ge00_shared; | ||
482 | mv78xx0_ge11.dev.platform_data = eth_data; | ||
483 | |||
484 | platform_device_register(&mv78xx0_ge11_shared); | ||
485 | platform_device_register(&mv78xx0_ge11); | ||
486 | } | ||
487 | |||
488 | |||
489 | /***************************************************************************** | ||
490 | * SATA | ||
491 | ****************************************************************************/ | ||
492 | static struct resource mv78xx0_sata_resources[] = { | ||
493 | { | ||
494 | .name = "sata base", | ||
495 | .start = SATA_PHYS_BASE, | ||
496 | .end = SATA_PHYS_BASE + 0x5000 - 1, | ||
497 | .flags = IORESOURCE_MEM, | ||
498 | }, { | ||
499 | .name = "sata irq", | ||
500 | .start = IRQ_MV78XX0_SATA, | ||
501 | .end = IRQ_MV78XX0_SATA, | ||
502 | .flags = IORESOURCE_IRQ, | ||
503 | }, | ||
504 | }; | ||
505 | |||
506 | static struct platform_device mv78xx0_sata = { | ||
507 | .name = "sata_mv", | ||
508 | .id = 0, | ||
509 | .dev = { | ||
510 | .coherent_dma_mask = 0xffffffff, | ||
511 | }, | ||
512 | .num_resources = ARRAY_SIZE(mv78xx0_sata_resources), | ||
513 | .resource = mv78xx0_sata_resources, | ||
514 | }; | ||
515 | |||
516 | void __init mv78xx0_sata_init(struct mv_sata_platform_data *sata_data) | ||
517 | { | ||
518 | sata_data->dram = &mv78xx0_mbus_dram_info; | ||
519 | mv78xx0_sata.dev.platform_data = sata_data; | ||
520 | platform_device_register(&mv78xx0_sata); | ||
521 | } | ||
522 | |||
523 | |||
524 | /***************************************************************************** | ||
525 | * UART0 | ||
526 | ****************************************************************************/ | ||
527 | static struct plat_serial8250_port mv78xx0_uart0_data[] = { | ||
528 | { | ||
529 | .mapbase = UART0_PHYS_BASE, | ||
530 | .membase = (char *)UART0_VIRT_BASE, | ||
531 | .irq = IRQ_MV78XX0_UART_0, | ||
532 | .flags = UPF_SKIP_TEST | UPF_BOOT_AUTOCONF, | ||
533 | .iotype = UPIO_MEM, | ||
534 | .regshift = 2, | ||
535 | .uartclk = 0, | ||
536 | }, { | ||
537 | }, | ||
538 | }; | ||
539 | |||
540 | static struct resource mv78xx0_uart0_resources[] = { | ||
541 | { | ||
542 | .start = UART0_PHYS_BASE, | ||
543 | .end = UART0_PHYS_BASE + 0xff, | ||
544 | .flags = IORESOURCE_MEM, | ||
545 | }, { | ||
546 | .start = IRQ_MV78XX0_UART_0, | ||
547 | .end = IRQ_MV78XX0_UART_0, | ||
548 | .flags = IORESOURCE_IRQ, | ||
549 | }, | ||
550 | }; | ||
551 | |||
552 | static struct platform_device mv78xx0_uart0 = { | ||
553 | .name = "serial8250", | ||
554 | .id = 0, | ||
555 | .dev = { | ||
556 | .platform_data = mv78xx0_uart0_data, | ||
557 | }, | ||
558 | .resource = mv78xx0_uart0_resources, | ||
559 | .num_resources = ARRAY_SIZE(mv78xx0_uart0_resources), | ||
560 | }; | ||
561 | |||
562 | void __init mv78xx0_uart0_init(void) | ||
563 | { | ||
564 | platform_device_register(&mv78xx0_uart0); | ||
565 | } | ||
566 | |||
567 | |||
568 | /***************************************************************************** | ||
569 | * UART1 | ||
570 | ****************************************************************************/ | ||
571 | static struct plat_serial8250_port mv78xx0_uart1_data[] = { | ||
572 | { | ||
573 | .mapbase = UART1_PHYS_BASE, | ||
574 | .membase = (char *)UART1_VIRT_BASE, | ||
575 | .irq = IRQ_MV78XX0_UART_1, | ||
576 | .flags = UPF_SKIP_TEST | UPF_BOOT_AUTOCONF, | ||
577 | .iotype = UPIO_MEM, | ||
578 | .regshift = 2, | ||
579 | .uartclk = 0, | ||
580 | }, { | ||
581 | }, | ||
582 | }; | ||
583 | |||
584 | static struct resource mv78xx0_uart1_resources[] = { | ||
585 | { | ||
586 | .start = UART1_PHYS_BASE, | ||
587 | .end = UART1_PHYS_BASE + 0xff, | ||
588 | .flags = IORESOURCE_MEM, | ||
589 | }, { | ||
590 | .start = IRQ_MV78XX0_UART_1, | ||
591 | .end = IRQ_MV78XX0_UART_1, | ||
592 | .flags = IORESOURCE_IRQ, | ||
593 | }, | ||
594 | }; | ||
595 | |||
596 | static struct platform_device mv78xx0_uart1 = { | ||
597 | .name = "serial8250", | ||
598 | .id = 1, | ||
599 | .dev = { | ||
600 | .platform_data = mv78xx0_uart1_data, | ||
601 | }, | ||
602 | .resource = mv78xx0_uart1_resources, | ||
603 | .num_resources = ARRAY_SIZE(mv78xx0_uart1_resources), | ||
604 | }; | ||
605 | |||
606 | void __init mv78xx0_uart1_init(void) | ||
607 | { | ||
608 | platform_device_register(&mv78xx0_uart1); | ||
609 | } | ||
610 | |||
611 | |||
612 | /***************************************************************************** | ||
613 | * UART2 | ||
614 | ****************************************************************************/ | ||
615 | static struct plat_serial8250_port mv78xx0_uart2_data[] = { | ||
616 | { | ||
617 | .mapbase = UART2_PHYS_BASE, | ||
618 | .membase = (char *)UART2_VIRT_BASE, | ||
619 | .irq = IRQ_MV78XX0_UART_2, | ||
620 | .flags = UPF_SKIP_TEST | UPF_BOOT_AUTOCONF, | ||
621 | .iotype = UPIO_MEM, | ||
622 | .regshift = 2, | ||
623 | .uartclk = 0, | ||
624 | }, { | ||
625 | }, | ||
626 | }; | ||
627 | |||
628 | static struct resource mv78xx0_uart2_resources[] = { | ||
629 | { | ||
630 | .start = UART2_PHYS_BASE, | ||
631 | .end = UART2_PHYS_BASE + 0xff, | ||
632 | .flags = IORESOURCE_MEM, | ||
633 | }, { | ||
634 | .start = IRQ_MV78XX0_UART_2, | ||
635 | .end = IRQ_MV78XX0_UART_2, | ||
636 | .flags = IORESOURCE_IRQ, | ||
637 | }, | ||
638 | }; | ||
639 | |||
640 | static struct platform_device mv78xx0_uart2 = { | ||
641 | .name = "serial8250", | ||
642 | .id = 2, | ||
643 | .dev = { | ||
644 | .platform_data = mv78xx0_uart2_data, | ||
645 | }, | ||
646 | .resource = mv78xx0_uart2_resources, | ||
647 | .num_resources = ARRAY_SIZE(mv78xx0_uart2_resources), | ||
648 | }; | ||
649 | |||
650 | void __init mv78xx0_uart2_init(void) | ||
651 | { | ||
652 | platform_device_register(&mv78xx0_uart2); | ||
653 | } | ||
654 | |||
655 | |||
656 | /***************************************************************************** | ||
657 | * UART3 | ||
658 | ****************************************************************************/ | ||
659 | static struct plat_serial8250_port mv78xx0_uart3_data[] = { | ||
660 | { | ||
661 | .mapbase = UART3_PHYS_BASE, | ||
662 | .membase = (char *)UART3_VIRT_BASE, | ||
663 | .irq = IRQ_MV78XX0_UART_3, | ||
664 | .flags = UPF_SKIP_TEST | UPF_BOOT_AUTOCONF, | ||
665 | .iotype = UPIO_MEM, | ||
666 | .regshift = 2, | ||
667 | .uartclk = 0, | ||
668 | }, { | ||
669 | }, | ||
670 | }; | ||
671 | |||
672 | static struct resource mv78xx0_uart3_resources[] = { | ||
673 | { | ||
674 | .start = UART3_PHYS_BASE, | ||
675 | .end = UART3_PHYS_BASE + 0xff, | ||
676 | .flags = IORESOURCE_MEM, | ||
677 | }, { | ||
678 | .start = IRQ_MV78XX0_UART_3, | ||
679 | .end = IRQ_MV78XX0_UART_3, | ||
680 | .flags = IORESOURCE_IRQ, | ||
681 | }, | ||
682 | }; | ||
683 | |||
684 | static struct platform_device mv78xx0_uart3 = { | ||
685 | .name = "serial8250", | ||
686 | .id = 3, | ||
687 | .dev = { | ||
688 | .platform_data = mv78xx0_uart3_data, | ||
689 | }, | ||
690 | .resource = mv78xx0_uart3_resources, | ||
691 | .num_resources = ARRAY_SIZE(mv78xx0_uart3_resources), | ||
692 | }; | ||
693 | |||
694 | void __init mv78xx0_uart3_init(void) | ||
695 | { | ||
696 | platform_device_register(&mv78xx0_uart3); | ||
697 | } | ||
698 | |||
699 | |||
700 | /***************************************************************************** | ||
701 | * Time handling | ||
702 | ****************************************************************************/ | ||
703 | static void mv78xx0_timer_init(void) | ||
704 | { | ||
705 | orion_time_init(IRQ_MV78XX0_TIMER_1, get_tclk()); | ||
706 | } | ||
707 | |||
708 | struct sys_timer mv78xx0_timer = { | ||
709 | .init = mv78xx0_timer_init, | ||
710 | }; | ||
711 | |||
712 | |||
713 | /***************************************************************************** | ||
714 | * General | ||
715 | ****************************************************************************/ | ||
716 | static int __init is_l2_writethrough(void) | ||
717 | { | ||
718 | return !!(readl(CPU_CONTROL) & L2_WRITETHROUGH); | ||
719 | } | ||
720 | |||
721 | void __init mv78xx0_init(void) | ||
722 | { | ||
723 | int core_index; | ||
724 | int hclk; | ||
725 | int pclk; | ||
726 | int l2clk; | ||
727 | int tclk; | ||
728 | |||
729 | core_index = mv78xx0_core_index(); | ||
730 | hclk = get_hclk(); | ||
731 | get_pclk_l2clk(hclk, core_index, &pclk, &l2clk); | ||
732 | tclk = get_tclk(); | ||
733 | |||
734 | printk(KERN_INFO "MV78xx0 core #%d, ", core_index); | ||
735 | printk("PCLK = %dMHz, ", (pclk + 499999) / 1000000); | ||
736 | printk("L2 = %dMHz, ", (l2clk + 499999) / 1000000); | ||
737 | printk("HCLK = %dMHz, ", (hclk + 499999) / 1000000); | ||
738 | printk("TCLK = %dMHz\n", (tclk + 499999) / 1000000); | ||
739 | |||
740 | mv78xx0_setup_cpu_mbus(); | ||
741 | |||
742 | #ifdef CONFIG_CACHE_FEROCEON_L2 | ||
743 | feroceon_l2_init(is_l2_writethrough()); | ||
744 | #endif | ||
745 | |||
746 | mv78xx0_ge00_shared_data.t_clk = tclk; | ||
747 | mv78xx0_ge01_shared_data.t_clk = tclk; | ||
748 | mv78xx0_ge10_shared_data.t_clk = tclk; | ||
749 | mv78xx0_ge11_shared_data.t_clk = tclk; | ||
750 | mv78xx0_uart0_data[0].uartclk = tclk; | ||
751 | mv78xx0_uart1_data[0].uartclk = tclk; | ||
752 | mv78xx0_uart2_data[0].uartclk = tclk; | ||
753 | mv78xx0_uart3_data[0].uartclk = tclk; | ||
754 | } | ||
diff --git a/arch/arm/mach-mv78xx0/common.h b/arch/arm/mach-mv78xx0/common.h new file mode 100644 index 000000000000..78af5de319dd --- /dev/null +++ b/arch/arm/mach-mv78xx0/common.h | |||
@@ -0,0 +1,49 @@ | |||
1 | /* | ||
2 | * arch/arm/mach-mv78xx0/common.h | ||
3 | * | ||
4 | * Core functions for Marvell MV78xx0 SoCs | ||
5 | * | ||
6 | * This file is licensed under the terms of the GNU General Public | ||
7 | * License version 2. This program is licensed "as is" without any | ||
8 | * warranty of any kind, whether express or implied. | ||
9 | */ | ||
10 | |||
11 | #ifndef __ARCH_MV78XX0_COMMON_H | ||
12 | #define __ARCH_MV78XX0_COMMON_H | ||
13 | |||
14 | struct mv643xx_eth_platform_data; | ||
15 | struct mv_sata_platform_data; | ||
16 | |||
17 | /* | ||
18 | * Basic MV78xx0 init functions used early by machine-setup. | ||
19 | */ | ||
20 | int mv78xx0_core_index(void); | ||
21 | void mv78xx0_map_io(void); | ||
22 | void mv78xx0_init(void); | ||
23 | void mv78xx0_init_irq(void); | ||
24 | |||
25 | extern struct mbus_dram_target_info mv78xx0_mbus_dram_info; | ||
26 | void mv78xx0_setup_cpu_mbus(void); | ||
27 | void mv78xx0_setup_pcie_io_win(int window, u32 base, u32 size, | ||
28 | int maj, int min); | ||
29 | void mv78xx0_setup_pcie_mem_win(int window, u32 base, u32 size, | ||
30 | int maj, int min); | ||
31 | |||
32 | void mv78xx0_ehci0_init(void); | ||
33 | void mv78xx0_ehci1_init(void); | ||
34 | void mv78xx0_ehci2_init(void); | ||
35 | void mv78xx0_ge00_init(struct mv643xx_eth_platform_data *eth_data); | ||
36 | void mv78xx0_ge01_init(struct mv643xx_eth_platform_data *eth_data); | ||
37 | void mv78xx0_ge10_init(struct mv643xx_eth_platform_data *eth_data); | ||
38 | void mv78xx0_ge11_init(struct mv643xx_eth_platform_data *eth_data); | ||
39 | void mv78xx0_pcie_init(int init_port0, int init_port1); | ||
40 | void mv78xx0_sata_init(struct mv_sata_platform_data *sata_data); | ||
41 | void mv78xx0_uart0_init(void); | ||
42 | void mv78xx0_uart1_init(void); | ||
43 | void mv78xx0_uart2_init(void); | ||
44 | void mv78xx0_uart3_init(void); | ||
45 | |||
46 | extern struct sys_timer mv78xx0_timer; | ||
47 | |||
48 | |||
49 | #endif | ||
diff --git a/arch/arm/mach-mv78xx0/db78x00-bp-setup.c b/arch/arm/mach-mv78xx0/db78x00-bp-setup.c new file mode 100644 index 000000000000..0c93d19193df --- /dev/null +++ b/arch/arm/mach-mv78xx0/db78x00-bp-setup.c | |||
@@ -0,0 +1,94 @@ | |||
1 | /* | ||
2 | * arch/arm/mach-mv78xx0/db78x00-bp-setup.c | ||
3 | * | ||
4 | * Marvell DB-78x00-BP Development Board Setup | ||
5 | * | ||
6 | * This file is licensed under the terms of the GNU General Public | ||
7 | * License version 2. This program is licensed "as is" without any | ||
8 | * warranty of any kind, whether express or implied. | ||
9 | */ | ||
10 | |||
11 | #include <linux/kernel.h> | ||
12 | #include <linux/init.h> | ||
13 | #include <linux/platform_device.h> | ||
14 | #include <linux/ata_platform.h> | ||
15 | #include <linux/mv643xx_eth.h> | ||
16 | #include <asm/arch/mv78xx0.h> | ||
17 | #include <asm/mach-types.h> | ||
18 | #include <asm/mach/arch.h> | ||
19 | #include "common.h" | ||
20 | |||
21 | static struct mv643xx_eth_platform_data db78x00_ge00_data = { | ||
22 | .phy_addr = 8, | ||
23 | }; | ||
24 | |||
25 | static struct mv643xx_eth_platform_data db78x00_ge01_data = { | ||
26 | .phy_addr = 9, | ||
27 | }; | ||
28 | |||
29 | static struct mv643xx_eth_platform_data db78x00_ge10_data = { | ||
30 | .phy_addr = -1, | ||
31 | }; | ||
32 | |||
33 | static struct mv643xx_eth_platform_data db78x00_ge11_data = { | ||
34 | .phy_addr = -1, | ||
35 | }; | ||
36 | |||
37 | static struct mv_sata_platform_data db78x00_sata_data = { | ||
38 | .n_ports = 2, | ||
39 | }; | ||
40 | |||
41 | static void __init db78x00_init(void) | ||
42 | { | ||
43 | /* | ||
44 | * Basic MV78xx0 setup. Needs to be called early. | ||
45 | */ | ||
46 | mv78xx0_init(); | ||
47 | |||
48 | /* | ||
49 | * Partition on-chip peripherals between the two CPU cores. | ||
50 | */ | ||
51 | if (mv78xx0_core_index() == 0) { | ||
52 | mv78xx0_ehci0_init(); | ||
53 | mv78xx0_ehci1_init(); | ||
54 | mv78xx0_ehci2_init(); | ||
55 | mv78xx0_ge00_init(&db78x00_ge00_data); | ||
56 | mv78xx0_ge01_init(&db78x00_ge01_data); | ||
57 | mv78xx0_ge10_init(&db78x00_ge10_data); | ||
58 | mv78xx0_ge11_init(&db78x00_ge11_data); | ||
59 | mv78xx0_sata_init(&db78x00_sata_data); | ||
60 | mv78xx0_uart0_init(); | ||
61 | mv78xx0_uart2_init(); | ||
62 | } else { | ||
63 | mv78xx0_uart1_init(); | ||
64 | mv78xx0_uart3_init(); | ||
65 | } | ||
66 | } | ||
67 | |||
68 | static int __init db78x00_pci_init(void) | ||
69 | { | ||
70 | if (machine_is_db78x00_bp()) { | ||
71 | /* | ||
72 | * Assign the x16 PCIe slot on the board to CPU core | ||
73 | * #0, and let CPU core #1 have the four x1 slots. | ||
74 | */ | ||
75 | if (mv78xx0_core_index() == 0) | ||
76 | mv78xx0_pcie_init(0, 1); | ||
77 | else | ||
78 | mv78xx0_pcie_init(1, 0); | ||
79 | } | ||
80 | |||
81 | return 0; | ||
82 | } | ||
83 | subsys_initcall(db78x00_pci_init); | ||
84 | |||
85 | MACHINE_START(DB78X00_BP, "Marvell DB-78x00-BP Development Board") | ||
86 | /* Maintainer: Lennert Buytenhek <buytenh@marvell.com> */ | ||
87 | .phys_io = MV78XX0_REGS_PHYS_BASE, | ||
88 | .io_pg_offst = ((MV78XX0_REGS_VIRT_BASE) >> 18) & 0xfffc, | ||
89 | .boot_params = 0x00000100, | ||
90 | .init_machine = db78x00_init, | ||
91 | .map_io = mv78xx0_map_io, | ||
92 | .init_irq = mv78xx0_init_irq, | ||
93 | .timer = &mv78xx0_timer, | ||
94 | MACHINE_END | ||
diff --git a/arch/arm/mach-mv78xx0/irq.c b/arch/arm/mach-mv78xx0/irq.c new file mode 100644 index 000000000000..60f4ee4d4532 --- /dev/null +++ b/arch/arm/mach-mv78xx0/irq.c | |||
@@ -0,0 +1,22 @@ | |||
1 | /* | ||
2 | * arch/arm/mach-mv78xx0/irq.c | ||
3 | * | ||
4 | * MV78xx0 IRQ handling. | ||
5 | * | ||
6 | * This file is licensed under the terms of the GNU General Public | ||
7 | * License version 2. This program is licensed "as is" without any | ||
8 | * warranty of any kind, whether express or implied. | ||
9 | */ | ||
10 | |||
11 | #include <linux/kernel.h> | ||
12 | #include <linux/init.h> | ||
13 | #include <linux/pci.h> | ||
14 | #include <asm/arch/mv78xx0.h> | ||
15 | #include <asm/plat-orion/irq.h> | ||
16 | #include "common.h" | ||
17 | |||
18 | void __init mv78xx0_init_irq(void) | ||
19 | { | ||
20 | orion_irq_init(0, (void __iomem *)(IRQ_VIRT_BASE + IRQ_MASK_LOW_OFF)); | ||
21 | orion_irq_init(32, (void __iomem *)(IRQ_VIRT_BASE + IRQ_MASK_HIGH_OFF)); | ||
22 | } | ||
diff --git a/arch/arm/mach-mv78xx0/pcie.c b/arch/arm/mach-mv78xx0/pcie.c new file mode 100644 index 000000000000..b78e1443159f --- /dev/null +++ b/arch/arm/mach-mv78xx0/pcie.c | |||
@@ -0,0 +1,312 @@ | |||
1 | /* | ||
2 | * arch/arm/mach-mv78xx0/pcie.c | ||
3 | * | ||
4 | * PCIe functions for Marvell MV78xx0 SoCs | ||
5 | * | ||
6 | * This file is licensed under the terms of the GNU General Public | ||
7 | * License version 2. This program is licensed "as is" without any | ||
8 | * warranty of any kind, whether express or implied. | ||
9 | */ | ||
10 | |||
11 | #include <linux/kernel.h> | ||
12 | #include <linux/pci.h> | ||
13 | #include <linux/mbus.h> | ||
14 | #include <asm/mach/pci.h> | ||
15 | #include <asm/plat-orion/pcie.h> | ||
16 | #include "common.h" | ||
17 | |||
18 | struct pcie_port { | ||
19 | u8 maj; | ||
20 | u8 min; | ||
21 | u8 root_bus_nr; | ||
22 | void __iomem *base; | ||
23 | spinlock_t conf_lock; | ||
24 | char io_space_name[16]; | ||
25 | char mem_space_name[16]; | ||
26 | struct resource res[2]; | ||
27 | }; | ||
28 | |||
29 | static struct pcie_port pcie_port[8]; | ||
30 | static int num_pcie_ports; | ||
31 | static struct resource pcie_io_space; | ||
32 | static struct resource pcie_mem_space; | ||
33 | |||
34 | |||
35 | static void __init mv78xx0_pcie_preinit(void) | ||
36 | { | ||
37 | int i; | ||
38 | u32 size_each; | ||
39 | u32 start; | ||
40 | int win; | ||
41 | |||
42 | pcie_io_space.name = "PCIe I/O Space"; | ||
43 | pcie_io_space.start = MV78XX0_PCIE_IO_PHYS_BASE(0); | ||
44 | pcie_io_space.end = | ||
45 | MV78XX0_PCIE_IO_PHYS_BASE(0) + MV78XX0_PCIE_IO_SIZE * 8 - 1; | ||
46 | pcie_io_space.flags = IORESOURCE_IO; | ||
47 | if (request_resource(&iomem_resource, &pcie_io_space)) | ||
48 | panic("can't allocate PCIe I/O space"); | ||
49 | |||
50 | pcie_mem_space.name = "PCIe MEM Space"; | ||
51 | pcie_mem_space.start = MV78XX0_PCIE_MEM_PHYS_BASE; | ||
52 | pcie_mem_space.end = | ||
53 | MV78XX0_PCIE_MEM_PHYS_BASE + MV78XX0_PCIE_MEM_SIZE - 1; | ||
54 | pcie_mem_space.flags = IORESOURCE_MEM; | ||
55 | if (request_resource(&iomem_resource, &pcie_mem_space)) | ||
56 | panic("can't allocate PCIe MEM space"); | ||
57 | |||
58 | for (i = 0; i < num_pcie_ports; i++) { | ||
59 | struct pcie_port *pp = pcie_port + i; | ||
60 | |||
61 | snprintf(pp->io_space_name, sizeof(pp->io_space_name), | ||
62 | "PCIe %d.%d I/O", pp->maj, pp->min); | ||
63 | pp->io_space_name[sizeof(pp->io_space_name) - 1] = 0; | ||
64 | pp->res[0].name = pp->io_space_name; | ||
65 | pp->res[0].start = MV78XX0_PCIE_IO_PHYS_BASE(i); | ||
66 | pp->res[0].end = pp->res[0].start + MV78XX0_PCIE_IO_SIZE - 1; | ||
67 | pp->res[0].flags = IORESOURCE_IO; | ||
68 | |||
69 | snprintf(pp->mem_space_name, sizeof(pp->mem_space_name), | ||
70 | "PCIe %d.%d MEM", pp->maj, pp->min); | ||
71 | pp->mem_space_name[sizeof(pp->mem_space_name) - 1] = 0; | ||
72 | pp->res[1].name = pp->mem_space_name; | ||
73 | pp->res[1].flags = IORESOURCE_MEM; | ||
74 | } | ||
75 | |||
76 | switch (num_pcie_ports) { | ||
77 | case 0: | ||
78 | size_each = 0; | ||
79 | break; | ||
80 | |||
81 | case 1: | ||
82 | size_each = 0x30000000; | ||
83 | break; | ||
84 | |||
85 | case 2 ... 3: | ||
86 | size_each = 0x10000000; | ||
87 | break; | ||
88 | |||
89 | case 4 ... 6: | ||
90 | size_each = 0x08000000; | ||
91 | break; | ||
92 | |||
93 | case 7: | ||
94 | size_each = 0x04000000; | ||
95 | break; | ||
96 | |||
97 | default: | ||
98 | panic("invalid number of PCIe ports"); | ||
99 | } | ||
100 | |||
101 | start = MV78XX0_PCIE_MEM_PHYS_BASE; | ||
102 | for (i = 0; i < num_pcie_ports; i++) { | ||
103 | struct pcie_port *pp = pcie_port + i; | ||
104 | |||
105 | pp->res[1].start = start; | ||
106 | pp->res[1].end = start + size_each - 1; | ||
107 | start += size_each; | ||
108 | } | ||
109 | |||
110 | for (i = 0; i < num_pcie_ports; i++) { | ||
111 | struct pcie_port *pp = pcie_port + i; | ||
112 | |||
113 | if (request_resource(&pcie_io_space, &pp->res[0])) | ||
114 | panic("can't allocate PCIe I/O sub-space"); | ||
115 | |||
116 | if (request_resource(&pcie_mem_space, &pp->res[1])) | ||
117 | panic("can't allocate PCIe MEM sub-space"); | ||
118 | } | ||
119 | |||
120 | win = 0; | ||
121 | for (i = 0; i < num_pcie_ports; i++) { | ||
122 | struct pcie_port *pp = pcie_port + i; | ||
123 | |||
124 | mv78xx0_setup_pcie_io_win(win++, pp->res[0].start, | ||
125 | pp->res[0].end - pp->res[0].start + 1, | ||
126 | pp->maj, pp->min); | ||
127 | |||
128 | mv78xx0_setup_pcie_mem_win(win++, pp->res[1].start, | ||
129 | pp->res[1].end - pp->res[1].start + 1, | ||
130 | pp->maj, pp->min); | ||
131 | } | ||
132 | } | ||
133 | |||
134 | static int __init mv78xx0_pcie_setup(int nr, struct pci_sys_data *sys) | ||
135 | { | ||
136 | struct pcie_port *pp; | ||
137 | |||
138 | if (nr >= num_pcie_ports) | ||
139 | return 0; | ||
140 | |||
141 | pp = &pcie_port[nr]; | ||
142 | pp->root_bus_nr = sys->busnr; | ||
143 | |||
144 | /* | ||
145 | * Generic PCIe unit setup. | ||
146 | */ | ||
147 | orion_pcie_set_local_bus_nr(pp->base, sys->busnr); | ||
148 | orion_pcie_setup(pp->base, &mv78xx0_mbus_dram_info); | ||
149 | |||
150 | sys->resource[0] = &pp->res[0]; | ||
151 | sys->resource[1] = &pp->res[1]; | ||
152 | sys->resource[2] = NULL; | ||
153 | |||
154 | return 1; | ||
155 | } | ||
156 | |||
157 | static struct pcie_port *bus_to_port(int bus) | ||
158 | { | ||
159 | int i; | ||
160 | |||
161 | for (i = num_pcie_ports - 1; i >= 0; i--) { | ||
162 | int rbus = pcie_port[i].root_bus_nr; | ||
163 | if (rbus != -1 && rbus <= bus) | ||
164 | break; | ||
165 | } | ||
166 | |||
167 | return i >= 0 ? pcie_port + i : NULL; | ||
168 | } | ||
169 | |||
170 | static int pcie_valid_config(struct pcie_port *pp, int bus, int dev) | ||
171 | { | ||
172 | /* | ||
173 | * Don't go out when trying to access nonexisting devices | ||
174 | * on the local bus. | ||
175 | */ | ||
176 | if (bus == pp->root_bus_nr && dev > 1) | ||
177 | return 0; | ||
178 | |||
179 | return 1; | ||
180 | } | ||
181 | |||
182 | static int pcie_rd_conf(struct pci_bus *bus, u32 devfn, int where, | ||
183 | int size, u32 *val) | ||
184 | { | ||
185 | struct pcie_port *pp = bus_to_port(bus->number); | ||
186 | unsigned long flags; | ||
187 | int ret; | ||
188 | |||
189 | if (pcie_valid_config(pp, bus->number, PCI_SLOT(devfn)) == 0) { | ||
190 | *val = 0xffffffff; | ||
191 | return PCIBIOS_DEVICE_NOT_FOUND; | ||
192 | } | ||
193 | |||
194 | spin_lock_irqsave(&pp->conf_lock, flags); | ||
195 | ret = orion_pcie_rd_conf(pp->base, bus, devfn, where, size, val); | ||
196 | spin_unlock_irqrestore(&pp->conf_lock, flags); | ||
197 | |||
198 | return ret; | ||
199 | } | ||
200 | |||
201 | static int pcie_wr_conf(struct pci_bus *bus, u32 devfn, | ||
202 | int where, int size, u32 val) | ||
203 | { | ||
204 | struct pcie_port *pp = bus_to_port(bus->number); | ||
205 | unsigned long flags; | ||
206 | int ret; | ||
207 | |||
208 | if (pcie_valid_config(pp, bus->number, PCI_SLOT(devfn)) == 0) | ||
209 | return PCIBIOS_DEVICE_NOT_FOUND; | ||
210 | |||
211 | spin_lock_irqsave(&pp->conf_lock, flags); | ||
212 | ret = orion_pcie_wr_conf(pp->base, bus, devfn, where, size, val); | ||
213 | spin_unlock_irqrestore(&pp->conf_lock, flags); | ||
214 | |||
215 | return ret; | ||
216 | } | ||
217 | |||
218 | static struct pci_ops pcie_ops = { | ||
219 | .read = pcie_rd_conf, | ||
220 | .write = pcie_wr_conf, | ||
221 | }; | ||
222 | |||
223 | static void __devinit rc_pci_fixup(struct pci_dev *dev) | ||
224 | { | ||
225 | /* | ||
226 | * Prevent enumeration of root complex. | ||
227 | */ | ||
228 | if (dev->bus->parent == NULL && dev->devfn == 0) { | ||
229 | int i; | ||
230 | |||
231 | for (i = 0; i < DEVICE_COUNT_RESOURCE; i++) { | ||
232 | dev->resource[i].start = 0; | ||
233 | dev->resource[i].end = 0; | ||
234 | dev->resource[i].flags = 0; | ||
235 | } | ||
236 | } | ||
237 | } | ||
238 | DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_MARVELL, PCI_ANY_ID, rc_pci_fixup); | ||
239 | |||
240 | static struct pci_bus __init * | ||
241 | mv78xx0_pcie_scan_bus(int nr, struct pci_sys_data *sys) | ||
242 | { | ||
243 | struct pci_bus *bus; | ||
244 | |||
245 | if (nr < num_pcie_ports) { | ||
246 | bus = pci_scan_bus(sys->busnr, &pcie_ops, sys); | ||
247 | } else { | ||
248 | bus = NULL; | ||
249 | BUG(); | ||
250 | } | ||
251 | |||
252 | return bus; | ||
253 | } | ||
254 | |||
255 | static int __init mv78xx0_pcie_map_irq(struct pci_dev *dev, u8 slot, u8 pin) | ||
256 | { | ||
257 | struct pcie_port *pp = bus_to_port(dev->bus->number); | ||
258 | |||
259 | return IRQ_MV78XX0_PCIE_00 + (pp->maj << 2) + pp->min; | ||
260 | } | ||
261 | |||
262 | static struct hw_pci mv78xx0_pci __initdata = { | ||
263 | .nr_controllers = 8, | ||
264 | .preinit = mv78xx0_pcie_preinit, | ||
265 | .swizzle = pci_std_swizzle, | ||
266 | .setup = mv78xx0_pcie_setup, | ||
267 | .scan = mv78xx0_pcie_scan_bus, | ||
268 | .map_irq = mv78xx0_pcie_map_irq, | ||
269 | }; | ||
270 | |||
271 | static void __init add_pcie_port(int maj, int min, unsigned long base) | ||
272 | { | ||
273 | printk(KERN_INFO "MV78xx0 PCIe port %d.%d: ", maj, min); | ||
274 | |||
275 | if (orion_pcie_link_up((void __iomem *)base)) { | ||
276 | struct pcie_port *pp = &pcie_port[num_pcie_ports++]; | ||
277 | |||
278 | printk("link up\n"); | ||
279 | |||
280 | pp->maj = maj; | ||
281 | pp->min = min; | ||
282 | pp->root_bus_nr = -1; | ||
283 | pp->base = (void __iomem *)base; | ||
284 | spin_lock_init(&pp->conf_lock); | ||
285 | memset(pp->res, 0, sizeof(pp->res)); | ||
286 | } else { | ||
287 | printk("link down, ignoring\n"); | ||
288 | } | ||
289 | } | ||
290 | |||
291 | void __init mv78xx0_pcie_init(int init_port0, int init_port1) | ||
292 | { | ||
293 | if (init_port0) { | ||
294 | add_pcie_port(0, 0, PCIE00_VIRT_BASE); | ||
295 | if (!orion_pcie_x4_mode((void __iomem *)PCIE00_VIRT_BASE)) { | ||
296 | add_pcie_port(0, 1, PCIE01_VIRT_BASE); | ||
297 | add_pcie_port(0, 2, PCIE02_VIRT_BASE); | ||
298 | add_pcie_port(0, 3, PCIE03_VIRT_BASE); | ||
299 | } | ||
300 | } | ||
301 | |||
302 | if (init_port1) { | ||
303 | add_pcie_port(1, 0, PCIE10_VIRT_BASE); | ||
304 | if (!orion_pcie_x4_mode((void __iomem *)PCIE10_VIRT_BASE)) { | ||
305 | add_pcie_port(1, 1, PCIE11_VIRT_BASE); | ||
306 | add_pcie_port(1, 2, PCIE12_VIRT_BASE); | ||
307 | add_pcie_port(1, 3, PCIE13_VIRT_BASE); | ||
308 | } | ||
309 | } | ||
310 | |||
311 | pci_common_init(&mv78xx0_pci); | ||
312 | } | ||
diff --git a/arch/arm/mm/Kconfig b/arch/arm/mm/Kconfig index a7a6efec8974..236603bbafdc 100644 --- a/arch/arm/mm/Kconfig +++ b/arch/arm/mm/Kconfig | |||
@@ -365,7 +365,7 @@ config CPU_XSC3 | |||
365 | # Feroceon | 365 | # Feroceon |
366 | config CPU_FEROCEON | 366 | config CPU_FEROCEON |
367 | bool | 367 | bool |
368 | depends on ARCH_ORION5X || ARCH_LOKI || ARCH_KIRKWOOD | 368 | depends on ARCH_ORION5X || ARCH_LOKI || ARCH_KIRKWOOD || ARCH_MV78XX0 |
369 | default y | 369 | default y |
370 | select CPU_32v5 | 370 | select CPU_32v5 |
371 | select CPU_ABRT_EV5T | 371 | select CPU_ABRT_EV5T |
@@ -716,7 +716,7 @@ config OUTER_CACHE | |||
716 | 716 | ||
717 | config CACHE_FEROCEON_L2 | 717 | config CACHE_FEROCEON_L2 |
718 | bool "Enable the Feroceon L2 cache controller" | 718 | bool "Enable the Feroceon L2 cache controller" |
719 | depends on ARCH_KIRKWOOD | 719 | depends on ARCH_KIRKWOOD || ARCH_MV78XX0 |
720 | default y | 720 | default y |
721 | select OUTER_CACHE | 721 | select OUTER_CACHE |
722 | help | 722 | help |
diff --git a/include/asm-arm/arch-mv78xx0/debug-macro.S b/include/asm-arm/arch-mv78xx0/debug-macro.S new file mode 100644 index 000000000000..d0595bd645e5 --- /dev/null +++ b/include/asm-arm/arch-mv78xx0/debug-macro.S | |||
@@ -0,0 +1,20 @@ | |||
1 | /* | ||
2 | * include/asm-arm/arch-mv78xx0/debug-macro.S | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | */ | ||
8 | |||
9 | #include <asm/arch/mv78xx0.h> | ||
10 | |||
11 | .macro addruart,rx | ||
12 | mrc p15, 0, \rx, c1, c0 | ||
13 | tst \rx, #1 @ MMU enabled? | ||
14 | ldreq \rx, =MV78XX0_REGS_PHYS_BASE | ||
15 | ldrne \rx, =MV78XX0_REGS_VIRT_BASE | ||
16 | orr \rx, \rx, #0x00012000 | ||
17 | .endm | ||
18 | |||
19 | #define UART_SHIFT 2 | ||
20 | #include <asm/hardware/debug-8250.S> | ||
diff --git a/include/asm-arm/arch-mv78xx0/dma.h b/include/asm-arm/arch-mv78xx0/dma.h new file mode 100644 index 000000000000..40a8c178f10d --- /dev/null +++ b/include/asm-arm/arch-mv78xx0/dma.h | |||
@@ -0,0 +1 @@ | |||
/* empty */ | |||
diff --git a/include/asm-arm/arch-mv78xx0/entry-macro.S b/include/asm-arm/arch-mv78xx0/entry-macro.S new file mode 100644 index 000000000000..e9a606b12669 --- /dev/null +++ b/include/asm-arm/arch-mv78xx0/entry-macro.S | |||
@@ -0,0 +1,39 @@ | |||
1 | /* | ||
2 | * include/asm-arm/arch-mv78xx0/entry-macro.S | ||
3 | * | ||
4 | * Low-level IRQ helper macros for Marvell MV78xx0 platforms | ||
5 | * | ||
6 | * This file is licensed under the terms of the GNU General Public | ||
7 | * License version 2. This program is licensed "as is" without any | ||
8 | * warranty of any kind, whether express or implied. | ||
9 | */ | ||
10 | |||
11 | #include <asm/arch/mv78xx0.h> | ||
12 | |||
13 | .macro disable_fiq | ||
14 | .endm | ||
15 | |||
16 | .macro arch_ret_to_user, tmp1, tmp2 | ||
17 | .endm | ||
18 | |||
19 | .macro get_irqnr_preamble, base, tmp | ||
20 | ldr \base, =IRQ_VIRT_BASE | ||
21 | .endm | ||
22 | |||
23 | .macro get_irqnr_and_base, irqnr, irqstat, base, tmp | ||
24 | @ check low interrupts | ||
25 | ldr \irqstat, [\base, #IRQ_CAUSE_LOW_OFF] | ||
26 | ldr \tmp, [\base, #IRQ_MASK_LOW_OFF] | ||
27 | mov \irqnr, #31 | ||
28 | ands \irqstat, \irqstat, \tmp | ||
29 | |||
30 | @ if no low interrupts set, check high interrupts | ||
31 | ldreq \irqstat, [\base, #IRQ_CAUSE_HIGH_OFF] | ||
32 | ldreq \tmp, [\base, #IRQ_MASK_HIGH_OFF] | ||
33 | moveq \irqnr, #63 | ||
34 | andeqs \irqstat, \irqstat, \tmp | ||
35 | |||
36 | @ find first active interrupt source | ||
37 | clzne \irqstat, \irqstat | ||
38 | subne \irqnr, \irqnr, \irqstat | ||
39 | .endm | ||
diff --git a/include/asm-arm/arch-mv78xx0/hardware.h b/include/asm-arm/arch-mv78xx0/hardware.h new file mode 100644 index 000000000000..8e17926086c6 --- /dev/null +++ b/include/asm-arm/arch-mv78xx0/hardware.h | |||
@@ -0,0 +1,21 @@ | |||
1 | /* | ||
2 | * include/asm-arm/arch-mv78xx0/hardware.h | ||
3 | * | ||
4 | * This file is licensed under the terms of the GNU General Public | ||
5 | * License version 2. This program is licensed "as is" without any | ||
6 | * warranty of any kind, whether express or implied. | ||
7 | */ | ||
8 | |||
9 | #ifndef __ASM_ARCH_HARDWARE_H | ||
10 | #define __ASM_ARCH_HARDWARE_H | ||
11 | |||
12 | #include "mv78xx0.h" | ||
13 | |||
14 | #define pcibios_assign_all_busses() 1 | ||
15 | |||
16 | #define PCIBIOS_MIN_IO 0x00001000 | ||
17 | #define PCIBIOS_MIN_MEM 0x01000000 | ||
18 | #define PCIMEM_BASE MV78XX0_PCIE_MEM_PHYS_BASE /* mem base for VGA */ | ||
19 | |||
20 | |||
21 | #endif | ||
diff --git a/include/asm-arm/arch-mv78xx0/io.h b/include/asm-arm/arch-mv78xx0/io.h new file mode 100644 index 000000000000..415d4c98e3d1 --- /dev/null +++ b/include/asm-arm/arch-mv78xx0/io.h | |||
@@ -0,0 +1,26 @@ | |||
1 | /* | ||
2 | * include/asm-arm/arch-mv78xx0/io.h | ||
3 | * | ||
4 | * This file is licensed under the terms of the GNU General Public | ||
5 | * License version 2. This program is licensed "as is" without any | ||
6 | * warranty of any kind, whether express or implied. | ||
7 | */ | ||
8 | |||
9 | #ifndef __ASM_ARCH_IO_H | ||
10 | #define __ASM_ARCH_IO_H | ||
11 | |||
12 | #include "mv78xx0.h" | ||
13 | |||
14 | #define IO_SPACE_LIMIT 0xffffffff | ||
15 | |||
16 | static inline void __iomem *__io(unsigned long addr) | ||
17 | { | ||
18 | return (void __iomem *)((addr - MV78XX0_PCIE_IO_PHYS_BASE(0)) | ||
19 | + MV78XX0_PCIE_IO_VIRT_BASE(0)); | ||
20 | } | ||
21 | |||
22 | #define __io(a) __io(a) | ||
23 | #define __mem_pci(a) (a) | ||
24 | |||
25 | |||
26 | #endif | ||
diff --git a/include/asm-arm/arch-mv78xx0/irqs.h b/include/asm-arm/arch-mv78xx0/irqs.h new file mode 100644 index 000000000000..75930450cd65 --- /dev/null +++ b/include/asm-arm/arch-mv78xx0/irqs.h | |||
@@ -0,0 +1,91 @@ | |||
1 | /* | ||
2 | * include/asm-arm/arch-mv78xx0/irqs.h | ||
3 | * | ||
4 | * IRQ definitions for Marvell MV78xx0 SoCs | ||
5 | * | ||
6 | * This file is licensed under the terms of the GNU General Public | ||
7 | * License version 2. This program is licensed "as is" without any | ||
8 | * warranty of any kind, whether express or implied. | ||
9 | */ | ||
10 | |||
11 | #ifndef __ASM_ARCH_IRQS_H | ||
12 | #define __ASM_ARCH_IRQS_H | ||
13 | |||
14 | #include "mv78xx0.h" /* need GPIO_MAX */ | ||
15 | |||
16 | /* | ||
17 | * MV78xx0 Low Interrupt Controller | ||
18 | */ | ||
19 | #define IRQ_MV78XX0_ERR 0 | ||
20 | #define IRQ_MV78XX0_SPI 1 | ||
21 | #define IRQ_MV78XX0_I2C_0 2 | ||
22 | #define IRQ_MV78XX0_I2C_1 3 | ||
23 | #define IRQ_MV78XX0_IDMA_0 4 | ||
24 | #define IRQ_MV78XX0_IDMA_1 5 | ||
25 | #define IRQ_MV78XX0_IDMA_2 6 | ||
26 | #define IRQ_MV78XX0_IDMA_3 7 | ||
27 | #define IRQ_MV78XX0_TIMER_0 8 | ||
28 | #define IRQ_MV78XX0_TIMER_1 9 | ||
29 | #define IRQ_MV78XX0_TIMER_2 10 | ||
30 | #define IRQ_MV78XX0_TIMER_3 11 | ||
31 | #define IRQ_MV78XX0_UART_0 12 | ||
32 | #define IRQ_MV78XX0_UART_1 13 | ||
33 | #define IRQ_MV78XX0_UART_2 14 | ||
34 | #define IRQ_MV78XX0_UART_3 15 | ||
35 | #define IRQ_MV78XX0_USB_0 16 | ||
36 | #define IRQ_MV78XX0_USB_1 17 | ||
37 | #define IRQ_MV78XX0_USB_2 18 | ||
38 | #define IRQ_MV78XX0_CRYPTO 19 | ||
39 | #define IRQ_MV78XX0_SDIO_0 20 | ||
40 | #define IRQ_MV78XX0_SDIO_1 21 | ||
41 | #define IRQ_MV78XX0_XOR_0 22 | ||
42 | #define IRQ_MV78XX0_XOR_1 23 | ||
43 | #define IRQ_MV78XX0_I2S_0 24 | ||
44 | #define IRQ_MV78XX0_I2S_1 25 | ||
45 | #define IRQ_MV78XX0_SATA 26 | ||
46 | #define IRQ_MV78XX0_TDMI 27 | ||
47 | |||
48 | /* | ||
49 | * MV78xx0 High Interrupt Controller | ||
50 | */ | ||
51 | #define IRQ_MV78XX0_PCIE_00 32 | ||
52 | #define IRQ_MV78XX0_PCIE_01 33 | ||
53 | #define IRQ_MV78XX0_PCIE_02 34 | ||
54 | #define IRQ_MV78XX0_PCIE_03 35 | ||
55 | #define IRQ_MV78XX0_PCIE_10 36 | ||
56 | #define IRQ_MV78XX0_PCIE_11 37 | ||
57 | #define IRQ_MV78XX0_PCIE_12 38 | ||
58 | #define IRQ_MV78XX0_PCIE_13 39 | ||
59 | #define IRQ_MV78XX0_GE00_SUM 40 | ||
60 | #define IRQ_MV78XX0_GE00_RX 41 | ||
61 | #define IRQ_MV78XX0_GE00_TX 42 | ||
62 | #define IRQ_MV78XX0_GE00_MISC 43 | ||
63 | #define IRQ_MV78XX0_GE01_SUM 44 | ||
64 | #define IRQ_MV78XX0_GE01_RX 45 | ||
65 | #define IRQ_MV78XX0_GE01_TX 46 | ||
66 | #define IRQ_MV78XX0_GE01_MISC 47 | ||
67 | #define IRQ_MV78XX0_GE10_SUM 48 | ||
68 | #define IRQ_MV78XX0_GE10_RX 49 | ||
69 | #define IRQ_MV78XX0_GE10_TX 50 | ||
70 | #define IRQ_MV78XX0_GE10_MISC 51 | ||
71 | #define IRQ_MV78XX0_GE11_SUM 52 | ||
72 | #define IRQ_MV78XX0_GE11_RX 53 | ||
73 | #define IRQ_MV78XX0_GE11_TX 54 | ||
74 | #define IRQ_MV78XX0_GE11_MISC 55 | ||
75 | #define IRQ_MV78XX0_GPIO_0_7 56 | ||
76 | #define IRQ_MV78XX0_GPIO_8_15 57 | ||
77 | #define IRQ_MV78XX0_GPIO_16_23 58 | ||
78 | #define IRQ_MV78XX0_GPIO_24_31 59 | ||
79 | #define IRQ_MV78XX0_DB_IN 60 | ||
80 | #define IRQ_MV78XX0_DB_OUT 61 | ||
81 | |||
82 | /* | ||
83 | * MV78XX0 General Purpose Pins | ||
84 | */ | ||
85 | #define IRQ_MV78XX0_GPIO_START 64 | ||
86 | #define NR_GPIO_IRQS GPIO_MAX | ||
87 | |||
88 | #define NR_IRQS (IRQ_MV78XX0_GPIO_START + NR_GPIO_IRQS) | ||
89 | |||
90 | |||
91 | #endif | ||
diff --git a/include/asm-arm/arch-mv78xx0/memory.h b/include/asm-arm/arch-mv78xx0/memory.h new file mode 100644 index 000000000000..721a6b185b91 --- /dev/null +++ b/include/asm-arm/arch-mv78xx0/memory.h | |||
@@ -0,0 +1,14 @@ | |||
1 | /* | ||
2 | * include/asm-arm/arch-mv78xx0/memory.h | ||
3 | */ | ||
4 | |||
5 | #ifndef __ASM_ARCH_MEMORY_H | ||
6 | #define __ASM_ARCH_MEMORY_H | ||
7 | |||
8 | #define PHYS_OFFSET UL(0x00000000) | ||
9 | |||
10 | #define __virt_to_bus(x) __virt_to_phys(x) | ||
11 | #define __bus_to_virt(x) __phys_to_virt(x) | ||
12 | |||
13 | |||
14 | #endif | ||
diff --git a/include/asm-arm/arch-mv78xx0/mv78xx0.h b/include/asm-arm/arch-mv78xx0/mv78xx0.h new file mode 100644 index 000000000000..9f5d83c73faa --- /dev/null +++ b/include/asm-arm/arch-mv78xx0/mv78xx0.h | |||
@@ -0,0 +1,126 @@ | |||
1 | /* | ||
2 | * include/asm-arm/arch-mv78xx0/mv78xx0.h | ||
3 | * | ||
4 | * Generic definitions for Marvell MV78xx0 SoC flavors: | ||
5 | * MV781x0 and MV782x0. | ||
6 | * | ||
7 | * This file is licensed under the terms of the GNU General Public | ||
8 | * License version 2. This program is licensed "as is" without any | ||
9 | * warranty of any kind, whether express or implied. | ||
10 | */ | ||
11 | |||
12 | #ifndef __ASM_ARCH_MV78XX0_H | ||
13 | #define __ASM_ARCH_MV78XX0_H | ||
14 | |||
15 | /* | ||
16 | * Marvell MV78xx0 address maps. | ||
17 | * | ||
18 | * phys | ||
19 | * c0000000 PCIe Memory space | ||
20 | * f0800000 PCIe #0 I/O space | ||
21 | * f0900000 PCIe #1 I/O space | ||
22 | * f0a00000 PCIe #2 I/O space | ||
23 | * f0b00000 PCIe #3 I/O space | ||
24 | * f0c00000 PCIe #4 I/O space | ||
25 | * f0d00000 PCIe #5 I/O space | ||
26 | * f0e00000 PCIe #6 I/O space | ||
27 | * f0f00000 PCIe #7 I/O space | ||
28 | * f1000000 on-chip peripheral registers | ||
29 | * | ||
30 | * virt phys size | ||
31 | * fe400000 f102x000 16K core-specific peripheral registers | ||
32 | * fe700000 f0800000 1M PCIe #0 I/O space | ||
33 | * fe800000 f0900000 1M PCIe #1 I/O space | ||
34 | * fe900000 f0a00000 1M PCIe #2 I/O space | ||
35 | * fea00000 f0b00000 1M PCIe #3 I/O space | ||
36 | * feb00000 f0c00000 1M PCIe #4 I/O space | ||
37 | * fec00000 f0d00000 1M PCIe #5 I/O space | ||
38 | * fed00000 f0e00000 1M PCIe #6 I/O space | ||
39 | * fee00000 f0f00000 1M PCIe #7 I/O space | ||
40 | * fef00000 f1000000 1M on-chip peripheral registers | ||
41 | */ | ||
42 | #define MV78XX0_CORE0_REGS_PHYS_BASE 0xf1020000 | ||
43 | #define MV78XX0_CORE1_REGS_PHYS_BASE 0xf1024000 | ||
44 | #define MV78XX0_CORE_REGS_VIRT_BASE 0xfe400000 | ||
45 | #define MV78XX0_CORE_REGS_SIZE SZ_16K | ||
46 | |||
47 | #define MV78XX0_PCIE_IO_PHYS_BASE(i) (0xf0800000 + ((i) << 20)) | ||
48 | #define MV78XX0_PCIE_IO_VIRT_BASE(i) (0xfe700000 + ((i) << 20)) | ||
49 | #define MV78XX0_PCIE_IO_SIZE SZ_1M | ||
50 | |||
51 | #define MV78XX0_REGS_PHYS_BASE 0xf1000000 | ||
52 | #define MV78XX0_REGS_VIRT_BASE 0xfef00000 | ||
53 | #define MV78XX0_REGS_SIZE SZ_1M | ||
54 | |||
55 | #define MV78XX0_PCIE_MEM_PHYS_BASE 0xc0000000 | ||
56 | #define MV78XX0_PCIE_MEM_SIZE 0x30000000 | ||
57 | |||
58 | /* | ||
59 | * Core-specific peripheral registers. | ||
60 | */ | ||
61 | #define BRIDGE_VIRT_BASE (MV78XX0_CORE_REGS_VIRT_BASE) | ||
62 | #define CPU_CONTROL (BRIDGE_VIRT_BASE | 0x0104) | ||
63 | #define L2_WRITETHROUGH 0x00020000 | ||
64 | #define RSTOUTn_MASK (BRIDGE_VIRT_BASE | 0x0108) | ||
65 | #define SOFT_RESET_OUT_EN 0x00000004 | ||
66 | #define SYSTEM_SOFT_RESET (BRIDGE_VIRT_BASE | 0x010c) | ||
67 | #define SOFT_RESET 0x00000001 | ||
68 | #define BRIDGE_CAUSE (BRIDGE_VIRT_BASE | 0x0110) | ||
69 | #define BRIDGE_MASK (BRIDGE_VIRT_BASE | 0x0114) | ||
70 | #define BRIDGE_INT_TIMER0 0x0002 | ||
71 | #define BRIDGE_INT_TIMER1 0x0004 | ||
72 | #define BRIDGE_INT_TIMER1_CLR (~0x0004) | ||
73 | #define IRQ_VIRT_BASE (BRIDGE_VIRT_BASE | 0x0200) | ||
74 | #define IRQ_CAUSE_LOW_OFF 0x0004 | ||
75 | #define IRQ_CAUSE_HIGH_OFF 0x0008 | ||
76 | #define IRQ_MASK_LOW_OFF 0x0010 | ||
77 | #define IRQ_MASK_HIGH_OFF 0x0014 | ||
78 | #define TIMER_VIRT_BASE (BRIDGE_VIRT_BASE | 0x0300) | ||
79 | |||
80 | /* | ||
81 | * Register Map | ||
82 | */ | ||
83 | #define DDR_VIRT_BASE (MV78XX0_REGS_VIRT_BASE | 0x00000) | ||
84 | #define DDR_WINDOW_CPU0_BASE (DDR_VIRT_BASE | 0x1500) | ||
85 | #define DDR_WINDOW_CPU1_BASE (DDR_VIRT_BASE | 0x1700) | ||
86 | |||
87 | #define DEV_BUS_PHYS_BASE (MV78XX0_REGS_PHYS_BASE | 0x10000) | ||
88 | #define DEV_BUS_VIRT_BASE (MV78XX0_REGS_VIRT_BASE | 0x10000) | ||
89 | #define SAMPLE_AT_RESET_LOW (DEV_BUS_VIRT_BASE | 0x0030) | ||
90 | #define SAMPLE_AT_RESET_HIGH (DEV_BUS_VIRT_BASE | 0x0034) | ||
91 | #define UART0_PHYS_BASE (DEV_BUS_PHYS_BASE | 0x2000) | ||
92 | #define UART0_VIRT_BASE (DEV_BUS_VIRT_BASE | 0x2000) | ||
93 | #define UART1_PHYS_BASE (DEV_BUS_PHYS_BASE | 0x2100) | ||
94 | #define UART1_VIRT_BASE (DEV_BUS_VIRT_BASE | 0x2100) | ||
95 | #define UART2_PHYS_BASE (DEV_BUS_PHYS_BASE | 0x2200) | ||
96 | #define UART2_VIRT_BASE (DEV_BUS_VIRT_BASE | 0x2200) | ||
97 | #define UART3_PHYS_BASE (DEV_BUS_PHYS_BASE | 0x2300) | ||
98 | #define UART3_VIRT_BASE (DEV_BUS_VIRT_BASE | 0x2300) | ||
99 | |||
100 | #define GE10_PHYS_BASE (MV78XX0_REGS_PHYS_BASE | 0x30000) | ||
101 | #define GE11_PHYS_BASE (MV78XX0_REGS_PHYS_BASE | 0x34000) | ||
102 | |||
103 | #define PCIE00_VIRT_BASE (MV78XX0_REGS_VIRT_BASE | 0x40000) | ||
104 | #define PCIE01_VIRT_BASE (MV78XX0_REGS_VIRT_BASE | 0x44000) | ||
105 | #define PCIE02_VIRT_BASE (MV78XX0_REGS_VIRT_BASE | 0x48000) | ||
106 | #define PCIE03_VIRT_BASE (MV78XX0_REGS_VIRT_BASE | 0x4c000) | ||
107 | |||
108 | #define USB0_PHYS_BASE (MV78XX0_REGS_PHYS_BASE | 0x50000) | ||
109 | #define USB1_PHYS_BASE (MV78XX0_REGS_PHYS_BASE | 0x51000) | ||
110 | #define USB2_PHYS_BASE (MV78XX0_REGS_PHYS_BASE | 0x52000) | ||
111 | |||
112 | #define GE00_PHYS_BASE (MV78XX0_REGS_PHYS_BASE | 0x70000) | ||
113 | #define GE01_PHYS_BASE (MV78XX0_REGS_PHYS_BASE | 0x74000) | ||
114 | |||
115 | #define PCIE10_VIRT_BASE (MV78XX0_REGS_VIRT_BASE | 0x80000) | ||
116 | #define PCIE11_VIRT_BASE (MV78XX0_REGS_VIRT_BASE | 0x84000) | ||
117 | #define PCIE12_VIRT_BASE (MV78XX0_REGS_VIRT_BASE | 0x88000) | ||
118 | #define PCIE13_VIRT_BASE (MV78XX0_REGS_VIRT_BASE | 0x8c000) | ||
119 | |||
120 | #define SATA_PHYS_BASE (MV78XX0_REGS_PHYS_BASE | 0xa0000) | ||
121 | |||
122 | |||
123 | #define GPIO_MAX 32 | ||
124 | |||
125 | |||
126 | #endif | ||
diff --git a/include/asm-arm/arch-mv78xx0/system.h b/include/asm-arm/arch-mv78xx0/system.h new file mode 100644 index 000000000000..7eb47d376db9 --- /dev/null +++ b/include/asm-arm/arch-mv78xx0/system.h | |||
@@ -0,0 +1,37 @@ | |||
1 | /* | ||
2 | * include/asm-arm/arch-mv78xx0/system.h | ||
3 | * | ||
4 | * This file is licensed under the terms of the GNU General Public | ||
5 | * License version 2. This program is licensed "as is" without any | ||
6 | * warranty of any kind, whether express or implied. | ||
7 | */ | ||
8 | |||
9 | #ifndef __ASM_ARCH_SYSTEM_H | ||
10 | #define __ASM_ARCH_SYSTEM_H | ||
11 | |||
12 | #include <asm/arch/hardware.h> | ||
13 | #include <asm/arch/mv78xx0.h> | ||
14 | |||
15 | static inline void arch_idle(void) | ||
16 | { | ||
17 | cpu_do_idle(); | ||
18 | } | ||
19 | |||
20 | static inline void arch_reset(char mode) | ||
21 | { | ||
22 | /* | ||
23 | * Enable soft reset to assert RSTOUTn. | ||
24 | */ | ||
25 | writel(SOFT_RESET_OUT_EN, RSTOUTn_MASK); | ||
26 | |||
27 | /* | ||
28 | * Assert soft reset. | ||
29 | */ | ||
30 | writel(SOFT_RESET, SYSTEM_SOFT_RESET); | ||
31 | |||
32 | while (1) | ||
33 | ; | ||
34 | } | ||
35 | |||
36 | |||
37 | #endif | ||
diff --git a/include/asm-arm/arch-mv78xx0/timex.h b/include/asm-arm/arch-mv78xx0/timex.h new file mode 100644 index 000000000000..a854b1ccbd01 --- /dev/null +++ b/include/asm-arm/arch-mv78xx0/timex.h | |||
@@ -0,0 +1,9 @@ | |||
1 | /* | ||
2 | * include/asm-arm/arch-mv78xx0/timex.h | ||
3 | * | ||
4 | * This file is licensed under the terms of the GNU General Public | ||
5 | * License version 2. This program is licensed "as is" without any | ||
6 | * warranty of any kind, whether express or implied. | ||
7 | */ | ||
8 | |||
9 | #define CLOCK_TICK_RATE (100 * HZ) | ||
diff --git a/include/asm-arm/arch-mv78xx0/uncompress.h b/include/asm-arm/arch-mv78xx0/uncompress.h new file mode 100644 index 000000000000..3bfe0a293ef7 --- /dev/null +++ b/include/asm-arm/arch-mv78xx0/uncompress.h | |||
@@ -0,0 +1,47 @@ | |||
1 | /* | ||
2 | * include/asm-arm/arch-mv78xx0/uncompress.h | ||
3 | * | ||
4 | * This file is licensed under the terms of the GNU General Public | ||
5 | * License version 2. This program is licensed "as is" without any | ||
6 | * warranty of any kind, whether express or implied. | ||
7 | */ | ||
8 | |||
9 | #include <linux/serial_reg.h> | ||
10 | #include <asm/arch/mv78xx0.h> | ||
11 | |||
12 | #define SERIAL_BASE ((unsigned char *)UART0_PHYS_BASE) | ||
13 | |||
14 | static void putc(const char c) | ||
15 | { | ||
16 | unsigned char *base = SERIAL_BASE; | ||
17 | int i; | ||
18 | |||
19 | for (i = 0; i < 0x1000; i++) { | ||
20 | if (base[UART_LSR << 2] & UART_LSR_THRE) | ||
21 | break; | ||
22 | barrier(); | ||
23 | } | ||
24 | |||
25 | base[UART_TX << 2] = c; | ||
26 | } | ||
27 | |||
28 | static void flush(void) | ||
29 | { | ||
30 | unsigned char *base = SERIAL_BASE; | ||
31 | unsigned char mask; | ||
32 | int i; | ||
33 | |||
34 | mask = UART_LSR_TEMT | UART_LSR_THRE; | ||
35 | |||
36 | for (i = 0; i < 0x1000; i++) { | ||
37 | if ((base[UART_LSR << 2] & mask) == mask) | ||
38 | break; | ||
39 | barrier(); | ||
40 | } | ||
41 | } | ||
42 | |||
43 | /* | ||
44 | * nothing to do | ||
45 | */ | ||
46 | #define arch_decomp_setup() | ||
47 | #define arch_decomp_wdog() | ||
diff --git a/include/asm-arm/arch-mv78xx0/vmalloc.h b/include/asm-arm/arch-mv78xx0/vmalloc.h new file mode 100644 index 000000000000..f2c512197579 --- /dev/null +++ b/include/asm-arm/arch-mv78xx0/vmalloc.h | |||
@@ -0,0 +1,5 @@ | |||
1 | /* | ||
2 | * include/asm-arm/arch-mv78xx0/vmalloc.h | ||
3 | */ | ||
4 | |||
5 | #define VMALLOC_END 0xfe000000 | ||