Commit 9def7ccf authored by Heiko Stuebner's avatar Heiko Stuebner

ARM: rockchip: add support smp for rk3036

The dual-core Cortex A7 rk3036 is a bit special in that it does not allow
to control the actual powerdomain of the cpu cores, while the rest of the
smp-bringup like reset control and entry address handling stays the same.
Its bigger sibling, the quad-core rk3128 again allows powerdomain control.

So allow that case by introducing a separate smp-enable-method, that simply
disables powerdomain handling in the common code.
Signed-off-by: default avatarHeiko Stuebner <heiko@sntech.de>
Tested-by: default avatarXing Zheng <zhengxing@rock-chips.com>
Acked-by: default avatarRob Herring <robh@kernel.org>
parent 8005c49d
...@@ -200,6 +200,7 @@ nodes to be present and contain the properties described below. ...@@ -200,6 +200,7 @@ nodes to be present and contain the properties described below.
"qcom,gcc-msm8660" "qcom,gcc-msm8660"
"qcom,kpss-acc-v1" "qcom,kpss-acc-v1"
"qcom,kpss-acc-v2" "qcom,kpss-acc-v2"
"rockchip,rk3036-smp"
"rockchip,rk3066-smp" "rockchip,rk3066-smp"
"ste,dbx500-smp" "ste,dbx500-smp"
......
...@@ -42,6 +42,7 @@ static int ncores; ...@@ -42,6 +42,7 @@ static int ncores;
#define PMU_PWRDN_SCU 4 #define PMU_PWRDN_SCU 4
static struct regmap *pmu; static struct regmap *pmu;
static int has_pmu = true;
static int pmu_power_domain_is_on(int pd) static int pmu_power_domain_is_on(int pd)
{ {
...@@ -89,20 +90,23 @@ static int pmu_set_power_domain(int pd, bool on) ...@@ -89,20 +90,23 @@ static int pmu_set_power_domain(int pd, bool on)
if (!IS_ERR(rstc) && !on) if (!IS_ERR(rstc) && !on)
reset_control_assert(rstc); reset_control_assert(rstc);
ret = regmap_update_bits(pmu, PMU_PWRDN_CON, BIT(pd), val); if (has_pmu) {
if (ret < 0) { ret = regmap_update_bits(pmu, PMU_PWRDN_CON, BIT(pd), val);
pr_err("%s: could not update power domain\n", __func__);
return ret;
}
ret = -1;
while (ret != on) {
ret = pmu_power_domain_is_on(pd);
if (ret < 0) { if (ret < 0) {
pr_err("%s: could not read power domain state\n", pr_err("%s: could not update power domain\n",
__func__); __func__);
return ret; return ret;
} }
ret = -1;
while (ret != on) {
ret = pmu_power_domain_is_on(pd);
if (ret < 0) {
pr_err("%s: could not read power domain state\n",
__func__);
return ret;
}
}
} }
if (!IS_ERR(rstc)) { if (!IS_ERR(rstc)) {
...@@ -122,7 +126,7 @@ static int rockchip_boot_secondary(unsigned int cpu, struct task_struct *idle) ...@@ -122,7 +126,7 @@ static int rockchip_boot_secondary(unsigned int cpu, struct task_struct *idle)
{ {
int ret; int ret;
if (!sram_base_addr || !pmu) { if (!sram_base_addr || (has_pmu && !pmu)) {
pr_err("%s: sram or pmu missing for cpu boot\n", __func__); pr_err("%s: sram or pmu missing for cpu boot\n", __func__);
return -ENXIO; return -ENXIO;
} }
...@@ -275,7 +279,7 @@ static void __init rockchip_smp_prepare_cpus(unsigned int max_cpus) ...@@ -275,7 +279,7 @@ static void __init rockchip_smp_prepare_cpus(unsigned int max_cpus)
return; return;
} }
if (rockchip_smp_prepare_pmu()) if (has_pmu && rockchip_smp_prepare_pmu())
return; return;
if (read_cpuid_part() == ARM_CPU_PART_CORTEX_A9) { if (read_cpuid_part() == ARM_CPU_PART_CORTEX_A9) {
...@@ -318,6 +322,13 @@ static void __init rockchip_smp_prepare_cpus(unsigned int max_cpus) ...@@ -318,6 +322,13 @@ static void __init rockchip_smp_prepare_cpus(unsigned int max_cpus)
pmu_set_power_domain(0 + i, false); pmu_set_power_domain(0 + i, false);
} }
static void __init rk3036_smp_prepare_cpus(unsigned int max_cpus)
{
has_pmu = false;
rockchip_smp_prepare_cpus(max_cpus);
}
#ifdef CONFIG_HOTPLUG_CPU #ifdef CONFIG_HOTPLUG_CPU
static int rockchip_cpu_kill(unsigned int cpu) static int rockchip_cpu_kill(unsigned int cpu)
{ {
...@@ -340,6 +351,15 @@ static void rockchip_cpu_die(unsigned int cpu) ...@@ -340,6 +351,15 @@ static void rockchip_cpu_die(unsigned int cpu)
} }
#endif #endif
static struct smp_operations rk3036_smp_ops __initdata = {
.smp_prepare_cpus = rk3036_smp_prepare_cpus,
.smp_boot_secondary = rockchip_boot_secondary,
#ifdef CONFIG_HOTPLUG_CPU
.cpu_kill = rockchip_cpu_kill,
.cpu_die = rockchip_cpu_die,
#endif
};
static struct smp_operations rockchip_smp_ops __initdata = { static struct smp_operations rockchip_smp_ops __initdata = {
.smp_prepare_cpus = rockchip_smp_prepare_cpus, .smp_prepare_cpus = rockchip_smp_prepare_cpus,
.smp_boot_secondary = rockchip_boot_secondary, .smp_boot_secondary = rockchip_boot_secondary,
...@@ -349,4 +369,5 @@ static struct smp_operations rockchip_smp_ops __initdata = { ...@@ -349,4 +369,5 @@ static struct smp_operations rockchip_smp_ops __initdata = {
#endif #endif
}; };
CPU_METHOD_OF_DECLARE(rk3036_smp, "rockchip,rk3036-smp", &rk3036_smp_ops);
CPU_METHOD_OF_DECLARE(rk3066_smp, "rockchip,rk3066-smp", &rockchip_smp_ops); CPU_METHOD_OF_DECLARE(rk3066_smp, "rockchip,rk3066-smp", &rockchip_smp_ops);
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment