[RESEND PATCH] ARM: EXYNOS: Fix failed second suspend on Exynos4
Chanwoo Choi
cw00.choi at samsung.com
Mon Feb 23 18:38:52 PST 2015
Hi Krzysztof,
I tested this patch for suspend-to-ram on Exynos4412-based trats2 board.
When I tested suspend-to-ram repetitively, I faced on hang issue of
suspend-to-ram for Exynos4 as before.
Could you send .config file for test?
Thanks,
Chanwoo Choi
On 02/18/2015 07:45 PM, Krzysztof Kozlowski wrote:
> On Exynos4412 boards (Trats2, Odroid U3) after enabling L2 cache in
> 56b60b8bce4a ("ARM: 8265/1: dts: exynos4: Add nodes for L2 cache
> controller") the second suspend to RAM failed. First suspend worked fine
> but the next one hang just after powering down of secondary CPUs (system
> consumed energy as it would be running but was not responsive).
>
> The issue was caused by enabling delayed reset assertion for CPU0 just
> after issuing power down of cores. This was introduced for Exynos4 in
> 13cfa6c4f7fa ("ARM: EXYNOS: Fix CPU idle clock down after CPU off").
>
> The whole behavior is not well documented but after checking with vendor
> code this should be done like this (on Exynos4):
> 1. Enable delayed reset assertion when system is running (for all CPUs).
> 2. Disable delayed reset assertion before suspending the system.
> This can be done after powering off secondary CPUs.
> 3. Re-enable the delayed reset assertion when system is resumed.
>
> Signed-off-by: Krzysztof Kozlowski <k.kozlowski at samsung.com>
> Fixes: 13cfa6c4f7fa ("ARM: EXYNOS: Fix CPU idle clock down after CPU off")
> Cc: <stable at vger.kernel.org>
> ---
> arch/arm/mach-exynos/common.h | 2 ++
> arch/arm/mach-exynos/exynos.c | 27 +++++++++++++++++++++++++++
> arch/arm/mach-exynos/platsmp.c | 39 ++-------------------------------------
> arch/arm/mach-exynos/suspend.c | 3 +++
> 4 files changed, 34 insertions(+), 37 deletions(-)
>
> diff --git a/arch/arm/mach-exynos/common.h b/arch/arm/mach-exynos/common.h
> index f70eca7ee705..0ef8d4b47102 100644
> --- a/arch/arm/mach-exynos/common.h
> +++ b/arch/arm/mach-exynos/common.h
> @@ -153,6 +153,8 @@ extern void exynos_enter_aftr(void);
>
> extern struct cpuidle_exynos_data cpuidle_coupled_exynos_data;
>
> +extern void exynos_set_delayed_reset_assertion(bool enable);
> +
> extern void s5p_init_cpu(void __iomem *cpuid_addr);
> extern unsigned int samsung_rev(void);
> extern void __iomem *cpu_boot_reg_base(void);
> diff --git a/arch/arm/mach-exynos/exynos.c b/arch/arm/mach-exynos/exynos.c
> index 2013f73797ed..e2f46295eed7 100644
> --- a/arch/arm/mach-exynos/exynos.c
> +++ b/arch/arm/mach-exynos/exynos.c
> @@ -166,6 +166,33 @@ static void __init exynos_init_io(void)
> exynos_map_io();
> }
>
> +/*
> + * Set or clear the USE_DELAYED_RESET_ASSERTION option. Used by smp code
> + * and suspend.
> + *
> + * This is necessary only on Exynos4 SoCs. When system is running
> + * USE_DELAYED_RESET_ASSERTION should be set so the ARM CLK clock down
> + * feature could properly detect global idle state when secondary CPU is
> + * powered down.
> + *
> + * However this should not be set when such system is going into suspend.
> + */
> +void exynos_set_delayed_reset_assertion(bool enable)
> +{
> + if (soc_is_exynos4()) {
> + unsigned int tmp, core_id;
> +
> + for (core_id = 0; core_id < num_possible_cpus(); core_id++) {
> + tmp = pmu_raw_readl(EXYNOS_ARM_CORE_OPTION(core_id));
> + if (enable)
> + tmp |= S5P_USE_DELAYED_RESET_ASSERTION;
> + else
> + tmp &= ~(S5P_USE_DELAYED_RESET_ASSERTION);
> + pmu_raw_writel(tmp, EXYNOS_ARM_CORE_OPTION(core_id));
> + }
> + }
> +}
> +
> static const struct of_device_id exynos_dt_pmu_match[] = {
> { .compatible = "samsung,exynos3250-pmu" },
> { .compatible = "samsung,exynos4210-pmu" },
> diff --git a/arch/arm/mach-exynos/platsmp.c b/arch/arm/mach-exynos/platsmp.c
> index 3f32c47a6d74..19d5c87c842c 100644
> --- a/arch/arm/mach-exynos/platsmp.c
> +++ b/arch/arm/mach-exynos/platsmp.c
> @@ -34,30 +34,6 @@
>
> extern void exynos4_secondary_startup(void);
>
> -/*
> - * Set or clear the USE_DELAYED_RESET_ASSERTION option, set on Exynos4 SoCs
> - * during hot-(un)plugging CPUx.
> - *
> - * The feature can be cleared safely during first boot of secondary CPU.
> - *
> - * Exynos4 SoCs require setting USE_DELAYED_RESET_ASSERTION during powering
> - * down a CPU so the CPU idle clock down feature could properly detect global
> - * idle state when CPUx is off.
> - */
> -static void exynos_set_delayed_reset_assertion(u32 core_id, bool enable)
> -{
> - if (soc_is_exynos4()) {
> - unsigned int tmp;
> -
> - tmp = pmu_raw_readl(EXYNOS_ARM_CORE_OPTION(core_id));
> - if (enable)
> - tmp |= S5P_USE_DELAYED_RESET_ASSERTION;
> - else
> - tmp &= ~(S5P_USE_DELAYED_RESET_ASSERTION);
> - pmu_raw_writel(tmp, EXYNOS_ARM_CORE_OPTION(core_id));
> - }
> -}
> -
> #ifdef CONFIG_HOTPLUG_CPU
> static inline void cpu_leave_lowpower(u32 core_id)
> {
> @@ -73,8 +49,6 @@ static inline void cpu_leave_lowpower(u32 core_id)
> : "=&r" (v)
> : "Ir" (CR_C), "Ir" (0x40)
> : "cc");
> -
> - exynos_set_delayed_reset_assertion(core_id, false);
> }
>
> static inline void platform_do_lowpower(unsigned int cpu, int *spurious)
> @@ -87,14 +61,6 @@ static inline void platform_do_lowpower(unsigned int cpu, int *spurious)
> /* Turn the CPU off on next WFI instruction. */
> exynos_cpu_power_down(core_id);
>
> - /*
> - * Exynos4 SoCs require setting
> - * USE_DELAYED_RESET_ASSERTION so the CPU idle
> - * clock down feature could properly detect
> - * global idle state when CPUx is off.
> - */
> - exynos_set_delayed_reset_assertion(core_id, true);
> -
> wfi();
>
> if (pen_release == core_id) {
> @@ -355,9 +321,6 @@ static int exynos_boot_secondary(unsigned int cpu, struct task_struct *idle)
> udelay(10);
> }
>
> - /* No harm if this is called during first boot of secondary CPU */
> - exynos_set_delayed_reset_assertion(core_id, false);
> -
> /*
> * now the secondary core is starting up let it run its
> * calibrations, then wait for it to finish
> @@ -404,6 +367,8 @@ static void __init exynos_smp_prepare_cpus(unsigned int max_cpus)
>
> exynos_sysram_init();
>
> + exynos_set_delayed_reset_assertion(true);
> +
> if (read_cpuid_part() == ARM_CPU_PART_CORTEX_A9)
> scu_enable(scu_base_addr());
>
> diff --git a/arch/arm/mach-exynos/suspend.c b/arch/arm/mach-exynos/suspend.c
> index 666ec3e5b03f..d4da035cc88a 100644
> --- a/arch/arm/mach-exynos/suspend.c
> +++ b/arch/arm/mach-exynos/suspend.c
> @@ -235,6 +235,8 @@ static void exynos_pm_enter_sleep_mode(void)
>
> static void exynos_pm_prepare(void)
> {
> + exynos_set_delayed_reset_assertion(false);
> +
> /* Set wake-up mask registers */
> exynos_pm_set_wakeup_mask();
>
> @@ -383,6 +385,7 @@ early_wakeup:
>
> /* Clear SLEEP mode set in INFORM1 */
> pmu_raw_writel(0x0, S5P_INFORM1);
> + exynos_set_delayed_reset_assertion(true);
> }
>
> static void exynos3250_pm_resume(void)
>
More information about the linux-arm-kernel
mailing list