ARM: tegra: Support L2 cache maintenance done via firmware
[linux] / arch / arm / mach-tegra / pm.c
index 1ad5719..66c8cd6 100644 (file)
@@ -38,6 +38,7 @@
 #include <asm/smp_plat.h>
 #include <asm/suspend.h>
 #include <asm/tlbflush.h>
+#include <asm/trusted_foundations.h>
 
 #include "iomap.h"
 #include "pm.h"
@@ -195,8 +196,27 @@ void tegra_idle_lp2_last(void)
        cpu_cluster_pm_enter();
        suspend_cpu_complex();
 
+       /*
+        * L2 cache disabling using kernel API only allowed when all
+        * secondary CPU's are offline. Cache have to be disabled early
+        * if cache maintenance is done via Trusted Foundations firmware.
+        * Note that CPUIDLE won't ever enter powergate on Tegra30 if any
+        * of secondary CPU's is online and this is the LP2 codepath only
+        * for Tegra20/30.
+        */
+       if (trusted_foundations_registered())
+               outer_disable();
+
        cpu_suspend(PHYS_OFFSET - PAGE_OFFSET, &tegra_sleep_cpu);
 
+       /*
+        * Resume L2 cache if it wasn't re-enabled early during resume,
+        * which is the case for Tegra30 that has to re-enable the cache
+        * via firmware call. In other cases cache is already enabled and
+        * hence re-enabling is a no-op.
+        */
+       outer_resume();
+
        restore_cpu_complex();
        cpu_cluster_pm_exit();
 }
@@ -340,8 +360,24 @@ static int tegra_suspend_enter(suspend_state_t state)
                break;
        }
 
+       /*
+        * Cache have to be disabled early if cache maintenance is done
+        * via Trusted Foundations firmware. Otherwise this is a no-op,
+        * like on Tegra114+.
+        */
+       if (trusted_foundations_registered())
+               outer_disable();
+
        cpu_suspend(PHYS_OFFSET - PAGE_OFFSET, tegra_sleep_func);
 
+       /*
+        * Resume L2 cache if it wasn't re-enabled early during resume,
+        * which is the case for Tegra30 that has to re-enable the cache
+        * via firmware call. In other cases cache is already enabled and
+        * hence re-enabling is a no-op.
+        */
+       outer_resume();
+
        switch (mode) {
        case TEGRA_SUSPEND_LP1:
                tegra_suspend_exit_lp1();