diff options
Diffstat (limited to 'arch/arm/mach-shmobile/pm-sh7372.c')
| -rw-r--r-- | arch/arm/mach-shmobile/pm-sh7372.c | 342 | 
1 files changed, 274 insertions, 68 deletions
| diff --git a/arch/arm/mach-shmobile/pm-sh7372.c b/arch/arm/mach-shmobile/pm-sh7372.c index 933fb411be0f..79612737c5b2 100644 --- a/arch/arm/mach-shmobile/pm-sh7372.c +++ b/arch/arm/mach-shmobile/pm-sh7372.c @@ -15,23 +15,61 @@  #include <linux/list.h>  #include <linux/err.h>  #include <linux/slab.h> -#include <linux/pm_runtime.h> +#include <linux/pm_clock.h>  #include <linux/platform_device.h>  #include <linux/delay.h> +#include <linux/irq.h> +#include <linux/bitrev.h>  #include <asm/system.h>  #include <asm/io.h>  #include <asm/tlbflush.h> +#include <asm/suspend.h>  #include <mach/common.h>  #include <mach/sh7372.h> -#define SMFRAM 0xe6a70000 -#define SYSTBCR 0xe6150024 -#define SBAR 0xe6180020 -#define APARMBAREA 0xe6f10020 +/* DBG */ +#define DBGREG1 0xe6100020 +#define DBGREG9 0xe6100040 +/* CPGA */ +#define SYSTBCR 0xe6150024 +#define MSTPSR0 0xe6150030 +#define MSTPSR1 0xe6150038 +#define MSTPSR2 0xe6150040 +#define MSTPSR3 0xe6150048 +#define MSTPSR4 0xe615004c +#define PLLC01STPCR 0xe61500c8 + +/* SYSC */  #define SPDCR 0xe6180008  #define SWUCR 0xe6180014 +#define SBAR 0xe6180020 +#define WUPRMSK 0xe6180028 +#define WUPSMSK 0xe618002c +#define WUPSMSK2 0xe6180048  #define PSTR 0xe6180080 +#define WUPSFAC 0xe6180098 +#define IRQCR 0xe618022c +#define IRQCR2 0xe6180238 +#define IRQCR3 0xe6180244 +#define IRQCR4 0xe6180248 +#define PDNSEL 0xe6180254 + +/* INTC */ +#define ICR1A 0xe6900000 +#define ICR2A 0xe6900004 +#define ICR3A 0xe6900008 +#define ICR4A 0xe690000c +#define INTMSK00A 0xe6900040 +#define INTMSK10A 0xe6900044 +#define INTMSK20A 0xe6900048 +#define INTMSK30A 0xe690004c + +/* MFIS */ +#define SMFRAM 0xe6a70000 + +/* AP-System Core */ +#define APARMBAREA 0xe6f10020  #define PSTR_RETRIES 100  #define PSTR_DELAY_US 10 @@ -43,6 +81,12 @@ static int pd_power_down(struct generic_pm_domain *genpd)  	struct sh7372_pm_domain *sh7372_pd = to_sh7372_pd(genpd);  	unsigned int mask = 1 << sh7372_pd->bit_shift; +	if (sh7372_pd->suspend) +		sh7372_pd->suspend(); + +	if (sh7372_pd->stay_on) +		return 0; +  	if (__raw_readl(PSTR) & mask) {  		unsigned int retry_count; @@ -55,8 +99,9 @@ static int pd_power_down(struct generic_pm_domain *genpd)  		}  	} -	pr_debug("sh7372 power domain down 0x%08x -> PSTR = 0x%08x\n", -		 mask, __raw_readl(PSTR)); +	if (!sh7372_pd->no_debug) +		pr_debug("sh7372 power domain down 0x%08x -> PSTR = 0x%08x\n", +			 mask, __raw_readl(PSTR));  	return 0;  } @@ -68,6 +113,9 @@ static int pd_power_up(struct generic_pm_domain *genpd)  	unsigned int retry_count;  	int ret = 0; +	if (sh7372_pd->stay_on) +		goto out; +  	if (__raw_readl(PSTR) & mask)  		goto out; @@ -84,66 +132,48 @@ static int pd_power_up(struct generic_pm_domain *genpd)  	if (__raw_readl(SWUCR) & mask)  		ret = -EIO; +	if (!sh7372_pd->no_debug) +		pr_debug("sh7372 power domain up 0x%08x -> PSTR = 0x%08x\n", +			 mask, __raw_readl(PSTR)); +   out: -	pr_debug("sh7372 power domain up 0x%08x -> PSTR = 0x%08x\n", -		 mask, __raw_readl(PSTR)); +	if (ret == 0 && sh7372_pd->resume) +		sh7372_pd->resume();  	return ret;  } -static int pd_power_up_a3rv(struct generic_pm_domain *genpd) +static void sh7372_a4r_suspend(void)  { -	int ret = pd_power_up(genpd); - -	/* force A4LC on after A3RV has been requested on */ -	pm_genpd_poweron(&sh7372_a4lc.genpd); - -	return ret; +	sh7372_intcs_suspend(); +	__raw_writel(0x300fffff, WUPRMSK); /* avoid wakeup */  } -static int pd_power_down_a3rv(struct generic_pm_domain *genpd) +static bool pd_active_wakeup(struct device *dev)  { -	int ret = pd_power_down(genpd); - -	/* try to power down A4LC after A3RV is requested off */ -	genpd_queue_power_off_work(&sh7372_a4lc.genpd); - -	return ret; +	return true;  } -static int pd_power_down_a4lc(struct generic_pm_domain *genpd) +static bool sh7372_power_down_forbidden(struct dev_pm_domain *domain)  { -	/* only power down A4LC if A3RV is off */ -	if (!(__raw_readl(PSTR) & (1 << sh7372_a3rv.bit_shift))) -		return pd_power_down(genpd); - -	return -EBUSY; +	return false;  } -static bool pd_active_wakeup(struct device *dev) -{ -	return true; -} +struct dev_power_governor sh7372_always_on_gov = { +	.power_down_ok = sh7372_power_down_forbidden, +};  void sh7372_init_pm_domain(struct sh7372_pm_domain *sh7372_pd)  {  	struct generic_pm_domain *genpd = &sh7372_pd->genpd; -	pm_genpd_init(genpd, NULL, false); +	pm_genpd_init(genpd, sh7372_pd->gov, false);  	genpd->stop_device = pm_clk_suspend;  	genpd->start_device = pm_clk_resume; +	genpd->dev_irq_safe = true;  	genpd->active_wakeup = pd_active_wakeup; - -	if (sh7372_pd == &sh7372_a4lc) { -		genpd->power_off = pd_power_down_a4lc; -		genpd->power_on = pd_power_up; -	} else if (sh7372_pd == &sh7372_a3rv) { -		genpd->power_off = pd_power_down_a3rv; -		genpd->power_on = pd_power_up_a3rv; -	} else { -		genpd->power_off = pd_power_down; -		genpd->power_on = pd_power_up; -	} +	genpd->power_off = pd_power_down; +	genpd->power_on = pd_power_up;  	genpd->power_on(&sh7372_pd->genpd);  } @@ -152,11 +182,15 @@ void sh7372_add_device_to_domain(struct sh7372_pm_domain *sh7372_pd,  {  	struct device *dev = &pdev->dev; -	if (!dev->power.subsys_data) { -		pm_clk_init(dev); -		pm_clk_add(dev, NULL); -	}  	pm_genpd_add_device(&sh7372_pd->genpd, dev); +	if (pm_clk_no_clocks(dev)) +		pm_clk_add(dev, NULL); +} + +void sh7372_pm_add_subdomain(struct sh7372_pm_domain *sh7372_pd, +			     struct sh7372_pm_domain *sh7372_sd) +{ +	pm_genpd_add_subdomain(&sh7372_pd->genpd, &sh7372_sd->genpd);  }  struct sh7372_pm_domain sh7372_a4lc = { @@ -171,6 +205,14 @@ struct sh7372_pm_domain sh7372_d4 = {  	.bit_shift = 3,  }; +struct sh7372_pm_domain sh7372_a4r = { +	.bit_shift = 5, +	.gov = &sh7372_always_on_gov, +	.suspend = sh7372_a4r_suspend, +	.resume = sh7372_intcs_resume, +	.stay_on = true, +}; +  struct sh7372_pm_domain sh7372_a3rv = {  	.bit_shift = 6,  }; @@ -179,39 +221,187 @@ struct sh7372_pm_domain sh7372_a3ri = {  	.bit_shift = 8,  }; +struct sh7372_pm_domain sh7372_a3sp = { +	.bit_shift = 11, +	.gov = &sh7372_always_on_gov, +	.no_debug = true, +}; +  struct sh7372_pm_domain sh7372_a3sg = {  	.bit_shift = 13,  };  #endif /* CONFIG_PM */ +#if defined(CONFIG_SUSPEND) || defined(CONFIG_CPU_IDLE) +static int sh7372_do_idle_core_standby(unsigned long unused) +{ +	cpu_do_idle(); /* WFI when SYSTBCR == 0x10 -> Core Standby */ +	return 0; +} +  static void sh7372_enter_core_standby(void)  { -	void __iomem *smfram = (void __iomem *)SMFRAM; +	/* set reset vector, translate 4k */ +	__raw_writel(__pa(sh7372_resume_core_standby_a3sm), SBAR); +	__raw_writel(0, APARMBAREA); -	__raw_writel(0, APARMBAREA); /* translate 4k */ -	__raw_writel(__pa(sh7372_cpu_resume), SBAR); /* set reset vector */ -	__raw_writel(0x10, SYSTBCR); /* enable core standby */ +	/* enter sleep mode with SYSTBCR to 0x10 */ +	__raw_writel(0x10, SYSTBCR); +	cpu_suspend(0, sh7372_do_idle_core_standby); +	__raw_writel(0, SYSTBCR); -	__raw_writel(0, smfram + 0x3c); /* clear page table address */ +	 /* disable reset vector translation */ +	__raw_writel(0, SBAR); +} +#endif + +#ifdef CONFIG_SUSPEND +static void sh7372_enter_a3sm_common(int pllc0_on) +{ +	/* set reset vector, translate 4k */ +	__raw_writel(__pa(sh7372_resume_core_standby_a3sm), SBAR); +	__raw_writel(0, APARMBAREA); + +	if (pllc0_on) +		__raw_writel(0, PLLC01STPCR); +	else +		__raw_writel(1 << 28, PLLC01STPCR); + +	__raw_writel(0, PDNSEL); /* power-down A3SM only, not A4S */ +	__raw_readl(WUPSFAC); /* read wakeup int. factor before sleep */ +	cpu_suspend(0, sh7372_do_idle_a3sm); +	__raw_readl(WUPSFAC); /* read wakeup int. factor after wakeup */ + +	 /* disable reset vector translation */ +	__raw_writel(0, SBAR); +} + +static int sh7372_a3sm_valid(unsigned long *mskp, unsigned long *msk2p) +{ +	unsigned long mstpsr0, mstpsr1, mstpsr2, mstpsr3, mstpsr4; +	unsigned long msk, msk2; + +	/* check active clocks to determine potential wakeup sources */ + +	mstpsr0 = __raw_readl(MSTPSR0); +	if ((mstpsr0 & 0x00000003) != 0x00000003) { +		pr_debug("sh7372 mstpsr0 0x%08lx\n", mstpsr0); +		return 0; +	} + +	mstpsr1 = __raw_readl(MSTPSR1); +	if ((mstpsr1 & 0xff079b7f) != 0xff079b7f) { +		pr_debug("sh7372 mstpsr1 0x%08lx\n", mstpsr1); +		return 0; +	} -	sh7372_cpu_suspend(); -	cpu_init(); +	mstpsr2 = __raw_readl(MSTPSR2); +	if ((mstpsr2 & 0x000741ff) != 0x000741ff) { +		pr_debug("sh7372 mstpsr2 0x%08lx\n", mstpsr2); +		return 0; +	} -	/* if page table address is non-NULL then we have been powered down */ -	if (__raw_readl(smfram + 0x3c)) { -		__raw_writel(__raw_readl(smfram + 0x40), -			     __va(__raw_readl(smfram + 0x3c))); +	mstpsr3 = __raw_readl(MSTPSR3); +	if ((mstpsr3 & 0x1a60f010) != 0x1a60f010) { +		pr_debug("sh7372 mstpsr3 0x%08lx\n", mstpsr3); +		return 0; +	} -		flush_tlb_all(); -		set_cr(__raw_readl(smfram + 0x38)); +	mstpsr4 = __raw_readl(MSTPSR4); +	if ((mstpsr4 & 0x00008cf0) != 0x00008cf0) { +		pr_debug("sh7372 mstpsr4 0x%08lx\n", mstpsr4); +		return 0;  	} -	__raw_writel(0, SYSTBCR); /* disable core standby */ -	__raw_writel(0, SBAR); /* disable reset vector translation */ +	msk = 0; +	msk2 = 0; + +	/* make bitmaps of limited number of wakeup sources */ + +	if ((mstpsr2 & (1 << 23)) == 0) /* SPU2 */ +		msk |= 1 << 31; + +	if ((mstpsr2 & (1 << 12)) == 0) /* MFI_MFIM */ +		msk |= 1 << 21; + +	if ((mstpsr4 & (1 << 3)) == 0) /* KEYSC */ +		msk |= 1 << 2; + +	if ((mstpsr1 & (1 << 24)) == 0) /* CMT0 */ +		msk |= 1 << 1; + +	if ((mstpsr3 & (1 << 29)) == 0) /* CMT1 */ +		msk |= 1 << 1; + +	if ((mstpsr4 & (1 << 0)) == 0) /* CMT2 */ +		msk |= 1 << 1; + +	if ((mstpsr2 & (1 << 13)) == 0) /* MFI_MFIS */ +		msk2 |= 1 << 17; + +	*mskp = msk; +	*msk2p = msk2; + +	return 1; +} + +static void sh7372_icr_to_irqcr(unsigned long icr, u16 *irqcr1p, u16 *irqcr2p) +{ +	u16 tmp, irqcr1, irqcr2; +	int k; + +	irqcr1 = 0; +	irqcr2 = 0; + +	/* convert INTCA ICR register layout to SYSC IRQCR+IRQCR2 */ +	for (k = 0; k <= 7; k++) { +		tmp = (icr >> ((7 - k) * 4)) & 0xf; +		irqcr1 |= (tmp & 0x03) << (k * 2); +		irqcr2 |= (tmp >> 2) << (k * 2); +	} + +	*irqcr1p = irqcr1; +	*irqcr2p = irqcr2; +} + +static void sh7372_setup_a3sm(unsigned long msk, unsigned long msk2) +{ +	u16 irqcrx_low, irqcrx_high, irqcry_low, irqcry_high; +	unsigned long tmp; + +	/* read IRQ0A -> IRQ15A mask */ +	tmp = bitrev8(__raw_readb(INTMSK00A)); +	tmp |= bitrev8(__raw_readb(INTMSK10A)) << 8; + +	/* setup WUPSMSK from clocks and external IRQ mask */ +	msk = (~msk & 0xc030000f) | (tmp << 4); +	__raw_writel(msk, WUPSMSK); + +	/* propage level/edge trigger for external IRQ 0->15 */ +	sh7372_icr_to_irqcr(__raw_readl(ICR1A), &irqcrx_low, &irqcry_low); +	sh7372_icr_to_irqcr(__raw_readl(ICR2A), &irqcrx_high, &irqcry_high); +	__raw_writel((irqcrx_high << 16) | irqcrx_low, IRQCR); +	__raw_writel((irqcry_high << 16) | irqcry_low, IRQCR2); + +	/* read IRQ16A -> IRQ31A mask */ +	tmp = bitrev8(__raw_readb(INTMSK20A)); +	tmp |= bitrev8(__raw_readb(INTMSK30A)) << 8; + +	/* setup WUPSMSK2 from clocks and external IRQ mask */ +	msk2 = (~msk2 & 0x00030000) | tmp; +	__raw_writel(msk2, WUPSMSK2); + +	/* propage level/edge trigger for external IRQ 16->31 */ +	sh7372_icr_to_irqcr(__raw_readl(ICR3A), &irqcrx_low, &irqcry_low); +	sh7372_icr_to_irqcr(__raw_readl(ICR4A), &irqcrx_high, &irqcry_high); +	__raw_writel((irqcrx_high << 16) | irqcrx_low, IRQCR3); +	__raw_writel((irqcry_high << 16) | irqcry_low, IRQCR4);  } +#endif  #ifdef CONFIG_CPU_IDLE +  static void sh7372_cpuidle_setup(struct cpuidle_device *dev)  {  	struct cpuidle_state *state; @@ -239,9 +429,25 @@ static void sh7372_cpuidle_init(void) {}  #endif  #ifdef CONFIG_SUSPEND +  static int sh7372_enter_suspend(suspend_state_t suspend_state)  { -	sh7372_enter_core_standby(); +	unsigned long msk, msk2; + +	/* check active clocks to determine potential wakeup sources */ +	if (sh7372_a3sm_valid(&msk, &msk2)) { + +		/* convert INTC mask and sense to SYSC mask and sense */ +		sh7372_setup_a3sm(msk, msk2); + +		/* enter A3SM sleep with PLLC0 off */ +		pr_debug("entering A3SM\n"); +		sh7372_enter_a3sm_common(0); +	} else { +		/* default to Core Standby that supports all wakeup sources */ +		pr_debug("entering Core Standby\n"); +		sh7372_enter_core_standby(); +	}  	return 0;  } @@ -253,9 +459,6 @@ static void sh7372_suspend_init(void)  static void sh7372_suspend_init(void) {}  #endif -#define DBGREG1 0xe6100020 -#define DBGREG9 0xe6100040 -  void __init sh7372_pm_init(void)  {  	/* enable DBG hardware block to kick SYSC */ @@ -263,6 +466,9 @@ void __init sh7372_pm_init(void)  	__raw_writel(0x0000a501, DBGREG9);  	__raw_writel(0x00000000, DBGREG1); +	/* do not convert A3SM, A3SP, A3SG, A4R power down into A4S */ +	__raw_writel(0, PDNSEL); +  	sh7372_suspend_init();  	sh7372_cpuidle_init();  } | 
