diff options
Diffstat (limited to 'arch/arm/mach-mvebu')
-rw-r--r-- | arch/arm/mach-mvebu/Kconfig | 6 | ||||
-rw-r--r-- | arch/arm/mach-mvebu/Makefile | 2 | ||||
-rw-r--r-- | arch/arm/mach-mvebu/armada-370-xp.h | 1 | ||||
-rw-r--r-- | arch/arm/mach-mvebu/board-v7.c | 9 | ||||
-rw-r--r-- | arch/arm/mach-mvebu/common.h | 2 | ||||
-rw-r--r-- | arch/arm/mach-mvebu/headsmp-a9.S | 15 | ||||
-rw-r--r-- | arch/arm/mach-mvebu/platsmp-a9.c | 42 | ||||
-rw-r--r-- | arch/arm/mach-mvebu/platsmp.c | 3 | ||||
-rw-r--r-- | arch/arm/mach-mvebu/pmsu.c | 435 | ||||
-rw-r--r-- | arch/arm/mach-mvebu/pmsu.h | 5 | ||||
-rw-r--r-- | arch/arm/mach-mvebu/pmsu_ll.S | 36 | ||||
-rw-r--r-- | arch/arm/mach-mvebu/system-controller.c | 31 |
12 files changed, 492 insertions, 95 deletions
diff --git a/arch/arm/mach-mvebu/Kconfig b/arch/arm/mach-mvebu/Kconfig index 955d4a3afabd..c1e4567a5ab3 100644 --- a/arch/arm/mach-mvebu/Kconfig +++ b/arch/arm/mach-mvebu/Kconfig | |||
@@ -14,11 +14,15 @@ menuconfig ARCH_MVEBU | |||
14 | 14 | ||
15 | if ARCH_MVEBU | 15 | if ARCH_MVEBU |
16 | 16 | ||
17 | config MACH_MVEBU_ANY | ||
18 | bool | ||
19 | |||
17 | config MACH_MVEBU_V7 | 20 | config MACH_MVEBU_V7 |
18 | bool | 21 | bool |
19 | select ARMADA_370_XP_TIMER | 22 | select ARMADA_370_XP_TIMER |
20 | select CACHE_L2X0 | 23 | select CACHE_L2X0 |
21 | select ARM_CPU_SUSPEND | 24 | select ARM_CPU_SUSPEND |
25 | select MACH_MVEBU_ANY | ||
22 | 26 | ||
23 | config MACH_ARMADA_370 | 27 | config MACH_ARMADA_370 |
24 | bool "Marvell Armada 370 boards" if ARCH_MULTI_V7 | 28 | bool "Marvell Armada 370 boards" if ARCH_MULTI_V7 |
@@ -75,6 +79,7 @@ config MACH_DOVE | |||
75 | select CACHE_L2X0 | 79 | select CACHE_L2X0 |
76 | select CPU_PJ4 | 80 | select CPU_PJ4 |
77 | select DOVE_CLK | 81 | select DOVE_CLK |
82 | select MACH_MVEBU_ANY | ||
78 | select ORION_IRQCHIP | 83 | select ORION_IRQCHIP |
79 | select ORION_TIMER | 84 | select ORION_TIMER |
80 | select PINCTRL_DOVE | 85 | select PINCTRL_DOVE |
@@ -87,6 +92,7 @@ config MACH_KIRKWOOD | |||
87 | select ARCH_REQUIRE_GPIOLIB | 92 | select ARCH_REQUIRE_GPIOLIB |
88 | select CPU_FEROCEON | 93 | select CPU_FEROCEON |
89 | select KIRKWOOD_CLK | 94 | select KIRKWOOD_CLK |
95 | select MACH_MVEBU_ANY | ||
90 | select ORION_IRQCHIP | 96 | select ORION_IRQCHIP |
91 | select ORION_TIMER | 97 | select ORION_TIMER |
92 | select PCI | 98 | select PCI |
diff --git a/arch/arm/mach-mvebu/Makefile b/arch/arm/mach-mvebu/Makefile index bc7689e530a4..e24136b42765 100644 --- a/arch/arm/mach-mvebu/Makefile +++ b/arch/arm/mach-mvebu/Makefile | |||
@@ -4,7 +4,7 @@ ccflags-$(CONFIG_ARCH_MULTIPLATFORM) := -I$(srctree)/$(src)/include \ | |||
4 | AFLAGS_coherency_ll.o := -Wa,-march=armv7-a | 4 | AFLAGS_coherency_ll.o := -Wa,-march=armv7-a |
5 | CFLAGS_pmsu.o := -march=armv7-a | 5 | CFLAGS_pmsu.o := -march=armv7-a |
6 | 6 | ||
7 | obj-y += system-controller.o mvebu-soc-id.o | 7 | obj-$(CONFIG_MACH_MVEBU_ANY) += system-controller.o mvebu-soc-id.o |
8 | 8 | ||
9 | ifeq ($(CONFIG_MACH_MVEBU_V7),y) | 9 | ifeq ($(CONFIG_MACH_MVEBU_V7),y) |
10 | obj-y += cpu-reset.o board-v7.o coherency.o coherency_ll.o pmsu.o pmsu_ll.o | 10 | obj-y += cpu-reset.o board-v7.o coherency.o coherency_ll.o pmsu.o pmsu_ll.o |
diff --git a/arch/arm/mach-mvebu/armada-370-xp.h b/arch/arm/mach-mvebu/armada-370-xp.h index 52c1603a4f92..84cd90d9b860 100644 --- a/arch/arm/mach-mvebu/armada-370-xp.h +++ b/arch/arm/mach-mvebu/armada-370-xp.h | |||
@@ -25,6 +25,5 @@ extern struct smp_operations armada_xp_smp_ops; | |||
25 | #endif | 25 | #endif |
26 | 26 | ||
27 | int armada_370_xp_pmsu_idle_enter(unsigned long deepidle); | 27 | int armada_370_xp_pmsu_idle_enter(unsigned long deepidle); |
28 | void armada_370_xp_pmsu_idle_exit(void); | ||
29 | 28 | ||
30 | #endif /* __MACH_ARMADA_370_XP_H */ | 29 | #endif /* __MACH_ARMADA_370_XP_H */ |
diff --git a/arch/arm/mach-mvebu/board-v7.c b/arch/arm/mach-mvebu/board-v7.c index f244622ffc00..6478626e3ff6 100644 --- a/arch/arm/mach-mvebu/board-v7.c +++ b/arch/arm/mach-mvebu/board-v7.c | |||
@@ -34,14 +34,14 @@ | |||
34 | #include "coherency.h" | 34 | #include "coherency.h" |
35 | #include "mvebu-soc-id.h" | 35 | #include "mvebu-soc-id.h" |
36 | 36 | ||
37 | static void __iomem *scu_base; | ||
38 | |||
37 | /* | 39 | /* |
38 | * Enables the SCU when available. Obviously, this is only useful on | 40 | * Enables the SCU when available. Obviously, this is only useful on |
39 | * Cortex-A based SOCs, not on PJ4B based ones. | 41 | * Cortex-A based SOCs, not on PJ4B based ones. |
40 | */ | 42 | */ |
41 | static void __init mvebu_scu_enable(void) | 43 | static void __init mvebu_scu_enable(void) |
42 | { | 44 | { |
43 | void __iomem *scu_base; | ||
44 | |||
45 | struct device_node *np = | 45 | struct device_node *np = |
46 | of_find_compatible_node(NULL, NULL, "arm,cortex-a9-scu"); | 46 | of_find_compatible_node(NULL, NULL, "arm,cortex-a9-scu"); |
47 | if (np) { | 47 | if (np) { |
@@ -51,6 +51,11 @@ static void __init mvebu_scu_enable(void) | |||
51 | } | 51 | } |
52 | } | 52 | } |
53 | 53 | ||
54 | void __iomem *mvebu_get_scu_base(void) | ||
55 | { | ||
56 | return scu_base; | ||
57 | } | ||
58 | |||
54 | /* | 59 | /* |
55 | * Early versions of Armada 375 SoC have a bug where the BootROM | 60 | * Early versions of Armada 375 SoC have a bug where the BootROM |
56 | * leaves an external data abort pending. The kernel is hit by this | 61 | * leaves an external data abort pending. The kernel is hit by this |
diff --git a/arch/arm/mach-mvebu/common.h b/arch/arm/mach-mvebu/common.h index a97778e28bf6..3ccb40c3bf94 100644 --- a/arch/arm/mach-mvebu/common.h +++ b/arch/arm/mach-mvebu/common.h | |||
@@ -23,4 +23,6 @@ void mvebu_pmsu_set_cpu_boot_addr(int hw_cpu, void *boot_addr); | |||
23 | void mvebu_system_controller_set_cpu_boot_addr(void *boot_addr); | 23 | void mvebu_system_controller_set_cpu_boot_addr(void *boot_addr); |
24 | int mvebu_system_controller_get_soc_id(u32 *dev, u32 *rev); | 24 | int mvebu_system_controller_get_soc_id(u32 *dev, u32 *rev); |
25 | 25 | ||
26 | void __iomem *mvebu_get_scu_base(void); | ||
27 | |||
26 | #endif | 28 | #endif |
diff --git a/arch/arm/mach-mvebu/headsmp-a9.S b/arch/arm/mach-mvebu/headsmp-a9.S index da5bb292b91c..be51c998c0cd 100644 --- a/arch/arm/mach-mvebu/headsmp-a9.S +++ b/arch/arm/mach-mvebu/headsmp-a9.S | |||
@@ -18,21 +18,6 @@ | |||
18 | #include <asm/assembler.h> | 18 | #include <asm/assembler.h> |
19 | 19 | ||
20 | __CPUINIT | 20 | __CPUINIT |
21 | #define CPU_RESUME_ADDR_REG 0xf10182d4 | ||
22 | |||
23 | .global armada_375_smp_cpu1_enable_code_start | ||
24 | .global armada_375_smp_cpu1_enable_code_end | ||
25 | |||
26 | armada_375_smp_cpu1_enable_code_start: | ||
27 | ARM_BE8(setend be) | ||
28 | adr r0, 1f | ||
29 | ldr r0, [r0] | ||
30 | ldr r1, [r0] | ||
31 | ARM_BE8(rev r1, r1) | ||
32 | mov pc, r1 | ||
33 | 1: | ||
34 | .word CPU_RESUME_ADDR_REG | ||
35 | armada_375_smp_cpu1_enable_code_end: | ||
36 | 21 | ||
37 | ENTRY(mvebu_cortex_a9_secondary_startup) | 22 | ENTRY(mvebu_cortex_a9_secondary_startup) |
38 | ARM_BE8(setend be) | 23 | ARM_BE8(setend be) |
diff --git a/arch/arm/mach-mvebu/platsmp-a9.c b/arch/arm/mach-mvebu/platsmp-a9.c index 43aaf3fa75ee..47a71a924b96 100644 --- a/arch/arm/mach-mvebu/platsmp-a9.c +++ b/arch/arm/mach-mvebu/platsmp-a9.c | |||
@@ -20,33 +20,8 @@ | |||
20 | #include <asm/smp_scu.h> | 20 | #include <asm/smp_scu.h> |
21 | #include <asm/smp_plat.h> | 21 | #include <asm/smp_plat.h> |
22 | #include "common.h" | 22 | #include "common.h" |
23 | #include "mvebu-soc-id.h" | ||
24 | #include "pmsu.h" | 23 | #include "pmsu.h" |
25 | 24 | ||
26 | #define CRYPT0_ENG_ID 41 | ||
27 | #define CRYPT0_ENG_ATTR 0x1 | ||
28 | #define SRAM_PHYS_BASE 0xFFFF0000 | ||
29 | |||
30 | #define BOOTROM_BASE 0xFFF00000 | ||
31 | #define BOOTROM_SIZE 0x100000 | ||
32 | |||
33 | extern unsigned char armada_375_smp_cpu1_enable_code_end; | ||
34 | extern unsigned char armada_375_smp_cpu1_enable_code_start; | ||
35 | |||
36 | static void armada_375_smp_cpu1_enable_wa(void) | ||
37 | { | ||
38 | void __iomem *sram_virt_base; | ||
39 | |||
40 | mvebu_mbus_del_window(BOOTROM_BASE, BOOTROM_SIZE); | ||
41 | mvebu_mbus_add_window_by_id(CRYPT0_ENG_ID, CRYPT0_ENG_ATTR, | ||
42 | SRAM_PHYS_BASE, SZ_64K); | ||
43 | sram_virt_base = ioremap(SRAM_PHYS_BASE, SZ_64K); | ||
44 | |||
45 | memcpy(sram_virt_base, &armada_375_smp_cpu1_enable_code_start, | ||
46 | &armada_375_smp_cpu1_enable_code_end | ||
47 | - &armada_375_smp_cpu1_enable_code_start); | ||
48 | } | ||
49 | |||
50 | extern void mvebu_cortex_a9_secondary_startup(void); | 25 | extern void mvebu_cortex_a9_secondary_startup(void); |
51 | 26 | ||
52 | static int __cpuinit mvebu_cortex_a9_boot_secondary(unsigned int cpu, | 27 | static int __cpuinit mvebu_cortex_a9_boot_secondary(unsigned int cpu, |
@@ -63,21 +38,10 @@ static int __cpuinit mvebu_cortex_a9_boot_secondary(unsigned int cpu, | |||
63 | * address. | 38 | * address. |
64 | */ | 39 | */ |
65 | hw_cpu = cpu_logical_map(cpu); | 40 | hw_cpu = cpu_logical_map(cpu); |
66 | 41 | if (of_machine_is_compatible("marvell,armada375")) | |
67 | if (of_machine_is_compatible("marvell,armada375")) { | ||
68 | u32 dev, rev; | ||
69 | |||
70 | if (mvebu_get_soc_id(&dev, &rev) == 0 && | ||
71 | rev == ARMADA_375_Z1_REV) | ||
72 | armada_375_smp_cpu1_enable_wa(); | ||
73 | |||
74 | mvebu_system_controller_set_cpu_boot_addr(mvebu_cortex_a9_secondary_startup); | 42 | mvebu_system_controller_set_cpu_boot_addr(mvebu_cortex_a9_secondary_startup); |
75 | } | 43 | else |
76 | else { | 44 | mvebu_pmsu_set_cpu_boot_addr(hw_cpu, mvebu_cortex_a9_secondary_startup); |
77 | mvebu_pmsu_set_cpu_boot_addr(hw_cpu, | ||
78 | mvebu_cortex_a9_secondary_startup); | ||
79 | } | ||
80 | |||
81 | smp_wmb(); | 45 | smp_wmb(); |
82 | ret = mvebu_cpu_reset_deassert(hw_cpu); | 46 | ret = mvebu_cpu_reset_deassert(hw_cpu); |
83 | if (ret) { | 47 | if (ret) { |
diff --git a/arch/arm/mach-mvebu/platsmp.c b/arch/arm/mach-mvebu/platsmp.c index b6fa9f0c98b8..895dc373c8a1 100644 --- a/arch/arm/mach-mvebu/platsmp.c +++ b/arch/arm/mach-mvebu/platsmp.c | |||
@@ -67,6 +67,7 @@ static void __init set_secondary_cpus_clock(void) | |||
67 | if (!cpu_clk) | 67 | if (!cpu_clk) |
68 | return; | 68 | return; |
69 | clk_set_rate(cpu_clk, rate); | 69 | clk_set_rate(cpu_clk, rate); |
70 | clk_prepare_enable(cpu_clk); | ||
70 | } | 71 | } |
71 | } | 72 | } |
72 | 73 | ||
@@ -108,7 +109,7 @@ static int armada_xp_boot_secondary(unsigned int cpu, struct task_struct *idle) | |||
108 | */ | 109 | */ |
109 | static void armada_xp_secondary_init(unsigned int cpu) | 110 | static void armada_xp_secondary_init(unsigned int cpu) |
110 | { | 111 | { |
111 | armada_370_xp_pmsu_idle_exit(); | 112 | mvebu_v7_pmsu_idle_exit(); |
112 | } | 113 | } |
113 | 114 | ||
114 | static void __init armada_xp_smp_init_cpus(void) | 115 | static void __init armada_xp_smp_init_cpus(void) |
diff --git a/arch/arm/mach-mvebu/pmsu.c b/arch/arm/mach-mvebu/pmsu.c index b31a8293a347..8a70a51533fd 100644 --- a/arch/arm/mach-mvebu/pmsu.c +++ b/arch/arm/mach-mvebu/pmsu.c | |||
@@ -18,22 +18,29 @@ | |||
18 | 18 | ||
19 | #define pr_fmt(fmt) "mvebu-pmsu: " fmt | 19 | #define pr_fmt(fmt) "mvebu-pmsu: " fmt |
20 | 20 | ||
21 | #include <linux/clk.h> | ||
21 | #include <linux/cpu_pm.h> | 22 | #include <linux/cpu_pm.h> |
22 | #include <linux/kernel.h> | 23 | #include <linux/delay.h> |
23 | #include <linux/init.h> | 24 | #include <linux/init.h> |
24 | #include <linux/of_address.h> | ||
25 | #include <linux/io.h> | 25 | #include <linux/io.h> |
26 | #include <linux/kernel.h> | ||
27 | #include <linux/mbus.h> | ||
28 | #include <linux/of_address.h> | ||
29 | #include <linux/of_device.h> | ||
26 | #include <linux/platform_device.h> | 30 | #include <linux/platform_device.h> |
27 | #include <linux/smp.h> | 31 | #include <linux/pm_opp.h> |
28 | #include <linux/resource.h> | 32 | #include <linux/resource.h> |
33 | #include <linux/slab.h> | ||
34 | #include <linux/smp.h> | ||
29 | #include <asm/cacheflush.h> | 35 | #include <asm/cacheflush.h> |
30 | #include <asm/cp15.h> | 36 | #include <asm/cp15.h> |
37 | #include <asm/smp_scu.h> | ||
31 | #include <asm/smp_plat.h> | 38 | #include <asm/smp_plat.h> |
32 | #include <asm/suspend.h> | 39 | #include <asm/suspend.h> |
33 | #include <asm/tlbflush.h> | 40 | #include <asm/tlbflush.h> |
34 | #include "common.h" | 41 | #include "common.h" |
42 | #include "armada-370-xp.h" | ||
35 | 43 | ||
36 | static void __iomem *pmsu_mp_base; | ||
37 | 44 | ||
38 | #define PMSU_BASE_OFFSET 0x100 | 45 | #define PMSU_BASE_OFFSET 0x100 |
39 | #define PMSU_REG_SIZE 0x1000 | 46 | #define PMSU_REG_SIZE 0x1000 |
@@ -57,20 +64,45 @@ static void __iomem *pmsu_mp_base; | |||
57 | #define PMSU_STATUS_AND_MASK_IRQ_MASK BIT(24) | 64 | #define PMSU_STATUS_AND_MASK_IRQ_MASK BIT(24) |
58 | #define PMSU_STATUS_AND_MASK_FIQ_MASK BIT(25) | 65 | #define PMSU_STATUS_AND_MASK_FIQ_MASK BIT(25) |
59 | 66 | ||
67 | #define PMSU_EVENT_STATUS_AND_MASK(cpu) ((cpu * 0x100) + 0x120) | ||
68 | #define PMSU_EVENT_STATUS_AND_MASK_DFS_DONE BIT(1) | ||
69 | #define PMSU_EVENT_STATUS_AND_MASK_DFS_DONE_MASK BIT(17) | ||
70 | |||
60 | #define PMSU_BOOT_ADDR_REDIRECT_OFFSET(cpu) ((cpu * 0x100) + 0x124) | 71 | #define PMSU_BOOT_ADDR_REDIRECT_OFFSET(cpu) ((cpu * 0x100) + 0x124) |
61 | 72 | ||
62 | /* PMSU fabric registers */ | 73 | /* PMSU fabric registers */ |
63 | #define L2C_NFABRIC_PM_CTL 0x4 | 74 | #define L2C_NFABRIC_PM_CTL 0x4 |
64 | #define L2C_NFABRIC_PM_CTL_PWR_DOWN BIT(20) | 75 | #define L2C_NFABRIC_PM_CTL_PWR_DOWN BIT(20) |
65 | 76 | ||
77 | /* PMSU delay registers */ | ||
78 | #define PMSU_POWERDOWN_DELAY 0xF04 | ||
79 | #define PMSU_POWERDOWN_DELAY_PMU BIT(1) | ||
80 | #define PMSU_POWERDOWN_DELAY_MASK 0xFFFE | ||
81 | #define PMSU_DFLT_ARMADA38X_DELAY 0x64 | ||
82 | |||
83 | /* CA9 MPcore SoC Control registers */ | ||
84 | |||
85 | #define MPCORE_RESET_CTL 0x64 | ||
86 | #define MPCORE_RESET_CTL_L2 BIT(0) | ||
87 | #define MPCORE_RESET_CTL_DEBUG BIT(16) | ||
88 | |||
89 | #define SRAM_PHYS_BASE 0xFFFF0000 | ||
90 | #define BOOTROM_BASE 0xFFF00000 | ||
91 | #define BOOTROM_SIZE 0x100000 | ||
92 | |||
93 | #define ARMADA_370_CRYPT0_ENG_TARGET 0x9 | ||
94 | #define ARMADA_370_CRYPT0_ENG_ATTR 0x1 | ||
95 | |||
66 | extern void ll_disable_coherency(void); | 96 | extern void ll_disable_coherency(void); |
67 | extern void ll_enable_coherency(void); | 97 | extern void ll_enable_coherency(void); |
68 | 98 | ||
69 | extern void armada_370_xp_cpu_resume(void); | 99 | extern void armada_370_xp_cpu_resume(void); |
100 | extern void armada_38x_cpu_resume(void); | ||
70 | 101 | ||
71 | static struct platform_device armada_xp_cpuidle_device = { | 102 | static phys_addr_t pmsu_mp_phys_base; |
72 | .name = "cpuidle-armada-370-xp", | 103 | static void __iomem *pmsu_mp_base; |
73 | }; | 104 | |
105 | static void *mvebu_cpu_resume; | ||
74 | 106 | ||
75 | static struct of_device_id of_pmsu_table[] = { | 107 | static struct of_device_id of_pmsu_table[] = { |
76 | { .compatible = "marvell,armada-370-pmsu", }, | 108 | { .compatible = "marvell,armada-370-pmsu", }, |
@@ -85,7 +117,49 @@ void mvebu_pmsu_set_cpu_boot_addr(int hw_cpu, void *boot_addr) | |||
85 | PMSU_BOOT_ADDR_REDIRECT_OFFSET(hw_cpu)); | 117 | PMSU_BOOT_ADDR_REDIRECT_OFFSET(hw_cpu)); |
86 | } | 118 | } |
87 | 119 | ||
88 | static int __init armada_370_xp_pmsu_init(void) | 120 | extern unsigned char mvebu_boot_wa_start; |
121 | extern unsigned char mvebu_boot_wa_end; | ||
122 | |||
123 | /* | ||
124 | * This function sets up the boot address workaround needed for SMP | ||
125 | * boot on Armada 375 Z1 and cpuidle on Armada 370. It unmaps the | ||
126 | * BootROM Mbus window, and instead remaps a crypto SRAM into which a | ||
127 | * custom piece of code is copied to replace the problematic BootROM. | ||
128 | */ | ||
129 | int mvebu_setup_boot_addr_wa(unsigned int crypto_eng_target, | ||
130 | unsigned int crypto_eng_attribute, | ||
131 | phys_addr_t resume_addr_reg) | ||
132 | { | ||
133 | void __iomem *sram_virt_base; | ||
134 | u32 code_len = &mvebu_boot_wa_end - &mvebu_boot_wa_start; | ||
135 | |||
136 | mvebu_mbus_del_window(BOOTROM_BASE, BOOTROM_SIZE); | ||
137 | mvebu_mbus_add_window_by_id(crypto_eng_target, crypto_eng_attribute, | ||
138 | SRAM_PHYS_BASE, SZ_64K); | ||
139 | |||
140 | sram_virt_base = ioremap(SRAM_PHYS_BASE, SZ_64K); | ||
141 | if (!sram_virt_base) { | ||
142 | pr_err("Unable to map SRAM to setup the boot address WA\n"); | ||
143 | return -ENOMEM; | ||
144 | } | ||
145 | |||
146 | memcpy(sram_virt_base, &mvebu_boot_wa_start, code_len); | ||
147 | |||
148 | /* | ||
149 | * The last word of the code copied in SRAM must contain the | ||
150 | * physical base address of the PMSU register. We | ||
151 | * intentionally store this address in the native endianness | ||
152 | * of the system. | ||
153 | */ | ||
154 | __raw_writel((unsigned long)resume_addr_reg, | ||
155 | sram_virt_base + code_len - 4); | ||
156 | |||
157 | iounmap(sram_virt_base); | ||
158 | |||
159 | return 0; | ||
160 | } | ||
161 | |||
162 | static int __init mvebu_v7_pmsu_init(void) | ||
89 | { | 163 | { |
90 | struct device_node *np; | 164 | struct device_node *np; |
91 | struct resource res; | 165 | struct resource res; |
@@ -116,6 +190,8 @@ static int __init armada_370_xp_pmsu_init(void) | |||
116 | goto out; | 190 | goto out; |
117 | } | 191 | } |
118 | 192 | ||
193 | pmsu_mp_phys_base = res.start; | ||
194 | |||
119 | pmsu_mp_base = ioremap(res.start, resource_size(&res)); | 195 | pmsu_mp_base = ioremap(res.start, resource_size(&res)); |
120 | if (!pmsu_mp_base) { | 196 | if (!pmsu_mp_base) { |
121 | pr_err("unable to map registers\n"); | 197 | pr_err("unable to map registers\n"); |
@@ -129,7 +205,7 @@ static int __init armada_370_xp_pmsu_init(void) | |||
129 | return ret; | 205 | return ret; |
130 | } | 206 | } |
131 | 207 | ||
132 | static void armada_370_xp_pmsu_enable_l2_powerdown_onidle(void) | 208 | static void mvebu_v7_pmsu_enable_l2_powerdown_onidle(void) |
133 | { | 209 | { |
134 | u32 reg; | 210 | u32 reg; |
135 | 211 | ||
@@ -142,8 +218,14 @@ static void armada_370_xp_pmsu_enable_l2_powerdown_onidle(void) | |||
142 | writel(reg, pmsu_mp_base + L2C_NFABRIC_PM_CTL); | 218 | writel(reg, pmsu_mp_base + L2C_NFABRIC_PM_CTL); |
143 | } | 219 | } |
144 | 220 | ||
221 | enum pmsu_idle_prepare_flags { | ||
222 | PMSU_PREPARE_NORMAL = 0, | ||
223 | PMSU_PREPARE_DEEP_IDLE = BIT(0), | ||
224 | PMSU_PREPARE_SNOOP_DISABLE = BIT(1), | ||
225 | }; | ||
226 | |||
145 | /* No locking is needed because we only access per-CPU registers */ | 227 | /* No locking is needed because we only access per-CPU registers */ |
146 | int armada_370_xp_pmsu_idle_enter(unsigned long deepidle) | 228 | static int mvebu_v7_pmsu_idle_prepare(unsigned long flags) |
147 | { | 229 | { |
148 | unsigned int hw_cpu = cpu_logical_map(smp_processor_id()); | 230 | unsigned int hw_cpu = cpu_logical_map(smp_processor_id()); |
149 | u32 reg; | 231 | u32 reg; |
@@ -167,17 +249,34 @@ int armada_370_xp_pmsu_idle_enter(unsigned long deepidle) | |||
167 | 249 | ||
168 | reg = readl(pmsu_mp_base + PMSU_CONTROL_AND_CONFIG(hw_cpu)); | 250 | reg = readl(pmsu_mp_base + PMSU_CONTROL_AND_CONFIG(hw_cpu)); |
169 | /* ask HW to power down the L2 Cache if needed */ | 251 | /* ask HW to power down the L2 Cache if needed */ |
170 | if (deepidle) | 252 | if (flags & PMSU_PREPARE_DEEP_IDLE) |
171 | reg |= PMSU_CONTROL_AND_CONFIG_L2_PWDDN; | 253 | reg |= PMSU_CONTROL_AND_CONFIG_L2_PWDDN; |
172 | 254 | ||
173 | /* request power down */ | 255 | /* request power down */ |
174 | reg |= PMSU_CONTROL_AND_CONFIG_PWDDN_REQ; | 256 | reg |= PMSU_CONTROL_AND_CONFIG_PWDDN_REQ; |
175 | writel(reg, pmsu_mp_base + PMSU_CONTROL_AND_CONFIG(hw_cpu)); | 257 | writel(reg, pmsu_mp_base + PMSU_CONTROL_AND_CONFIG(hw_cpu)); |
176 | 258 | ||
177 | /* Disable snoop disable by HW - SW is taking care of it */ | 259 | if (flags & PMSU_PREPARE_SNOOP_DISABLE) { |
178 | reg = readl(pmsu_mp_base + PMSU_CPU_POWER_DOWN_CONTROL(hw_cpu)); | 260 | /* Disable snoop disable by HW - SW is taking care of it */ |
179 | reg |= PMSU_CPU_POWER_DOWN_DIS_SNP_Q_SKIP; | 261 | reg = readl(pmsu_mp_base + PMSU_CPU_POWER_DOWN_CONTROL(hw_cpu)); |
180 | writel(reg, pmsu_mp_base + PMSU_CPU_POWER_DOWN_CONTROL(hw_cpu)); | 262 | reg |= PMSU_CPU_POWER_DOWN_DIS_SNP_Q_SKIP; |
263 | writel(reg, pmsu_mp_base + PMSU_CPU_POWER_DOWN_CONTROL(hw_cpu)); | ||
264 | } | ||
265 | |||
266 | return 0; | ||
267 | } | ||
268 | |||
269 | int armada_370_xp_pmsu_idle_enter(unsigned long deepidle) | ||
270 | { | ||
271 | unsigned long flags = PMSU_PREPARE_SNOOP_DISABLE; | ||
272 | int ret; | ||
273 | |||
274 | if (deepidle) | ||
275 | flags |= PMSU_PREPARE_DEEP_IDLE; | ||
276 | |||
277 | ret = mvebu_v7_pmsu_idle_prepare(flags); | ||
278 | if (ret) | ||
279 | return ret; | ||
181 | 280 | ||
182 | v7_exit_coherency_flush(all); | 281 | v7_exit_coherency_flush(all); |
183 | 282 | ||
@@ -203,7 +302,7 @@ int armada_370_xp_pmsu_idle_enter(unsigned long deepidle) | |||
203 | "isb " | 302 | "isb " |
204 | : : : "r0"); | 303 | : : : "r0"); |
205 | 304 | ||
206 | pr_warn("Failed to suspend the system\n"); | 305 | pr_debug("Failed to suspend the system\n"); |
207 | 306 | ||
208 | return 0; | 307 | return 0; |
209 | } | 308 | } |
@@ -213,15 +312,40 @@ static int armada_370_xp_cpu_suspend(unsigned long deepidle) | |||
213 | return cpu_suspend(deepidle, armada_370_xp_pmsu_idle_enter); | 312 | return cpu_suspend(deepidle, armada_370_xp_pmsu_idle_enter); |
214 | } | 313 | } |
215 | 314 | ||
315 | static int armada_38x_do_cpu_suspend(unsigned long deepidle) | ||
316 | { | ||
317 | unsigned long flags = 0; | ||
318 | |||
319 | if (deepidle) | ||
320 | flags |= PMSU_PREPARE_DEEP_IDLE; | ||
321 | |||
322 | mvebu_v7_pmsu_idle_prepare(flags); | ||
323 | /* | ||
324 | * Already flushed cache, but do it again as the outer cache | ||
325 | * functions dirty the cache with spinlocks | ||
326 | */ | ||
327 | v7_exit_coherency_flush(louis); | ||
328 | |||
329 | scu_power_mode(mvebu_get_scu_base(), SCU_PM_POWEROFF); | ||
330 | |||
331 | cpu_do_idle(); | ||
332 | |||
333 | return 1; | ||
334 | } | ||
335 | |||
336 | static int armada_38x_cpu_suspend(unsigned long deepidle) | ||
337 | { | ||
338 | return cpu_suspend(false, armada_38x_do_cpu_suspend); | ||
339 | } | ||
340 | |||
216 | /* No locking is needed because we only access per-CPU registers */ | 341 | /* No locking is needed because we only access per-CPU registers */ |
217 | void armada_370_xp_pmsu_idle_exit(void) | 342 | void mvebu_v7_pmsu_idle_exit(void) |
218 | { | 343 | { |
219 | unsigned int hw_cpu = cpu_logical_map(smp_processor_id()); | 344 | unsigned int hw_cpu = cpu_logical_map(smp_processor_id()); |
220 | u32 reg; | 345 | u32 reg; |
221 | 346 | ||
222 | if (pmsu_mp_base == NULL) | 347 | if (pmsu_mp_base == NULL) |
223 | return; | 348 | return; |
224 | |||
225 | /* cancel ask HW to power down the L2 Cache if possible */ | 349 | /* cancel ask HW to power down the L2 Cache if possible */ |
226 | reg = readl(pmsu_mp_base + PMSU_CONTROL_AND_CONFIG(hw_cpu)); | 350 | reg = readl(pmsu_mp_base + PMSU_CONTROL_AND_CONFIG(hw_cpu)); |
227 | reg &= ~PMSU_CONTROL_AND_CONFIG_L2_PWDDN; | 351 | reg &= ~PMSU_CONTROL_AND_CONFIG_L2_PWDDN; |
@@ -236,53 +360,292 @@ void armada_370_xp_pmsu_idle_exit(void) | |||
236 | writel(reg, pmsu_mp_base + PMSU_STATUS_AND_MASK(hw_cpu)); | 360 | writel(reg, pmsu_mp_base + PMSU_STATUS_AND_MASK(hw_cpu)); |
237 | } | 361 | } |
238 | 362 | ||
239 | static int armada_370_xp_cpu_pm_notify(struct notifier_block *self, | 363 | static int mvebu_v7_cpu_pm_notify(struct notifier_block *self, |
240 | unsigned long action, void *hcpu) | 364 | unsigned long action, void *hcpu) |
241 | { | 365 | { |
242 | if (action == CPU_PM_ENTER) { | 366 | if (action == CPU_PM_ENTER) { |
243 | unsigned int hw_cpu = cpu_logical_map(smp_processor_id()); | 367 | unsigned int hw_cpu = cpu_logical_map(smp_processor_id()); |
244 | mvebu_pmsu_set_cpu_boot_addr(hw_cpu, armada_370_xp_cpu_resume); | 368 | mvebu_pmsu_set_cpu_boot_addr(hw_cpu, mvebu_cpu_resume); |
245 | } else if (action == CPU_PM_EXIT) { | 369 | } else if (action == CPU_PM_EXIT) { |
246 | armada_370_xp_pmsu_idle_exit(); | 370 | mvebu_v7_pmsu_idle_exit(); |
247 | } | 371 | } |
248 | 372 | ||
249 | return NOTIFY_OK; | 373 | return NOTIFY_OK; |
250 | } | 374 | } |
251 | 375 | ||
252 | static struct notifier_block armada_370_xp_cpu_pm_notifier = { | 376 | static struct notifier_block mvebu_v7_cpu_pm_notifier = { |
253 | .notifier_call = armada_370_xp_cpu_pm_notify, | 377 | .notifier_call = mvebu_v7_cpu_pm_notify, |
254 | }; | 378 | }; |
255 | 379 | ||
256 | static int __init armada_370_xp_cpu_pm_init(void) | 380 | static struct platform_device mvebu_v7_cpuidle_device; |
381 | |||
382 | static __init int armada_370_cpuidle_init(void) | ||
257 | { | 383 | { |
258 | struct device_node *np; | 384 | struct device_node *np; |
385 | phys_addr_t redirect_reg; | ||
386 | |||
387 | np = of_find_compatible_node(NULL, NULL, "marvell,coherency-fabric"); | ||
388 | if (!np) | ||
389 | return -ENODEV; | ||
390 | of_node_put(np); | ||
259 | 391 | ||
260 | /* | 392 | /* |
261 | * Check that all the requirements are available to enable | 393 | * On Armada 370, there is "a slow exit process from the deep |
262 | * cpuidle. So far, it is only supported on Armada XP, cpuidle | 394 | * idle state due to heavy L1/L2 cache cleanup operations |
263 | * needs the coherency fabric and the PMSU enabled | 395 | * performed by the BootROM software". To avoid this, we |
396 | * replace the restart code of the bootrom by a a simple jump | ||
397 | * to the boot address. Then the code located at this boot | ||
398 | * address will take care of the initialization. | ||
264 | */ | 399 | */ |
400 | redirect_reg = pmsu_mp_phys_base + PMSU_BOOT_ADDR_REDIRECT_OFFSET(0); | ||
401 | mvebu_setup_boot_addr_wa(ARMADA_370_CRYPT0_ENG_TARGET, | ||
402 | ARMADA_370_CRYPT0_ENG_ATTR, | ||
403 | redirect_reg); | ||
265 | 404 | ||
266 | if (!of_machine_is_compatible("marvell,armadaxp")) | 405 | mvebu_cpu_resume = armada_370_xp_cpu_resume; |
267 | return 0; | 406 | mvebu_v7_cpuidle_device.dev.platform_data = armada_370_xp_cpu_suspend; |
407 | mvebu_v7_cpuidle_device.name = "cpuidle-armada-370"; | ||
408 | |||
409 | return 0; | ||
410 | } | ||
411 | |||
412 | static __init int armada_38x_cpuidle_init(void) | ||
413 | { | ||
414 | struct device_node *np; | ||
415 | void __iomem *mpsoc_base; | ||
416 | u32 reg; | ||
417 | |||
418 | np = of_find_compatible_node(NULL, NULL, | ||
419 | "marvell,armada-380-coherency-fabric"); | ||
420 | if (!np) | ||
421 | return -ENODEV; | ||
422 | of_node_put(np); | ||
423 | |||
424 | np = of_find_compatible_node(NULL, NULL, | ||
425 | "marvell,armada-380-mpcore-soc-ctrl"); | ||
426 | if (!np) | ||
427 | return -ENODEV; | ||
428 | mpsoc_base = of_iomap(np, 0); | ||
429 | BUG_ON(!mpsoc_base); | ||
430 | of_node_put(np); | ||
431 | |||
432 | /* Set up reset mask when powering down the cpus */ | ||
433 | reg = readl(mpsoc_base + MPCORE_RESET_CTL); | ||
434 | reg |= MPCORE_RESET_CTL_L2; | ||
435 | reg |= MPCORE_RESET_CTL_DEBUG; | ||
436 | writel(reg, mpsoc_base + MPCORE_RESET_CTL); | ||
437 | iounmap(mpsoc_base); | ||
438 | |||
439 | /* Set up delay */ | ||
440 | reg = readl(pmsu_mp_base + PMSU_POWERDOWN_DELAY); | ||
441 | reg &= ~PMSU_POWERDOWN_DELAY_MASK; | ||
442 | reg |= PMSU_DFLT_ARMADA38X_DELAY; | ||
443 | reg |= PMSU_POWERDOWN_DELAY_PMU; | ||
444 | writel(reg, pmsu_mp_base + PMSU_POWERDOWN_DELAY); | ||
445 | |||
446 | mvebu_cpu_resume = armada_38x_cpu_resume; | ||
447 | mvebu_v7_cpuidle_device.dev.platform_data = armada_38x_cpu_suspend; | ||
448 | mvebu_v7_cpuidle_device.name = "cpuidle-armada-38x"; | ||
449 | |||
450 | return 0; | ||
451 | } | ||
452 | |||
453 | static __init int armada_xp_cpuidle_init(void) | ||
454 | { | ||
455 | struct device_node *np; | ||
268 | 456 | ||
269 | np = of_find_compatible_node(NULL, NULL, "marvell,coherency-fabric"); | 457 | np = of_find_compatible_node(NULL, NULL, "marvell,coherency-fabric"); |
270 | if (!np) | 458 | if (!np) |
271 | return 0; | 459 | return -ENODEV; |
272 | of_node_put(np); | 460 | of_node_put(np); |
273 | 461 | ||
462 | mvebu_cpu_resume = armada_370_xp_cpu_resume; | ||
463 | mvebu_v7_cpuidle_device.dev.platform_data = armada_370_xp_cpu_suspend; | ||
464 | mvebu_v7_cpuidle_device.name = "cpuidle-armada-xp"; | ||
465 | |||
466 | return 0; | ||
467 | } | ||
468 | |||
469 | static int __init mvebu_v7_cpu_pm_init(void) | ||
470 | { | ||
471 | struct device_node *np; | ||
472 | int ret; | ||
473 | |||
274 | np = of_find_matching_node(NULL, of_pmsu_table); | 474 | np = of_find_matching_node(NULL, of_pmsu_table); |
275 | if (!np) | 475 | if (!np) |
276 | return 0; | 476 | return 0; |
277 | of_node_put(np); | 477 | of_node_put(np); |
278 | 478 | ||
279 | armada_370_xp_pmsu_enable_l2_powerdown_onidle(); | 479 | if (of_machine_is_compatible("marvell,armadaxp")) |
280 | armada_xp_cpuidle_device.dev.platform_data = armada_370_xp_cpu_suspend; | 480 | ret = armada_xp_cpuidle_init(); |
281 | platform_device_register(&armada_xp_cpuidle_device); | 481 | else if (of_machine_is_compatible("marvell,armada370")) |
282 | cpu_pm_register_notifier(&armada_370_xp_cpu_pm_notifier); | 482 | ret = armada_370_cpuidle_init(); |
483 | else if (of_machine_is_compatible("marvell,armada380")) | ||
484 | ret = armada_38x_cpuidle_init(); | ||
485 | else | ||
486 | return 0; | ||
487 | |||
488 | if (ret) | ||
489 | return ret; | ||
490 | |||
491 | mvebu_v7_pmsu_enable_l2_powerdown_onidle(); | ||
492 | platform_device_register(&mvebu_v7_cpuidle_device); | ||
493 | cpu_pm_register_notifier(&mvebu_v7_cpu_pm_notifier); | ||
494 | |||
495 | return 0; | ||
496 | } | ||
497 | |||
498 | arch_initcall(mvebu_v7_cpu_pm_init); | ||
499 | early_initcall(mvebu_v7_pmsu_init); | ||
500 | |||
501 | static void mvebu_pmsu_dfs_request_local(void *data) | ||
502 | { | ||
503 | u32 reg; | ||
504 | u32 cpu = smp_processor_id(); | ||
505 | unsigned long flags; | ||
506 | |||
507 | local_irq_save(flags); | ||
508 | |||
509 | /* Prepare to enter idle */ | ||
510 | reg = readl(pmsu_mp_base + PMSU_STATUS_AND_MASK(cpu)); | ||
511 | reg |= PMSU_STATUS_AND_MASK_CPU_IDLE_WAIT | | ||
512 | PMSU_STATUS_AND_MASK_IRQ_MASK | | ||
513 | PMSU_STATUS_AND_MASK_FIQ_MASK; | ||
514 | writel(reg, pmsu_mp_base + PMSU_STATUS_AND_MASK(cpu)); | ||
515 | |||
516 | /* Request the DFS transition */ | ||
517 | reg = readl(pmsu_mp_base + PMSU_CONTROL_AND_CONFIG(cpu)); | ||
518 | reg |= PMSU_CONTROL_AND_CONFIG_DFS_REQ; | ||
519 | writel(reg, pmsu_mp_base + PMSU_CONTROL_AND_CONFIG(cpu)); | ||
520 | |||
521 | /* The fact of entering idle will trigger the DFS transition */ | ||
522 | wfi(); | ||
523 | |||
524 | /* | ||
525 | * We're back from idle, the DFS transition has completed, | ||
526 | * clear the idle wait indication. | ||
527 | */ | ||
528 | reg = readl(pmsu_mp_base + PMSU_STATUS_AND_MASK(cpu)); | ||
529 | reg &= ~PMSU_STATUS_AND_MASK_CPU_IDLE_WAIT; | ||
530 | writel(reg, pmsu_mp_base + PMSU_STATUS_AND_MASK(cpu)); | ||
531 | |||
532 | local_irq_restore(flags); | ||
533 | } | ||
534 | |||
535 | int mvebu_pmsu_dfs_request(int cpu) | ||
536 | { | ||
537 | unsigned long timeout; | ||
538 | int hwcpu = cpu_logical_map(cpu); | ||
539 | u32 reg; | ||
540 | |||
541 | /* Clear any previous DFS DONE event */ | ||
542 | reg = readl(pmsu_mp_base + PMSU_EVENT_STATUS_AND_MASK(hwcpu)); | ||
543 | reg &= ~PMSU_EVENT_STATUS_AND_MASK_DFS_DONE; | ||
544 | writel(reg, pmsu_mp_base + PMSU_EVENT_STATUS_AND_MASK(hwcpu)); | ||
545 | |||
546 | /* Mask the DFS done interrupt, since we are going to poll */ | ||
547 | reg = readl(pmsu_mp_base + PMSU_EVENT_STATUS_AND_MASK(hwcpu)); | ||
548 | reg |= PMSU_EVENT_STATUS_AND_MASK_DFS_DONE_MASK; | ||
549 | writel(reg, pmsu_mp_base + PMSU_EVENT_STATUS_AND_MASK(hwcpu)); | ||
550 | |||
551 | /* Trigger the DFS on the appropriate CPU */ | ||
552 | smp_call_function_single(cpu, mvebu_pmsu_dfs_request_local, | ||
553 | NULL, false); | ||
554 | |||
555 | /* Poll until the DFS done event is generated */ | ||
556 | timeout = jiffies + HZ; | ||
557 | while (time_before(jiffies, timeout)) { | ||
558 | reg = readl(pmsu_mp_base + PMSU_EVENT_STATUS_AND_MASK(hwcpu)); | ||
559 | if (reg & PMSU_EVENT_STATUS_AND_MASK_DFS_DONE) | ||
560 | break; | ||
561 | udelay(10); | ||
562 | } | ||
563 | |||
564 | if (time_after(jiffies, timeout)) | ||
565 | return -ETIME; | ||
566 | |||
567 | /* Restore the DFS mask to its original state */ | ||
568 | reg = readl(pmsu_mp_base + PMSU_EVENT_STATUS_AND_MASK(hwcpu)); | ||
569 | reg &= ~PMSU_EVENT_STATUS_AND_MASK_DFS_DONE_MASK; | ||
570 | writel(reg, pmsu_mp_base + PMSU_EVENT_STATUS_AND_MASK(hwcpu)); | ||
571 | |||
572 | return 0; | ||
573 | } | ||
574 | |||
575 | static int __init armada_xp_pmsu_cpufreq_init(void) | ||
576 | { | ||
577 | struct device_node *np; | ||
578 | struct resource res; | ||
579 | int ret, cpu; | ||
580 | |||
581 | if (!of_machine_is_compatible("marvell,armadaxp")) | ||
582 | return 0; | ||
583 | |||
584 | /* | ||
585 | * In order to have proper cpufreq handling, we need to ensure | ||
586 | * that the Device Tree description of the CPU clock includes | ||
587 | * the definition of the PMU DFS registers. If not, we do not | ||
588 | * register the clock notifier and the cpufreq driver. This | ||
589 | * piece of code is only for compatibility with old Device | ||
590 | * Trees. | ||
591 | */ | ||
592 | np = of_find_compatible_node(NULL, NULL, "marvell,armada-xp-cpu-clock"); | ||
593 | if (!np) | ||
594 | return 0; | ||
595 | |||
596 | ret = of_address_to_resource(np, 1, &res); | ||
597 | if (ret) { | ||
598 | pr_warn(FW_WARN "not enabling cpufreq, deprecated armada-xp-cpu-clock binding\n"); | ||
599 | of_node_put(np); | ||
600 | return 0; | ||
601 | } | ||
602 | |||
603 | of_node_put(np); | ||
604 | |||
605 | /* | ||
606 | * For each CPU, this loop registers the operating points | ||
607 | * supported (which are the nominal CPU frequency and half of | ||
608 | * it), and registers the clock notifier that will take care | ||
609 | * of doing the PMSU part of a frequency transition. | ||
610 | */ | ||
611 | for_each_possible_cpu(cpu) { | ||
612 | struct device *cpu_dev; | ||
613 | struct clk *clk; | ||
614 | int ret; | ||
615 | |||
616 | cpu_dev = get_cpu_device(cpu); | ||
617 | if (!cpu_dev) { | ||
618 | pr_err("Cannot get CPU %d\n", cpu); | ||
619 | continue; | ||
620 | } | ||
621 | |||
622 | clk = clk_get(cpu_dev, 0); | ||
623 | if (IS_ERR(clk)) { | ||
624 | pr_err("Cannot get clock for CPU %d\n", cpu); | ||
625 | return PTR_ERR(clk); | ||
626 | } | ||
627 | |||
628 | /* | ||
629 | * In case of a failure of dev_pm_opp_add(), we don't | ||
630 | * bother with cleaning up the registered OPP (there's | ||
631 | * no function to do so), and simply cancel the | ||
632 | * registration of the cpufreq device. | ||
633 | */ | ||
634 | ret = dev_pm_opp_add(cpu_dev, clk_get_rate(clk), 0); | ||
635 | if (ret) { | ||
636 | clk_put(clk); | ||
637 | return ret; | ||
638 | } | ||
639 | |||
640 | ret = dev_pm_opp_add(cpu_dev, clk_get_rate(clk) / 2, 0); | ||
641 | if (ret) { | ||
642 | clk_put(clk); | ||
643 | return ret; | ||
644 | } | ||
645 | } | ||
283 | 646 | ||
647 | platform_device_register_simple("cpufreq-generic", -1, NULL, 0); | ||
284 | return 0; | 648 | return 0; |
285 | } | 649 | } |
286 | 650 | ||
287 | arch_initcall(armada_370_xp_cpu_pm_init); | 651 | device_initcall(armada_xp_pmsu_cpufreq_init); |
288 | early_initcall(armada_370_xp_pmsu_init); | ||
diff --git a/arch/arm/mach-mvebu/pmsu.h b/arch/arm/mach-mvebu/pmsu.h index 07a737c6b95d..6b58c1fe2b0d 100644 --- a/arch/arm/mach-mvebu/pmsu.h +++ b/arch/arm/mach-mvebu/pmsu.h | |||
@@ -12,5 +12,10 @@ | |||
12 | #define __MACH_MVEBU_PMSU_H | 12 | #define __MACH_MVEBU_PMSU_H |
13 | 13 | ||
14 | int armada_xp_boot_cpu(unsigned int cpu_id, void *phys_addr); | 14 | int armada_xp_boot_cpu(unsigned int cpu_id, void *phys_addr); |
15 | int mvebu_setup_boot_addr_wa(unsigned int crypto_eng_target, | ||
16 | unsigned int crypto_eng_attribute, | ||
17 | phys_addr_t resume_addr_reg); | ||
18 | |||
19 | void mvebu_v7_pmsu_idle_exit(void); | ||
15 | 20 | ||
16 | #endif /* __MACH_370_XP_PMSU_H */ | 21 | #endif /* __MACH_370_XP_PMSU_H */ |
diff --git a/arch/arm/mach-mvebu/pmsu_ll.S b/arch/arm/mach-mvebu/pmsu_ll.S index fc3de68d8c54..a945756cfb45 100644 --- a/arch/arm/mach-mvebu/pmsu_ll.S +++ b/arch/arm/mach-mvebu/pmsu_ll.S | |||
@@ -23,3 +23,39 @@ ARM_BE8(setend be ) @ go BE8 if entered LE | |||
23 | b cpu_resume | 23 | b cpu_resume |
24 | ENDPROC(armada_370_xp_cpu_resume) | 24 | ENDPROC(armada_370_xp_cpu_resume) |
25 | 25 | ||
26 | ENTRY(armada_38x_cpu_resume) | ||
27 | /* do we need it for Armada 38x*/ | ||
28 | ARM_BE8(setend be ) @ go BE8 if entered LE | ||
29 | bl v7_invalidate_l1 | ||
30 | mrc p15, 4, r1, c15, c0 @ get SCU base address | ||
31 | orr r1, r1, #0x8 @ SCU CPU Power Status Register | ||
32 | mrc 15, 0, r0, cr0, cr0, 5 @ get the CPU ID | ||
33 | and r0, r0, #15 | ||
34 | add r1, r1, r0 | ||
35 | mov r0, #0x0 | ||
36 | strb r0, [r1] @ switch SCU power state to Normal mode | ||
37 | b cpu_resume | ||
38 | ENDPROC(armada_38x_cpu_resume) | ||
39 | |||
40 | .global mvebu_boot_wa_start | ||
41 | .global mvebu_boot_wa_end | ||
42 | |||
43 | /* The following code will be executed from SRAM */ | ||
44 | ENTRY(mvebu_boot_wa_start) | ||
45 | mvebu_boot_wa_start: | ||
46 | ARM_BE8(setend be) | ||
47 | adr r0, 1f | ||
48 | ldr r0, [r0] @ load the address of the | ||
49 | @ resume register | ||
50 | ldr r0, [r0] @ load the value in the | ||
51 | @ resume register | ||
52 | ARM_BE8(rev r0, r0) @ the value is stored LE | ||
53 | mov pc, r0 @ jump to this value | ||
54 | /* | ||
55 | * the last word of this piece of code will be filled by the physical | ||
56 | * address of the boot address register just after being copied in SRAM | ||
57 | */ | ||
58 | 1: | ||
59 | .long . | ||
60 | mvebu_boot_wa_end: | ||
61 | ENDPROC(mvebu_boot_wa_end) | ||
diff --git a/arch/arm/mach-mvebu/system-controller.c b/arch/arm/mach-mvebu/system-controller.c index b2b4e3d6558c..a068cb5c2ce8 100644 --- a/arch/arm/mach-mvebu/system-controller.c +++ b/arch/arm/mach-mvebu/system-controller.c | |||
@@ -28,8 +28,14 @@ | |||
28 | #include <linux/io.h> | 28 | #include <linux/io.h> |
29 | #include <linux/reboot.h> | 29 | #include <linux/reboot.h> |
30 | #include "common.h" | 30 | #include "common.h" |
31 | #include "mvebu-soc-id.h" | ||
32 | #include "pmsu.h" | ||
33 | |||
34 | #define ARMADA_375_CRYPT0_ENG_TARGET 41 | ||
35 | #define ARMADA_375_CRYPT0_ENG_ATTR 1 | ||
31 | 36 | ||
32 | static void __iomem *system_controller_base; | 37 | static void __iomem *system_controller_base; |
38 | static phys_addr_t system_controller_phys_base; | ||
33 | 39 | ||
34 | struct mvebu_system_controller { | 40 | struct mvebu_system_controller { |
35 | u32 rstoutn_mask_offset; | 41 | u32 rstoutn_mask_offset; |
@@ -121,10 +127,32 @@ int mvebu_system_controller_get_soc_id(u32 *dev, u32 *rev) | |||
121 | } | 127 | } |
122 | 128 | ||
123 | #ifdef CONFIG_SMP | 129 | #ifdef CONFIG_SMP |
130 | void mvebu_armada375_smp_wa_init(void) | ||
131 | { | ||
132 | u32 dev, rev; | ||
133 | phys_addr_t resume_addr_reg; | ||
134 | |||
135 | if (mvebu_get_soc_id(&dev, &rev) != 0) | ||
136 | return; | ||
137 | |||
138 | if (rev != ARMADA_375_Z1_REV) | ||
139 | return; | ||
140 | |||
141 | resume_addr_reg = system_controller_phys_base + | ||
142 | mvebu_sc->resume_boot_addr; | ||
143 | mvebu_setup_boot_addr_wa(ARMADA_375_CRYPT0_ENG_TARGET, | ||
144 | ARMADA_375_CRYPT0_ENG_ATTR, | ||
145 | resume_addr_reg); | ||
146 | } | ||
147 | |||
124 | void mvebu_system_controller_set_cpu_boot_addr(void *boot_addr) | 148 | void mvebu_system_controller_set_cpu_boot_addr(void *boot_addr) |
125 | { | 149 | { |
126 | BUG_ON(system_controller_base == NULL); | 150 | BUG_ON(system_controller_base == NULL); |
127 | BUG_ON(mvebu_sc->resume_boot_addr == 0); | 151 | BUG_ON(mvebu_sc->resume_boot_addr == 0); |
152 | |||
153 | if (of_machine_is_compatible("marvell,armada375")) | ||
154 | mvebu_armada375_smp_wa_init(); | ||
155 | |||
128 | writel(virt_to_phys(boot_addr), system_controller_base + | 156 | writel(virt_to_phys(boot_addr), system_controller_base + |
129 | mvebu_sc->resume_boot_addr); | 157 | mvebu_sc->resume_boot_addr); |
130 | } | 158 | } |
@@ -138,7 +166,10 @@ static int __init mvebu_system_controller_init(void) | |||
138 | np = of_find_matching_node_and_match(NULL, of_system_controller_table, | 166 | np = of_find_matching_node_and_match(NULL, of_system_controller_table, |
139 | &match); | 167 | &match); |
140 | if (np) { | 168 | if (np) { |
169 | struct resource res; | ||
141 | system_controller_base = of_iomap(np, 0); | 170 | system_controller_base = of_iomap(np, 0); |
171 | of_address_to_resource(np, 0, &res); | ||
172 | system_controller_phys_base = res.start; | ||
142 | mvebu_sc = (struct mvebu_system_controller *)match->data; | 173 | mvebu_sc = (struct mvebu_system_controller *)match->data; |
143 | of_node_put(np); | 174 | of_node_put(np); |
144 | } | 175 | } |