/* * linux/arch/arm/mach-omap2/pm34xx.c * * OMAP3 Power Management Routines * * Copyright (C) 2006-2008 Nokia Corporation * Tony Lindgren * Jouni Hogander * * Copyright (C) 2005 Texas Instruments, Inc. * Richard Woodruff * * Based on pm.c for omap1 * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License version 2 as * published by the Free Software Foundation. */ #include #include #include #include #include #include #include #include #include #include #include #include "cm.h" #include "cm-regbits-34xx.h" #include "prm-regbits-34xx.h" #include "prm.h" #include "pm.h" #include "smartreflex.h" struct power_state { struct powerdomain *pwrdm; u32 next_state; u32 saved_state; struct list_head node; }; static LIST_HEAD(pwrst_list); static void (*_omap_sram_idle)(u32 *addr, int save_state); static void (*saved_idle)(void); static struct powerdomain *mpu_pwrdm; /* PRCM Interrupt Handler for wakeups */ static irqreturn_t prcm_interrupt_handler (int irq, void *dev_id) { u32 wkst, irqstatus_mpu; u32 fclk, iclk; /* WKUP */ wkst = prm_read_mod_reg(WKUP_MOD, PM_WKST); if (wkst) { iclk = cm_read_mod_reg(WKUP_MOD, CM_ICLKEN); fclk = cm_read_mod_reg(WKUP_MOD, CM_FCLKEN); cm_set_mod_reg_bits(wkst, WKUP_MOD, CM_ICLKEN); cm_set_mod_reg_bits(wkst, WKUP_MOD, CM_FCLKEN); prm_write_mod_reg(wkst, WKUP_MOD, PM_WKST); while (prm_read_mod_reg(WKUP_MOD, PM_WKST)); cm_write_mod_reg(iclk, WKUP_MOD, CM_ICLKEN); cm_write_mod_reg(fclk, WKUP_MOD, CM_FCLKEN); } /* CORE */ wkst = prm_read_mod_reg(CORE_MOD, PM_WKST1); if (wkst) { iclk = cm_read_mod_reg(CORE_MOD, CM_ICLKEN1); fclk = cm_read_mod_reg(CORE_MOD, CM_FCLKEN1); cm_set_mod_reg_bits(wkst, CORE_MOD, CM_ICLKEN1); cm_set_mod_reg_bits(wkst, CORE_MOD, CM_FCLKEN1); prm_write_mod_reg(wkst, CORE_MOD, PM_WKST1); while (prm_read_mod_reg(CORE_MOD, PM_WKST1)); cm_write_mod_reg(iclk, CORE_MOD, CM_ICLKEN1); cm_write_mod_reg(fclk, CORE_MOD, CM_FCLKEN1); } wkst = prm_read_mod_reg(CORE_MOD, OMAP3430ES2_PM_WKST3); if (wkst) { iclk = cm_read_mod_reg(CORE_MOD, CM_ICLKEN3); fclk = cm_read_mod_reg(CORE_MOD, OMAP3430ES2_CM_FCLKEN3); cm_set_mod_reg_bits(wkst, CORE_MOD, CM_ICLKEN3); cm_set_mod_reg_bits(wkst, CORE_MOD, OMAP3430ES2_CM_FCLKEN3); prm_write_mod_reg(wkst, CORE_MOD, OMAP3430ES2_PM_WKST3); while (prm_read_mod_reg(CORE_MOD, OMAP3430ES2_PM_WKST3)); cm_write_mod_reg(iclk, CORE_MOD, CM_ICLKEN3); cm_write_mod_reg(fclk, CORE_MOD, OMAP3430ES2_CM_FCLKEN3); } /* PER */ wkst = prm_read_mod_reg(OMAP3430_PER_MOD, PM_WKST); if (wkst) { iclk = cm_read_mod_reg(OMAP3430_PER_MOD, CM_ICLKEN); fclk = cm_read_mod_reg(OMAP3430_PER_MOD, CM_FCLKEN); cm_set_mod_reg_bits(wkst, OMAP3430_PER_MOD, CM_ICLKEN); cm_set_mod_reg_bits(wkst, OMAP3430_PER_MOD, CM_FCLKEN); prm_write_mod_reg(wkst, OMAP3430_PER_MOD, PM_WKST); while (prm_read_mod_reg(OMAP3430_PER_MOD, PM_WKST)); cm_write_mod_reg(iclk, OMAP3430_PER_MOD, CM_ICLKEN); cm_write_mod_reg(fclk, OMAP3430_PER_MOD, CM_FCLKEN); } if (is_sil_rev_greater_than(OMAP3430_REV_ES1_0)) { /* USBHOST */ wkst = prm_read_mod_reg(OMAP3430ES2_USBHOST_MOD, PM_WKST); if (wkst) { iclk = cm_read_mod_reg(OMAP3430ES2_USBHOST_MOD, CM_ICLKEN); fclk = cm_read_mod_reg(OMAP3430ES2_USBHOST_MOD, CM_FCLKEN); cm_set_mod_reg_bits(wkst, OMAP3430ES2_USBHOST_MOD, CM_ICLKEN); cm_set_mod_reg_bits(wkst, OMAP3430ES2_USBHOST_MOD, CM_FCLKEN); prm_write_mod_reg(wkst, OMAP3430ES2_USBHOST_MOD, PM_WKST); while (prm_read_mod_reg(OMAP3430ES2_USBHOST_MOD, PM_WKST)); cm_write_mod_reg(iclk, OMAP3430ES2_USBHOST_MOD, CM_ICLKEN); cm_write_mod_reg(fclk, OMAP3430ES2_USBHOST_MOD, CM_FCLKEN); } } irqstatus_mpu = prm_read_mod_reg(OCP_MOD, OMAP2_PRM_IRQSTATUS_MPU_OFFSET); prm_write_mod_reg(irqstatus_mpu, OCP_MOD, OMAP2_PRM_IRQSTATUS_MPU_OFFSET); while (prm_read_mod_reg(OCP_MOD, OMAP2_PRM_IRQSTATUS_MPU_OFFSET)); return IRQ_HANDLED; } static void omap_sram_idle(void) { /* 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 */ int save_state = 0, mpu_next_state; if (!_omap_sram_idle) return; mpu_next_state = pwrdm_read_next_pwrst(mpu_pwrdm); switch (mpu_next_state) { case PWRDM_POWER_RET: /* No need to save context */ save_state = 0; break; default: /* Invalid state */ printk(KERN_ERR "Invalid mpu state in sram_idle\n"); return; } omap2_gpio_prepare_for_retention(); _omap_sram_idle(NULL, save_state); omap2_gpio_resume_after_retention(); } static int omap3_can_sleep(void) { if (!enable_dyn_sleep) return 0; if (atomic_read(&sleep_block) > 0) return 0; return 1; } /* _clkdm_deny_idle - private callback function used by set_pwrdm_state() */ static int _clkdm_deny_idle(struct powerdomain *pwrdm, struct clockdomain *clkdm) { omap2_clkdm_deny_idle(clkdm); return 0; } /* _clkdm_allow_idle - private callback function used by set_pwrdm_state() */ static int _clkdm_allow_idle(struct powerdomain *pwrdm, struct clockdomain *clkdm) { omap2_clkdm_allow_idle(clkdm); return 0; } /* This sets pwrdm state (other than mpu & core. Currently only ON & * RET are supported. Function is assuming that clkdm doesn't have * hw_sup mode enabled. */ static int set_pwrdm_state(struct powerdomain *pwrdm, u32 state) { u32 cur_state; int ret = 0; if (pwrdm == NULL || IS_ERR(pwrdm)) return -EINVAL; cur_state = pwrdm_read_next_pwrst(pwrdm); if (cur_state == state) return ret; pwrdm_for_each_clkdm(pwrdm, _clkdm_deny_idle); ret = pwrdm_set_next_pwrst(pwrdm, state); if (ret) { printk(KERN_ERR "Unable to set state of powerdomain: %s\n", pwrdm->name); goto err; } pwrdm_for_each_clkdm(pwrdm, _clkdm_allow_idle); err: return ret; } static void omap3_pm_idle(void) { local_irq_disable(); local_fiq_disable(); if (!omap3_can_sleep()) goto out; if (omap_irq_pending()) goto out; omap_sram_idle(); out: local_fiq_enable(); local_irq_enable(); } static int omap3_pm_prepare(void) { saved_idle = pm_idle; pm_idle = NULL; return 0; } static int omap3_pm_suspend(void) { struct power_state *pwrst; int state, ret = 0; /* XXX Disable smartreflex before entering suspend */ disable_smartreflex(SR1); disable_smartreflex(SR2); /* 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) { if (set_pwrdm_state(pwrst->pwrdm, pwrst->next_state)) goto restore; if (pwrdm_clear_all_prev_pwrst(pwrst->pwrdm)) goto restore; } omap_sram_idle(); restore: /* Restore next_pwrsts */ list_for_each_entry(pwrst, &pwrst_list, node) { set_pwrdm_state(pwrst->pwrdm, pwrst->saved_state); state = pwrdm_read_prev_pwrst(pwrst->pwrdm); if (state != pwrst->next_state) { printk(KERN_INFO "Powerdomain (%s) didn't enter " "target state %d\n", pwrst->pwrdm->name, pwrst->next_state); ret = -1; } } if (ret) printk(KERN_ERR "Could not enter target state in pm_suspend\n"); else printk(KERN_INFO "Successfully put all powerdomains " "to target state\n"); /* XXX Enable smartreflex after suspend */ enable_smartreflex(SR1); enable_smartreflex(SR2); return ret; } static int omap3_pm_enter(suspend_state_t state) { int ret = 0; switch (state) { case PM_SUSPEND_STANDBY: case PM_SUSPEND_MEM: ret = omap3_pm_suspend(); break; default: ret = -EINVAL; } return ret; } static void omap3_pm_finish(void) { pm_idle = saved_idle; } static struct platform_suspend_ops omap_pm_ops = { .prepare = omap3_pm_prepare, .enter = omap3_pm_enter, .finish = omap3_pm_finish, .valid = suspend_valid_only_mem, }; static void __init prcm_setup_regs(void) { /* XXX Reset all wkdeps. This should be done when initializing * powerdomains */ prm_write_mod_reg(0, OMAP3430_IVA2_MOD, PM_WKDEP); prm_write_mod_reg(0, MPU_MOD, PM_WKDEP); prm_write_mod_reg(0, OMAP3430_DSS_MOD, PM_WKDEP); prm_write_mod_reg(0, OMAP3430_NEON_MOD, PM_WKDEP); prm_write_mod_reg(0, OMAP3430_CAM_MOD, PM_WKDEP); prm_write_mod_reg(0, OMAP3430_PER_MOD, PM_WKDEP); if (is_sil_rev_greater_than(OMAP3430_REV_ES1_0)) { prm_write_mod_reg(0, OMAP3430ES2_SGX_MOD, PM_WKDEP); prm_write_mod_reg(0, OMAP3430ES2_USBHOST_MOD, PM_WKDEP); } else prm_write_mod_reg(0, GFX_MOD, PM_WKDEP); /* setup wakup source */ prm_write_mod_reg(OMAP3430_EN_IO | OMAP3430_EN_GPIO1 | OMAP3430_EN_GPT1, WKUP_MOD, PM_WKEN); /* No need to write EN_IO, that is always enabled */ prm_write_mod_reg(OMAP3430_EN_GPIO1 | OMAP3430_EN_GPT1, WKUP_MOD, OMAP3430_PM_MPUGRPSEL); /* For some reason IO doesn't generate wakeup event even if * it is selected to mpu wakeup goup */ prm_write_mod_reg(OMAP3430_IO_EN | OMAP3430_WKUP_EN, OCP_MOD, OMAP2_PRM_IRQENABLE_MPU_OFFSET); } static int __init pwrdms_setup(struct powerdomain *pwrdm) { struct power_state *pwrst; if (!pwrdm->pwrsts) return 0; pwrst = kmalloc(sizeof(struct power_state), GFP_KERNEL); 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); return set_pwrdm_state(pwrst->pwrdm, pwrst->next_state); } int __init omap3_pm_init(void) { struct power_state *pwrst; int ret; printk(KERN_ERR "Power Management for TI OMAP3.\n"); /* XXX prcm_setup_regs needs to be before enabling hw * supervised mode for powerdomains */ prcm_setup_regs(); ret = request_irq(INT_34XX_PRCM_MPU_IRQ, (irq_handler_t)prcm_interrupt_handler, IRQF_DISABLED, "prcm", NULL); if (ret) { printk(KERN_ERR "request_irq failed to register for 0x%x\n", INT_34XX_PRCM_MPU_IRQ); goto err1; } ret = pwrdm_for_each(pwrdms_setup); if (ret) { printk(KERN_ERR "Failed to setup powerdomains\n"); goto err2; } mpu_pwrdm = pwrdm_lookup("mpu_pwrdm"); if (mpu_pwrdm == NULL) { printk(KERN_ERR "Failed to get mpu_pwrdm\n"); goto err2; } _omap_sram_idle = omap_sram_push(omap34xx_cpu_suspend, omap34xx_cpu_suspend_sz); suspend_set_ops(&omap_pm_ops); pm_idle = omap3_pm_idle; err1: return ret; err2: free_irq(INT_34XX_PRCM_MPU_IRQ, NULL); list_for_each_entry(pwrst, &pwrst_list, node) { list_del(&pwrst->node); kfree(pwrst); } return ret; }