2019-06-04 16:11:33 +08:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-only
|
2009-05-29 01:56:16 +08:00
|
|
|
/*
|
|
|
|
* OMAP3 Power Management Routines
|
|
|
|
*
|
|
|
|
* Copyright (C) 2006-2008 Nokia Corporation
|
|
|
|
* Tony Lindgren <tony@atomide.com>
|
|
|
|
* Jouni Hogander
|
|
|
|
*
|
2008-09-26 20:20:07 +08:00
|
|
|
* Copyright (C) 2007 Texas Instruments, Inc.
|
|
|
|
* Rajendra Nayak <rnayak@ti.com>
|
|
|
|
*
|
2009-05-29 01:56:16 +08:00
|
|
|
* Copyright (C) 2005 Texas Instruments, Inc.
|
|
|
|
* Richard Woodruff <r-woodruff2@ti.com>
|
|
|
|
*
|
|
|
|
* Based on pm.c for omap1
|
|
|
|
*/
|
|
|
|
|
2018-09-21 03:35:31 +08:00
|
|
|
#include <linux/cpu_pm.h>
|
2009-05-29 01:56:16 +08:00
|
|
|
#include <linux/pm.h>
|
|
|
|
#include <linux/suspend.h>
|
|
|
|
#include <linux/interrupt.h>
|
|
|
|
#include <linux/module.h>
|
|
|
|
#include <linux/list.h>
|
|
|
|
#include <linux/err.h>
|
2009-10-07 05:25:09 +08:00
|
|
|
#include <linux/clk.h>
|
2009-11-18 00:34:53 +08:00
|
|
|
#include <linux/delay.h>
|
include cleanup: Update gfp.h and slab.h includes to prepare for breaking implicit slab.h inclusion from percpu.h
percpu.h is included by sched.h and module.h and thus ends up being
included when building most .c files. percpu.h includes slab.h which
in turn includes gfp.h making everything defined by the two files
universally available and complicating inclusion dependencies.
percpu.h -> slab.h dependency is about to be removed. Prepare for
this change by updating users of gfp and slab facilities include those
headers directly instead of assuming availability. As this conversion
needs to touch large number of source files, the following script is
used as the basis of conversion.
http://userweb.kernel.org/~tj/misc/slabh-sweep.py
The script does the followings.
* Scan files for gfp and slab usages and update includes such that
only the necessary includes are there. ie. if only gfp is used,
gfp.h, if slab is used, slab.h.
* When the script inserts a new include, it looks at the include
blocks and try to put the new include such that its order conforms
to its surrounding. It's put in the include block which contains
core kernel includes, in the same order that the rest are ordered -
alphabetical, Christmas tree, rev-Xmas-tree or at the end if there
doesn't seem to be any matching order.
* If the script can't find a place to put a new include (mostly
because the file doesn't have fitting include block), it prints out
an error message indicating which .h file needs to be added to the
file.
The conversion was done in the following steps.
1. The initial automatic conversion of all .c files updated slightly
over 4000 files, deleting around 700 includes and adding ~480 gfp.h
and ~3000 slab.h inclusions. The script emitted errors for ~400
files.
2. Each error was manually checked. Some didn't need the inclusion,
some needed manual addition while adding it to implementation .h or
embedding .c file was more appropriate for others. This step added
inclusions to around 150 files.
3. The script was run again and the output was compared to the edits
from #2 to make sure no file was left behind.
4. Several build tests were done and a couple of problems were fixed.
e.g. lib/decompress_*.c used malloc/free() wrappers around slab
APIs requiring slab.h to be added manually.
5. The script was run on all .h files but without automatically
editing them as sprinkling gfp.h and slab.h inclusions around .h
files could easily lead to inclusion dependency hell. Most gfp.h
inclusion directives were ignored as stuff from gfp.h was usually
wildly available and often used in preprocessor macros. Each
slab.h inclusion directive was examined and added manually as
necessary.
6. percpu.h was updated not to include slab.h.
7. Build test were done on the following configurations and failures
were fixed. CONFIG_GCOV_KERNEL was turned off for all tests (as my
distributed build env didn't work with gcov compiles) and a few
more options had to be turned off depending on archs to make things
build (like ipr on powerpc/64 which failed due to missing writeq).
* x86 and x86_64 UP and SMP allmodconfig and a custom test config.
* powerpc and powerpc64 SMP allmodconfig
* sparc and sparc64 SMP allmodconfig
* ia64 SMP allmodconfig
* s390 SMP allmodconfig
* alpha SMP allmodconfig
* um on x86_64 SMP allmodconfig
8. percpu.h modifications were reverted so that it could be applied as
a separate patch and serve as bisection point.
Given the fact that I had only a couple of failures from tests on step
6, I'm fairly confident about the coverage of this conversion patch.
If there is a breakage, it's likely to be something in one of the arch
headers which should be easily discoverable easily on most builds of
the specific arch.
Signed-off-by: Tejun Heo <tj@kernel.org>
Guess-its-ok-by: Christoph Lameter <cl@linux-foundation.org>
Cc: Ingo Molnar <mingo@redhat.com>
Cc: Lee Schermerhorn <Lee.Schermerhorn@hp.com>
2010-03-24 16:04:11 +08:00
|
|
|
#include <linux/slab.h>
|
2014-11-21 04:11:25 +08:00
|
|
|
#include <linux/omap-gpmc.h>
|
2012-08-31 06:37:24 +08:00
|
|
|
|
2011-03-03 18:25:43 +08:00
|
|
|
#include <trace/events/power.h>
|
2009-05-29 01:56:16 +08:00
|
|
|
|
2012-10-30 04:54:06 +08:00
|
|
|
#include <asm/fncpy.h>
|
2011-06-23 00:41:48 +08:00
|
|
|
#include <asm/suspend.h>
|
2012-03-29 01:30:01 +08:00
|
|
|
#include <asm/system_misc.h>
|
2011-06-23 00:41:48 +08:00
|
|
|
|
2010-12-22 12:05:15 +08:00
|
|
|
#include "clockdomain.h"
|
2010-12-22 12:05:16 +08:00
|
|
|
#include "powerdomain.h"
|
2012-10-06 04:25:59 +08:00
|
|
|
#include "soc.h"
|
2011-11-11 05:45:17 +08:00
|
|
|
#include "common.h"
|
2012-10-21 15:01:11 +08:00
|
|
|
#include "cm3xxx.h"
|
2009-05-29 01:56:16 +08:00
|
|
|
#include "cm-regbits-34xx.h"
|
|
|
|
#include "prm-regbits-34xx.h"
|
2012-10-21 15:01:10 +08:00
|
|
|
#include "prm3xxx.h"
|
2009-05-29 01:56:16 +08:00
|
|
|
#include "pm.h"
|
2008-10-13 18:17:06 +08:00
|
|
|
#include "sdrc.h"
|
2017-11-28 00:57:26 +08:00
|
|
|
#include "omap-secure.h"
|
2012-10-30 04:54:06 +08:00
|
|
|
#include "sram.h"
|
2010-10-09 01:40:20 +08:00
|
|
|
#include "control.h"
|
2014-05-06 08:27:35 +08:00
|
|
|
#include "vc.h"
|
2008-10-13 18:17:06 +08:00
|
|
|
|
2010-12-21 04:05:05 +08:00
|
|
|
/* pm34xx errata defined in pm.h */
|
|
|
|
u16 pm34xx_errata;
|
|
|
|
|
2009-05-29 01:56:16 +08:00
|
|
|
struct power_state {
|
|
|
|
struct powerdomain *pwrdm;
|
|
|
|
u32 next_state;
|
2009-06-25 02:39:18 +08:00
|
|
|
#ifdef CONFIG_SUSPEND
|
2009-05-29 01:56:16 +08:00
|
|
|
u32 saved_state;
|
2009-06-25 02:39:18 +08:00
|
|
|
#endif
|
2009-05-29 01:56:16 +08:00
|
|
|
struct list_head node;
|
|
|
|
};
|
|
|
|
|
|
|
|
static LIST_HEAD(pwrst_list);
|
|
|
|
|
2011-06-30 00:40:23 +08:00
|
|
|
void (*omap3_do_wfi_sram)(void);
|
2008-10-13 18:15:00 +08:00
|
|
|
|
2008-09-26 20:19:22 +08:00
|
|
|
static struct powerdomain *mpu_pwrdm, *neon_pwrdm;
|
|
|
|
static struct powerdomain *core_pwrdm, *per_pwrdm;
|
2009-03-26 21:59:01 +08:00
|
|
|
|
2008-09-26 20:20:07 +08:00
|
|
|
static void omap3_core_save_context(void)
|
|
|
|
{
|
2010-12-22 12:05:16 +08:00
|
|
|
omap3_ctrl_save_padconf();
|
2009-11-18 00:34:53 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Force write last pad into memory, as this can fail in some
|
2010-12-18 23:44:46 +08:00
|
|
|
* cases according to errata 1.157, 1.185
|
2009-11-18 00:34:53 +08:00
|
|
|
*/
|
|
|
|
omap_ctrl_writel(omap_ctrl_readl(OMAP343X_PADCONF_ETK_D14),
|
|
|
|
OMAP343X_CONTROL_MEM_WKUP + 0x2a0);
|
|
|
|
|
2008-09-26 20:20:07 +08:00
|
|
|
/* Save the Interrupt controller context */
|
|
|
|
omap_intc_save_context();
|
|
|
|
/* Save the GPMC context */
|
|
|
|
omap3_gpmc_save_context();
|
|
|
|
/* Save the system control module context, padconf already save above*/
|
|
|
|
omap3_control_save_context();
|
|
|
|
}
|
|
|
|
|
|
|
|
static void omap3_core_restore_context(void)
|
|
|
|
{
|
|
|
|
/* Restore the control module context, padconf restored by h/w */
|
|
|
|
omap3_control_restore_context();
|
|
|
|
/* Restore the GPMC context */
|
|
|
|
omap3_gpmc_restore_context();
|
|
|
|
/* Restore the interrupt controller context */
|
|
|
|
omap_intc_restore_context();
|
|
|
|
}
|
|
|
|
|
2008-12-12 17:20:05 +08:00
|
|
|
/*
|
|
|
|
* FIXME: This function should be called before entering off-mode after
|
|
|
|
* OMAP3 secure services have been accessed. Currently it is only called
|
|
|
|
* once during boot sequence, but this works as we are not using secure
|
|
|
|
* services.
|
|
|
|
*/
|
2011-01-26 08:40:01 +08:00
|
|
|
static void omap3_save_secure_ram_context(void)
|
2008-10-13 18:15:00 +08:00
|
|
|
{
|
|
|
|
u32 ret;
|
2011-01-26 08:40:01 +08:00
|
|
|
int mpu_next_state = pwrdm_read_next_pwrst(mpu_pwrdm);
|
2008-10-13 18:15:00 +08:00
|
|
|
|
|
|
|
if (omap_type() != OMAP2_DEVICE_TYPE_GP) {
|
|
|
|
/*
|
|
|
|
* MPU next state must be set to POWER_ON temporarily,
|
|
|
|
* otherwise the WFI executed inside the ROM code
|
|
|
|
* will hang the system.
|
|
|
|
*/
|
|
|
|
pwrdm_set_next_pwrst(mpu_pwrdm, PWRDM_POWER_ON);
|
2017-11-28 00:57:26 +08:00
|
|
|
ret = omap3_save_secure_ram(omap3_secure_ram_storage,
|
|
|
|
OMAP3_SAVE_SECURE_RAM_SZ);
|
2011-01-26 08:40:01 +08:00
|
|
|
pwrdm_set_next_pwrst(mpu_pwrdm, mpu_next_state);
|
2008-10-13 18:15:00 +08:00
|
|
|
/* Following is for error tracking, it should not happen */
|
|
|
|
if (ret) {
|
2012-03-18 09:22:48 +08:00
|
|
|
pr_err("save_secure_sram() returns %08x\n", ret);
|
2008-10-13 18:15:00 +08:00
|
|
|
while (1)
|
|
|
|
;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-12-17 05:36:59 +08:00
|
|
|
static irqreturn_t _prcm_int_handle_io(int irq, void *unused)
|
2009-07-23 01:29:02 +08:00
|
|
|
{
|
|
|
|
int c;
|
|
|
|
|
2014-04-04 17:31:51 +08:00
|
|
|
c = omap_prm_clear_mod_irqs(WKUP_MOD, 1, OMAP3430_ST_IO_MASK |
|
|
|
|
OMAP3430_ST_IO_CHAIN_MASK);
|
2009-07-23 01:29:02 +08:00
|
|
|
|
2011-12-17 05:36:59 +08:00
|
|
|
return c ? IRQ_HANDLED : IRQ_NONE;
|
2009-06-27 13:07:25 +08:00
|
|
|
}
|
2009-05-29 01:56:16 +08:00
|
|
|
|
2011-12-17 05:36:59 +08:00
|
|
|
static irqreturn_t _prcm_int_handle_wakeup(int irq, void *unused)
|
2009-06-27 13:07:25 +08:00
|
|
|
{
|
2011-12-17 05:36:59 +08:00
|
|
|
int c;
|
2010-04-27 05:59:09 +08:00
|
|
|
|
2011-12-17 05:36:59 +08:00
|
|
|
/*
|
|
|
|
* Clear all except ST_IO and ST_IO_CHAIN for wkup module,
|
|
|
|
* these are handled in a separate handler to avoid acking
|
|
|
|
* IO events before parsing in mux code
|
|
|
|
*/
|
2014-04-04 17:31:51 +08:00
|
|
|
c = omap_prm_clear_mod_irqs(WKUP_MOD, 1, ~(OMAP3430_ST_IO_MASK |
|
|
|
|
OMAP3430_ST_IO_CHAIN_MASK));
|
|
|
|
c += omap_prm_clear_mod_irqs(CORE_MOD, 1, ~0);
|
|
|
|
c += omap_prm_clear_mod_irqs(OMAP3430_PER_MOD, 1, ~0);
|
2011-12-17 05:36:59 +08:00
|
|
|
if (omap_rev() > OMAP3430_REV_ES1_0) {
|
2014-04-04 17:31:51 +08:00
|
|
|
c += omap_prm_clear_mod_irqs(CORE_MOD, 3, ~0);
|
|
|
|
c += omap_prm_clear_mod_irqs(OMAP3430ES2_USBHOST_MOD, 1, ~0);
|
2011-12-17 05:36:59 +08:00
|
|
|
}
|
2009-05-29 01:56:16 +08:00
|
|
|
|
2011-12-17 05:36:59 +08:00
|
|
|
return c ? IRQ_HANDLED : IRQ_NONE;
|
2009-05-29 01:56:16 +08:00
|
|
|
}
|
|
|
|
|
2011-06-30 15:45:49 +08:00
|
|
|
static void omap34xx_save_context(u32 *save)
|
|
|
|
{
|
|
|
|
u32 val;
|
|
|
|
|
|
|
|
/* Read Auxiliary Control Register */
|
|
|
|
asm("mrc p15, 0, %0, c1, c0, 1" : "=r" (val));
|
|
|
|
*save++ = 1;
|
|
|
|
*save++ = val;
|
|
|
|
|
|
|
|
/* Read L2 AUX ctrl register */
|
|
|
|
asm("mrc p15, 1, %0, c9, c0, 2" : "=r" (val));
|
|
|
|
*save++ = 1;
|
|
|
|
*save++ = val;
|
|
|
|
}
|
|
|
|
|
2011-07-02 16:54:01 +08:00
|
|
|
static int omap34xx_do_sram_idle(unsigned long save_state)
|
2008-09-26 20:19:34 +08:00
|
|
|
{
|
2011-06-30 15:45:49 +08:00
|
|
|
omap34xx_cpu_suspend(save_state);
|
2011-07-02 16:54:01 +08:00
|
|
|
return 0;
|
2008-09-26 20:19:34 +08:00
|
|
|
}
|
|
|
|
|
2008-10-08 20:00:58 +08:00
|
|
|
void omap_sram_idle(void)
|
2009-05-29 01:56:16 +08:00
|
|
|
{
|
|
|
|
/* Variable to tell what needs to be saved and restored
|
|
|
|
* in omap_sram_idle*/
|
|
|
|
/* save_state = 0 => Nothing to save and restored */
|
|
|
|
/* save_state = 1 => Only L1 and logic lost */
|
|
|
|
/* save_state = 2 => Only L2 lost */
|
|
|
|
/* save_state = 3 => L1, L2 and logic lost */
|
2008-09-26 20:19:22 +08:00
|
|
|
int save_state = 0;
|
|
|
|
int mpu_next_state = PWRDM_POWER_ON;
|
|
|
|
int per_next_state = PWRDM_POWER_ON;
|
|
|
|
int core_next_state = PWRDM_POWER_ON;
|
2008-10-13 18:17:06 +08:00
|
|
|
u32 sdrc_pwr = 0;
|
2020-03-05 06:54:30 +08:00
|
|
|
int error;
|
2009-05-29 01:56:16 +08:00
|
|
|
|
|
|
|
mpu_next_state = pwrdm_read_next_pwrst(mpu_pwrdm);
|
|
|
|
switch (mpu_next_state) {
|
2008-09-26 20:19:22 +08:00
|
|
|
case PWRDM_POWER_ON:
|
2009-05-29 01:56:16 +08:00
|
|
|
case PWRDM_POWER_RET:
|
|
|
|
/* No need to save context */
|
|
|
|
save_state = 0;
|
|
|
|
break;
|
2008-09-26 20:19:56 +08:00
|
|
|
case PWRDM_POWER_OFF:
|
|
|
|
save_state = 3;
|
|
|
|
break;
|
2009-05-29 01:56:16 +08:00
|
|
|
default:
|
|
|
|
/* Invalid state */
|
2012-03-18 09:22:48 +08:00
|
|
|
pr_err("Invalid mpu state in sram_idle\n");
|
2009-05-29 01:56:16 +08:00
|
|
|
return;
|
|
|
|
}
|
2008-10-15 22:48:44 +08:00
|
|
|
|
2008-09-26 20:19:22 +08:00
|
|
|
/* NEON control */
|
|
|
|
if (pwrdm_read_pwrst(neon_pwrdm) == PWRDM_POWER_ON)
|
2008-10-28 16:59:05 +08:00
|
|
|
pwrdm_set_next_pwrst(neon_pwrdm, mpu_next_state);
|
2008-09-26 20:19:22 +08:00
|
|
|
|
2010-05-04 07:04:06 +08:00
|
|
|
/* Enable IO-PAD and IO-CHAIN wakeups */
|
2008-11-05 12:50:52 +08:00
|
|
|
per_next_state = pwrdm_read_next_pwrst(per_pwrdm);
|
2008-12-01 19:17:29 +08:00
|
|
|
core_next_state = pwrdm_read_next_pwrst(core_pwrdm);
|
2010-05-04 07:04:06 +08:00
|
|
|
|
2012-08-08 02:28:06 +08:00
|
|
|
pwrdm_pre_transition(NULL);
|
2011-09-13 21:02:37 +08:00
|
|
|
|
2010-05-04 07:04:06 +08:00
|
|
|
/* PER */
|
2020-03-05 06:54:30 +08:00
|
|
|
if (per_next_state == PWRDM_POWER_OFF) {
|
|
|
|
error = cpu_cluster_pm_enter();
|
|
|
|
if (error)
|
|
|
|
return;
|
|
|
|
}
|
2008-11-05 12:50:52 +08:00
|
|
|
|
|
|
|
/* CORE */
|
2008-09-26 20:19:22 +08:00
|
|
|
if (core_next_state < PWRDM_POWER_ON) {
|
2008-09-26 20:20:07 +08:00
|
|
|
if (core_next_state == PWRDM_POWER_OFF) {
|
|
|
|
omap3_core_save_context();
|
2010-12-22 06:30:56 +08:00
|
|
|
omap3_cm_save_context();
|
2008-09-26 20:20:07 +08:00
|
|
|
}
|
2008-09-26 20:19:22 +08:00
|
|
|
}
|
2010-05-04 07:04:06 +08:00
|
|
|
|
2014-05-06 08:27:35 +08:00
|
|
|
/* Configure PMIC signaling for I2C4 or sys_off_mode */
|
|
|
|
omap3_vc_set_pmic_signaling(core_next_state);
|
|
|
|
|
2009-10-24 00:03:50 +08:00
|
|
|
omap3_intc_prepare_idle();
|
2009-05-29 01:56:16 +08:00
|
|
|
|
2008-10-13 18:17:06 +08:00
|
|
|
/*
|
2011-10-07 03:43:23 +08:00
|
|
|
* On EMU/HS devices ROM code restores a SRDC value
|
|
|
|
* from scratchpad which has automatic self refresh on timeout
|
|
|
|
* of AUTO_CNT = 1 enabled. This takes care of erratum ID i443.
|
|
|
|
* Hence store/restore the SDRC_POWER register here.
|
|
|
|
*/
|
|
|
|
if (cpu_is_omap3430() && omap_rev() >= OMAP3430_REV_ES3_0 &&
|
|
|
|
(omap_type() == OMAP2_DEVICE_TYPE_EMU ||
|
|
|
|
omap_type() == OMAP2_DEVICE_TYPE_SEC) &&
|
2009-06-10 01:00:41 +08:00
|
|
|
core_next_state == PWRDM_POWER_OFF)
|
2008-10-13 18:17:06 +08:00
|
|
|
sdrc_pwr = sdrc_read_reg(SDRC_POWER);
|
|
|
|
|
2008-09-26 20:19:56 +08:00
|
|
|
/*
|
2011-06-22 22:42:54 +08:00
|
|
|
* omap3_arm_context is the location where some ARM context
|
|
|
|
* get saved. The rest is placed on the stack, and restored
|
|
|
|
* from there before resuming.
|
2008-09-26 20:19:56 +08:00
|
|
|
*/
|
2011-06-30 15:45:49 +08:00
|
|
|
if (save_state)
|
|
|
|
omap34xx_save_context(omap3_arm_context);
|
2011-06-22 22:42:54 +08:00
|
|
|
if (save_state == 1 || save_state == 3)
|
2011-06-23 00:41:48 +08:00
|
|
|
cpu_suspend(save_state, omap34xx_do_sram_idle);
|
2011-06-22 22:42:54 +08:00
|
|
|
else
|
|
|
|
omap34xx_do_sram_idle(save_state);
|
2009-05-29 01:56:16 +08:00
|
|
|
|
2009-06-10 01:00:41 +08:00
|
|
|
/* Restore normal SDRC POWER settings */
|
2011-10-07 03:43:23 +08:00
|
|
|
if (cpu_is_omap3430() && omap_rev() >= OMAP3430_REV_ES3_0 &&
|
|
|
|
(omap_type() == OMAP2_DEVICE_TYPE_EMU ||
|
|
|
|
omap_type() == OMAP2_DEVICE_TYPE_SEC) &&
|
2008-10-13 18:17:06 +08:00
|
|
|
core_next_state == PWRDM_POWER_OFF)
|
|
|
|
sdrc_write_reg(sdrc_pwr, SDRC_POWER);
|
|
|
|
|
2008-11-05 12:50:52 +08:00
|
|
|
/* CORE */
|
2016-04-14 09:49:48 +08:00
|
|
|
if (core_next_state < PWRDM_POWER_ON &&
|
|
|
|
pwrdm_read_prev_pwrst(core_pwrdm) == PWRDM_POWER_OFF) {
|
|
|
|
omap3_core_restore_context();
|
|
|
|
omap3_cm_restore_context();
|
|
|
|
omap3_sram_restore_context();
|
|
|
|
omap2_sms_restore_context();
|
|
|
|
} else {
|
|
|
|
/*
|
|
|
|
* In off-mode resume path above, omap3_core_restore_context
|
|
|
|
* also handles the INTC autoidle restore done here so limit
|
|
|
|
* this to non-off mode resume paths so we don't do it twice.
|
|
|
|
*/
|
|
|
|
omap3_intc_resume_idle();
|
2008-11-05 12:50:52 +08:00
|
|
|
}
|
|
|
|
|
2012-08-08 02:28:06 +08:00
|
|
|
pwrdm_post_transition(NULL);
|
|
|
|
|
2008-11-05 12:50:52 +08:00
|
|
|
/* PER */
|
2018-09-21 03:35:31 +08:00
|
|
|
if (per_next_state == PWRDM_POWER_OFF)
|
|
|
|
cpu_cluster_pm_exit();
|
2009-05-29 01:56:16 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void omap3_pm_idle(void)
|
|
|
|
{
|
2012-01-05 05:27:48 +08:00
|
|
|
if (omap_irq_pending())
|
2013-02-11 21:59:45 +08:00
|
|
|
return;
|
2009-05-29 01:56:16 +08:00
|
|
|
|
2015-09-18 13:41:21 +08:00
|
|
|
trace_cpu_idle_rcuidle(1, smp_processor_id());
|
2011-03-03 18:25:43 +08:00
|
|
|
|
2009-05-29 01:56:16 +08:00
|
|
|
omap_sram_idle();
|
|
|
|
|
2015-09-18 13:41:21 +08:00
|
|
|
trace_cpu_idle_rcuidle(PWR_EVENT_EXIT, smp_processor_id());
|
2009-05-29 01:56:16 +08:00
|
|
|
}
|
|
|
|
|
2009-06-25 02:39:18 +08:00
|
|
|
#ifdef CONFIG_SUSPEND
|
2009-05-29 01:56:16 +08:00
|
|
|
static int omap3_pm_suspend(void)
|
|
|
|
{
|
|
|
|
struct power_state *pwrst;
|
|
|
|
int state, ret = 0;
|
|
|
|
|
|
|
|
/* Read current next_pwrsts */
|
|
|
|
list_for_each_entry(pwrst, &pwrst_list, node)
|
|
|
|
pwrst->saved_state = pwrdm_read_next_pwrst(pwrst->pwrdm);
|
|
|
|
/* Set ones wanted by suspend */
|
|
|
|
list_for_each_entry(pwrst, &pwrst_list, node) {
|
2010-09-15 03:34:01 +08:00
|
|
|
if (omap_set_pwrdm_state(pwrst->pwrdm, pwrst->next_state))
|
2009-05-29 01:56:16 +08:00
|
|
|
goto restore;
|
|
|
|
if (pwrdm_clear_all_prev_pwrst(pwrst->pwrdm))
|
|
|
|
goto restore;
|
|
|
|
}
|
|
|
|
|
2009-10-24 00:03:48 +08:00
|
|
|
omap3_intc_suspend();
|
|
|
|
|
2009-05-29 01:56:16 +08:00
|
|
|
omap_sram_idle();
|
|
|
|
|
|
|
|
restore:
|
|
|
|
/* Restore next_pwrsts */
|
|
|
|
list_for_each_entry(pwrst, &pwrst_list, node) {
|
|
|
|
state = pwrdm_read_prev_pwrst(pwrst->pwrdm);
|
|
|
|
if (state > pwrst->next_state) {
|
2012-07-26 14:54:26 +08:00
|
|
|
pr_info("Powerdomain (%s) didn't enter target state %d\n",
|
|
|
|
pwrst->pwrdm->name, pwrst->next_state);
|
2009-05-29 01:56:16 +08:00
|
|
|
ret = -1;
|
|
|
|
}
|
2010-09-15 03:34:01 +08:00
|
|
|
omap_set_pwrdm_state(pwrst->pwrdm, pwrst->saved_state);
|
2009-05-29 01:56:16 +08:00
|
|
|
}
|
|
|
|
if (ret)
|
2012-03-18 09:22:48 +08:00
|
|
|
pr_err("Could not enter target state in pm_suspend\n");
|
2009-05-29 01:56:16 +08:00
|
|
|
else
|
2012-03-18 09:22:48 +08:00
|
|
|
pr_info("Successfully put all powerdomains to target state\n");
|
2009-05-29 01:56:16 +08:00
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
2014-05-13 02:33:21 +08:00
|
|
|
#else
|
|
|
|
#define omap3_pm_suspend NULL
|
2009-06-25 02:39:18 +08:00
|
|
|
#endif /* CONFIG_SUSPEND */
|
2009-05-29 01:56:16 +08:00
|
|
|
|
2009-04-29 06:27:44 +08:00
|
|
|
static void __init prcm_setup_regs(void)
|
|
|
|
{
|
2014-03-04 23:43:04 +08:00
|
|
|
omap3_ctrl_init();
|
2009-10-24 00:03:49 +08:00
|
|
|
|
2014-02-26 23:30:43 +08:00
|
|
|
omap3_prm_init_pm(cpu_is_omap3630(), omap3_has_iva());
|
2009-05-29 01:56:16 +08:00
|
|
|
}
|
|
|
|
|
2009-10-07 05:25:09 +08:00
|
|
|
void omap3_pm_off_mode_enable(int enable)
|
|
|
|
{
|
|
|
|
struct power_state *pwrst;
|
|
|
|
u32 state;
|
|
|
|
|
|
|
|
if (enable)
|
|
|
|
state = PWRDM_POWER_OFF;
|
|
|
|
else
|
|
|
|
state = PWRDM_POWER_RET;
|
|
|
|
|
|
|
|
list_for_each_entry(pwrst, &pwrst_list, node) {
|
2010-12-21 04:05:09 +08:00
|
|
|
if (IS_PM34XX_ERRATUM(PM_SDRC_WAKEUP_ERRATUM_i583) &&
|
|
|
|
pwrst->pwrdm == core_pwrdm &&
|
|
|
|
state == PWRDM_POWER_OFF) {
|
|
|
|
pwrst->next_state = PWRDM_POWER_RET;
|
2011-01-31 21:35:25 +08:00
|
|
|
pr_warn("%s: Core OFF disabled due to errata i583\n",
|
2010-12-21 04:05:09 +08:00
|
|
|
__func__);
|
|
|
|
} else {
|
|
|
|
pwrst->next_state = state;
|
|
|
|
}
|
|
|
|
omap_set_pwrdm_state(pwrst->pwrdm, pwrst->next_state);
|
2009-10-07 05:25:09 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-11-26 18:26:24 +08:00
|
|
|
int omap3_pm_get_suspend_state(struct powerdomain *pwrdm)
|
|
|
|
{
|
|
|
|
struct power_state *pwrst;
|
|
|
|
|
|
|
|
list_for_each_entry(pwrst, &pwrst_list, node) {
|
|
|
|
if (pwrst->pwrdm == pwrdm)
|
|
|
|
return pwrst->next_state;
|
|
|
|
}
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
int omap3_pm_set_suspend_state(struct powerdomain *pwrdm, int state)
|
|
|
|
{
|
|
|
|
struct power_state *pwrst;
|
|
|
|
|
|
|
|
list_for_each_entry(pwrst, &pwrst_list, node) {
|
|
|
|
if (pwrst->pwrdm == pwrdm) {
|
|
|
|
pwrst->next_state = state;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2008-10-15 23:13:47 +08:00
|
|
|
static int __init pwrdms_setup(struct powerdomain *pwrdm, void *unused)
|
2009-05-29 01:56:16 +08:00
|
|
|
{
|
|
|
|
struct power_state *pwrst;
|
|
|
|
|
|
|
|
if (!pwrdm->pwrsts)
|
|
|
|
return 0;
|
|
|
|
|
2009-08-22 21:20:26 +08:00
|
|
|
pwrst = kmalloc(sizeof(struct power_state), GFP_ATOMIC);
|
2009-05-29 01:56:16 +08:00
|
|
|
if (!pwrst)
|
|
|
|
return -ENOMEM;
|
|
|
|
pwrst->pwrdm = pwrdm;
|
|
|
|
pwrst->next_state = PWRDM_POWER_RET;
|
|
|
|
list_add(&pwrst->node, &pwrst_list);
|
|
|
|
|
|
|
|
if (pwrdm_has_hdwr_sar(pwrdm))
|
|
|
|
pwrdm_enable_hdwr_sar(pwrdm);
|
|
|
|
|
2010-09-15 03:34:01 +08:00
|
|
|
return omap_set_pwrdm_state(pwrst->pwrdm, pwrst->next_state);
|
2009-05-29 01:56:16 +08:00
|
|
|
}
|
|
|
|
|
2011-06-30 00:40:23 +08:00
|
|
|
/*
|
|
|
|
* Push functions to SRAM
|
|
|
|
*
|
|
|
|
* The minimum set of functions is pushed to SRAM for execution:
|
|
|
|
* - omap3_do_wfi for erratum i581 WA,
|
|
|
|
*/
|
2008-09-26 20:19:14 +08:00
|
|
|
void omap_push_sram_idle(void)
|
|
|
|
{
|
2011-06-30 00:40:23 +08:00
|
|
|
omap3_do_wfi_sram = omap_sram_push(omap3_do_wfi, omap3_do_wfi_sz);
|
2008-09-26 20:19:14 +08:00
|
|
|
}
|
|
|
|
|
2010-12-21 04:05:05 +08:00
|
|
|
static void __init pm_errata_configure(void)
|
|
|
|
{
|
2010-12-21 04:05:07 +08:00
|
|
|
if (cpu_is_omap3630()) {
|
2010-12-21 04:05:06 +08:00
|
|
|
pm34xx_errata |= PM_RTA_ERRATUM_i608;
|
2010-12-21 04:05:07 +08:00
|
|
|
/* Enable the l2 cache toggling in sleep logic */
|
|
|
|
enable_omap3630_toggle_l2_on_restore();
|
2010-12-21 04:05:09 +08:00
|
|
|
if (omap_rev() < OMAP3630_REV_ES1_2)
|
2012-10-16 14:08:53 +08:00
|
|
|
pm34xx_errata |= (PM_SDRC_WAKEUP_ERRATUM_i583 |
|
|
|
|
PM_PER_MEMORIES_ERRATUM_i582);
|
|
|
|
} else if (cpu_is_omap34xx()) {
|
|
|
|
pm34xx_errata |= PM_PER_MEMORIES_ERRATUM_i582;
|
2010-12-21 04:05:07 +08:00
|
|
|
}
|
2010-12-21 04:05:05 +08:00
|
|
|
}
|
|
|
|
|
2012-04-26 16:06:50 +08:00
|
|
|
int __init omap3_pm_init(void)
|
2009-05-29 01:56:16 +08:00
|
|
|
{
|
|
|
|
struct power_state *pwrst, *tmp;
|
2012-10-16 14:08:53 +08:00
|
|
|
struct clockdomain *neon_clkdm, *mpu_clkdm, *per_clkdm, *wkup_clkdm;
|
2009-05-29 01:56:16 +08:00
|
|
|
int ret;
|
|
|
|
|
ARM: OMAP3: PM: fix I/O wakeup and I/O chain clock control detection
The way that we detect which OMAP3 chips support I/O wakeup and
software I/O chain clock control is broken.
Currently, I/O wakeup is marked as present for all OMAP3 SoCs other
than the AM3505/3517. The TI81xx family of SoCs are at present
considered to be OMAP3 SoCs, but don't support I/O wakeup. To resolve
this, convert the existing blacklist approach to an explicit,
whitelist support, in which only SoCs which are known to support I/O
wakeup are listed. (At present, this only includes OMAP34xx,
OMAP3503, OMAP3515, OMAP3525, OMAP3530, and OMAP36xx.)
Also, the current code incorrectly detects the presence of a
software-controllable I/O chain clock on several chips that don't
support it. This results in writes to reserved bitfields, unnecessary
delays, and console messages on kernels running on those chips:
http://www.spinics.net/lists/linux-omap/msg58735.html
Convert this test to a feature test with a chip-by-chip whitelist.
Thanks to Dave Hylands <dhylands@gmail.com> for reporting this problem
and doing some testing to help isolate the cause. Thanks to Steve
Sakoman <sakoman@gmail.com> for catching a bug in the first version of
this patch. Thanks to Russell King <linux@arm.linux.org.uk> for
comments.
Signed-off-by: Paul Walmsley <paul@pwsan.com>
Cc: Dave Hylands <dhylands@gmail.com>
Cc: Steve Sakoman <sakoman@gmail.com>
Tested-by: Steve Sakoman <sakoman@gmail.com>
Cc: Russell King - ARM Linux <linux@arm.linux.org.uk>
Signed-off-by: Kevin Hilman <khilman@ti.com>
2011-10-07 07:18:45 +08:00
|
|
|
if (!omap3_has_io_chain_ctrl())
|
2014-09-14 02:31:16 +08:00
|
|
|
pr_warn("PM: no software I/O chain control; some wakeups may be lost\n");
|
ARM: OMAP3: PM: fix I/O wakeup and I/O chain clock control detection
The way that we detect which OMAP3 chips support I/O wakeup and
software I/O chain clock control is broken.
Currently, I/O wakeup is marked as present for all OMAP3 SoCs other
than the AM3505/3517. The TI81xx family of SoCs are at present
considered to be OMAP3 SoCs, but don't support I/O wakeup. To resolve
this, convert the existing blacklist approach to an explicit,
whitelist support, in which only SoCs which are known to support I/O
wakeup are listed. (At present, this only includes OMAP34xx,
OMAP3503, OMAP3515, OMAP3525, OMAP3530, and OMAP36xx.)
Also, the current code incorrectly detects the presence of a
software-controllable I/O chain clock on several chips that don't
support it. This results in writes to reserved bitfields, unnecessary
delays, and console messages on kernels running on those chips:
http://www.spinics.net/lists/linux-omap/msg58735.html
Convert this test to a feature test with a chip-by-chip whitelist.
Thanks to Dave Hylands <dhylands@gmail.com> for reporting this problem
and doing some testing to help isolate the cause. Thanks to Steve
Sakoman <sakoman@gmail.com> for catching a bug in the first version of
this patch. Thanks to Russell King <linux@arm.linux.org.uk> for
comments.
Signed-off-by: Paul Walmsley <paul@pwsan.com>
Cc: Dave Hylands <dhylands@gmail.com>
Cc: Steve Sakoman <sakoman@gmail.com>
Tested-by: Steve Sakoman <sakoman@gmail.com>
Cc: Russell King - ARM Linux <linux@arm.linux.org.uk>
Signed-off-by: Kevin Hilman <khilman@ti.com>
2011-10-07 07:18:45 +08:00
|
|
|
|
2010-12-21 04:05:05 +08:00
|
|
|
pm_errata_configure();
|
|
|
|
|
2009-05-29 01:56:16 +08:00
|
|
|
/* XXX prcm_setup_regs needs to be before enabling hw
|
|
|
|
* supervised mode for powerdomains */
|
|
|
|
prcm_setup_regs();
|
|
|
|
|
2011-12-17 05:36:59 +08:00
|
|
|
ret = request_irq(omap_prcm_event_to_irq("wkup"),
|
|
|
|
_prcm_int_handle_wakeup, IRQF_NO_SUSPEND, "pm_wkup", NULL);
|
|
|
|
|
|
|
|
if (ret) {
|
|
|
|
pr_err("pm: Failed to request pm_wkup irq\n");
|
|
|
|
goto err1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* IO interrupt is shared with mux code */
|
|
|
|
ret = request_irq(omap_prcm_event_to_irq("io"),
|
|
|
|
_prcm_int_handle_io, IRQF_SHARED | IRQF_NO_SUSPEND, "pm_io",
|
|
|
|
omap3_pm_init);
|
|
|
|
|
2009-05-29 01:56:16 +08:00
|
|
|
if (ret) {
|
2011-12-17 05:36:59 +08:00
|
|
|
pr_err("pm: Failed to request pm_io irq\n");
|
2012-03-18 09:22:47 +08:00
|
|
|
goto err2;
|
2009-05-29 01:56:16 +08:00
|
|
|
}
|
|
|
|
|
2008-10-15 23:13:47 +08:00
|
|
|
ret = pwrdm_for_each(pwrdms_setup, NULL);
|
2009-05-29 01:56:16 +08:00
|
|
|
if (ret) {
|
2012-03-18 09:22:48 +08:00
|
|
|
pr_err("Failed to setup powerdomains\n");
|
2012-03-18 09:22:47 +08:00
|
|
|
goto err3;
|
2009-05-29 01:56:16 +08:00
|
|
|
}
|
|
|
|
|
2012-02-02 17:38:50 +08:00
|
|
|
(void) clkdm_for_each(omap_pm_clkdms_setup, NULL);
|
2009-05-29 01:56:16 +08:00
|
|
|
|
|
|
|
mpu_pwrdm = pwrdm_lookup("mpu_pwrdm");
|
|
|
|
if (mpu_pwrdm == NULL) {
|
2012-03-18 09:22:48 +08:00
|
|
|
pr_err("Failed to get mpu_pwrdm\n");
|
2012-03-18 09:22:47 +08:00
|
|
|
ret = -EINVAL;
|
|
|
|
goto err3;
|
2009-05-29 01:56:16 +08:00
|
|
|
}
|
|
|
|
|
2008-09-26 20:19:22 +08:00
|
|
|
neon_pwrdm = pwrdm_lookup("neon_pwrdm");
|
|
|
|
per_pwrdm = pwrdm_lookup("per_pwrdm");
|
|
|
|
core_pwrdm = pwrdm_lookup("core_pwrdm");
|
|
|
|
|
2010-01-27 11:12:59 +08:00
|
|
|
neon_clkdm = clkdm_lookup("neon_clkdm");
|
|
|
|
mpu_clkdm = clkdm_lookup("mpu_clkdm");
|
2012-10-16 14:08:53 +08:00
|
|
|
per_clkdm = clkdm_lookup("per_clkdm");
|
|
|
|
wkup_clkdm = clkdm_lookup("wkup_clkdm");
|
2010-01-27 11:12:59 +08:00
|
|
|
|
2014-05-13 02:33:21 +08:00
|
|
|
omap_common_suspend_init(omap3_pm_suspend);
|
2009-05-29 01:56:16 +08:00
|
|
|
|
2012-01-05 05:27:48 +08:00
|
|
|
arm_pm_idle = omap3_pm_idle;
|
2008-09-26 16:04:20 +08:00
|
|
|
omap3_idle_init();
|
2009-05-29 01:56:16 +08:00
|
|
|
|
2010-12-21 04:05:06 +08:00
|
|
|
/*
|
|
|
|
* RTA is disabled during initialization as per erratum i608
|
|
|
|
* it is safer to disable RTA by the bootloader, but we would like
|
|
|
|
* to be doubly sure here and prevent any mishaps.
|
|
|
|
*/
|
|
|
|
if (IS_PM34XX_ERRATUM(PM_RTA_ERRATUM_i608))
|
|
|
|
omap3630_ctrl_disable_rta();
|
|
|
|
|
2012-10-16 14:08:53 +08:00
|
|
|
/*
|
|
|
|
* The UART3/4 FIFO and the sidetone memory in McBSP2/3 are
|
|
|
|
* not correctly reset when the PER powerdomain comes back
|
|
|
|
* from OFF or OSWR when the CORE powerdomain is kept active.
|
|
|
|
* See OMAP36xx Erratum i582 "PER Domain reset issue after
|
|
|
|
* Domain-OFF/OSWR Wakeup". This wakeup dependency is not a
|
|
|
|
* complete workaround. The kernel must also prevent the PER
|
|
|
|
* powerdomain from going to OSWR/OFF while the CORE
|
|
|
|
* powerdomain is not going to OSWR/OFF. And if PER last
|
|
|
|
* power state was off while CORE last power state was ON, the
|
|
|
|
* UART3/4 and McBSP2/3 SIDETONE devices need to run a
|
|
|
|
* self-test using their loopback tests; if that fails, those
|
|
|
|
* devices are unusable until the PER/CORE can complete a transition
|
|
|
|
* from ON to OSWR/OFF and then back to ON.
|
|
|
|
*
|
|
|
|
* XXX Technically this workaround is only needed if off-mode
|
|
|
|
* or OSWR is enabled.
|
|
|
|
*/
|
|
|
|
if (IS_PM34XX_ERRATUM(PM_PER_MEMORIES_ERRATUM_i582))
|
|
|
|
clkdm_add_wkdep(per_clkdm, wkup_clkdm);
|
|
|
|
|
2010-01-27 11:12:59 +08:00
|
|
|
clkdm_add_wkdep(neon_clkdm, mpu_clkdm);
|
2008-10-13 18:15:00 +08:00
|
|
|
if (omap_type() != OMAP2_DEVICE_TYPE_GP) {
|
|
|
|
omap3_secure_ram_storage =
|
2017-11-28 00:57:26 +08:00
|
|
|
kmalloc(OMAP3_SAVE_SECURE_RAM_SZ, GFP_KERNEL);
|
2008-10-13 18:15:00 +08:00
|
|
|
if (!omap3_secure_ram_storage)
|
2012-07-26 14:54:26 +08:00
|
|
|
pr_err("Memory allocation failed when allocating for secure sram context\n");
|
2008-12-12 17:20:05 +08:00
|
|
|
|
|
|
|
local_irq_disable();
|
|
|
|
|
2011-01-26 08:40:01 +08:00
|
|
|
omap3_save_secure_ram_context();
|
2008-12-12 17:20:05 +08:00
|
|
|
|
|
|
|
local_irq_enable();
|
2008-10-13 18:15:00 +08:00
|
|
|
}
|
|
|
|
|
2008-12-12 17:20:05 +08:00
|
|
|
omap3_save_scratchpad_contents();
|
2009-05-29 01:56:16 +08:00
|
|
|
return ret;
|
2012-03-18 09:22:47 +08:00
|
|
|
|
|
|
|
err3:
|
2009-05-29 01:56:16 +08:00
|
|
|
list_for_each_entry_safe(pwrst, tmp, &pwrst_list, node) {
|
|
|
|
list_del(&pwrst->node);
|
|
|
|
kfree(pwrst);
|
|
|
|
}
|
2012-03-18 09:22:47 +08:00
|
|
|
free_irq(omap_prcm_event_to_irq("io"), omap3_pm_init);
|
|
|
|
err2:
|
|
|
|
free_irq(omap_prcm_event_to_irq("wkup"), NULL);
|
|
|
|
err1:
|
2009-05-29 01:56:16 +08:00
|
|
|
return ret;
|
|
|
|
}
|