mirror of
git://projects.qi-hardware.com/openwrt-xburst.git
synced 2025-01-19 15:21:07 +02:00
202 lines
4.5 KiB
C
202 lines
4.5 KiB
C
|
/*
|
||
|
* linux/arch/mips/jz4750d/pm.c
|
||
|
*
|
||
|
* JZ4750D Power Management Routines
|
||
|
*
|
||
|
* Copyright (C) 2006 Ingenic Semiconductor Inc.
|
||
|
* Author: <jlwei@ingenic.cn>
|
||
|
*
|
||
|
* This program is free software; you can distribute it and/or modify it
|
||
|
* under the terms of the GNU General Public License (Version 2) as
|
||
|
* published by the Free Software Foundation.
|
||
|
*
|
||
|
* This program is distributed in the hope it will be useful, but WITHOUT
|
||
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
||
|
* for more details.
|
||
|
*
|
||
|
* You should have received a copy of the GNU General Public License along
|
||
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
||
|
* 59 Temple Place - Suite 330, Boston MA 02111-1307, USA.
|
||
|
*
|
||
|
*/
|
||
|
|
||
|
#include <linux/init.h>
|
||
|
#include <linux/pm.h>
|
||
|
#include <linux/sched.h>
|
||
|
#include <linux/interrupt.h>
|
||
|
#include <linux/proc_fs.h>
|
||
|
#include <linux/sysctl.h>
|
||
|
#include <linux/module.h>
|
||
|
#include <linux/suspend.h>
|
||
|
#include <asm/cacheops.h>
|
||
|
#include <asm/jzsoc.h>
|
||
|
|
||
|
#undef DEBUG
|
||
|
//#define DEBUG
|
||
|
#ifdef DEBUG
|
||
|
#define dprintk(x...) printk(x)
|
||
|
#else
|
||
|
#define dprintk(x...)
|
||
|
#endif
|
||
|
|
||
|
extern void jz_board_do_sleep(unsigned long *ptr);
|
||
|
extern void jz_board_do_resume(unsigned long *ptr);
|
||
|
|
||
|
|
||
|
static void jz_pm_do_hibernate(void)
|
||
|
{
|
||
|
printk("Put CPU into hibernate mode.\n");
|
||
|
|
||
|
/* Mask all interrupts */
|
||
|
REG_INTC_IMSR = 0xffffffff;
|
||
|
|
||
|
/*
|
||
|
* RTC Wakeup or 1Hz interrupt can be enabled or disabled
|
||
|
* through RTC driver's ioctl (linux/driver/char/rtc_jz.c).
|
||
|
*/
|
||
|
|
||
|
/* Set minimum wakeup_n pin low-level assertion time for wakeup: 100ms */
|
||
|
while (!(REG_RTC_RCR & RTC_RCR_WRDY));
|
||
|
REG_RTC_HWFCR = (100 << RTC_HWFCR_BIT);
|
||
|
|
||
|
/* Set reset pin low-level assertion time after wakeup: must > 60ms */
|
||
|
while (!(REG_RTC_RCR & RTC_RCR_WRDY));
|
||
|
REG_RTC_HRCR = (60 << RTC_HRCR_BIT); /* 60 ms */
|
||
|
|
||
|
/* Scratch pad register to be reserved */
|
||
|
while (!(REG_RTC_RCR & RTC_RCR_WRDY));
|
||
|
REG_RTC_HSPR = 0x12345678;
|
||
|
|
||
|
/* clear wakeup status register */
|
||
|
while (!(REG_RTC_RCR & RTC_RCR_WRDY));
|
||
|
REG_RTC_HWRSR = 0x0;
|
||
|
|
||
|
/* Put CPU to power down mode */
|
||
|
while (!(REG_RTC_RCR & RTC_RCR_WRDY));
|
||
|
REG_RTC_HCR = RTC_HCR_PD;
|
||
|
|
||
|
while (!(REG_RTC_RCR & RTC_RCR_WRDY));
|
||
|
while(1);
|
||
|
|
||
|
}
|
||
|
|
||
|
static int jz_pm_do_sleep(void)
|
||
|
{
|
||
|
unsigned long delta;
|
||
|
unsigned long nfcsr = REG_EMC_NFCSR;
|
||
|
unsigned long opcr = REG_CPM_OPCR;
|
||
|
unsigned long imr = REG_INTC_IMR;
|
||
|
unsigned long sadc = REG_SADC_ENA;
|
||
|
unsigned long sleep_gpio_save[4*(GPIO_PORT_NUM-1)];
|
||
|
|
||
|
printk("Put CPU into sleep mode.\n");
|
||
|
|
||
|
/* Preserve current time */
|
||
|
delta = xtime.tv_sec - REG_RTC_RSR;
|
||
|
|
||
|
/* Disable nand flash */
|
||
|
REG_EMC_NFCSR = ~0xff;
|
||
|
|
||
|
/* stop sadc */
|
||
|
REG_SADC_ENA &= ~0x7;
|
||
|
while((REG_SADC_ENA & 0x7) != 0);
|
||
|
udelay(100);
|
||
|
|
||
|
/*stop udc and usb*/
|
||
|
__cpm_suspend_uhcphy();
|
||
|
__cpm_suspend_udcphy();
|
||
|
|
||
|
/* Mask all interrupts */
|
||
|
REG_INTC_IMSR = 0xffffffff;
|
||
|
|
||
|
/* Sleep on-board modules and setup wake event */
|
||
|
jz_board_do_sleep(sleep_gpio_save);
|
||
|
|
||
|
/* disable externel clock Oscillator in sleep mode */
|
||
|
__cpm_disable_osc_in_sleep();
|
||
|
/* select 32K crystal as RTC clock in sleep mode */
|
||
|
__cpm_select_rtcclk_rtc();
|
||
|
|
||
|
/* Enter SLEEP mode */
|
||
|
REG_CPM_LCR &= ~CPM_LCR_LPM_MASK;
|
||
|
REG_CPM_LCR |= CPM_LCR_LPM_SLEEP;
|
||
|
__asm__(".set\tmips3\n\t"
|
||
|
"wait\n\t"
|
||
|
".set\tmips0");
|
||
|
|
||
|
/* Restore to IDLE mode */
|
||
|
REG_CPM_LCR &= ~CPM_LCR_LPM_MASK;
|
||
|
REG_CPM_LCR |= CPM_LCR_LPM_IDLE;
|
||
|
|
||
|
/* Restore nand flash control register */
|
||
|
REG_EMC_NFCSR = nfcsr;
|
||
|
|
||
|
/* Restore interrupts */
|
||
|
REG_INTC_IMSR = imr;
|
||
|
REG_INTC_IMCR = ~imr;
|
||
|
|
||
|
/* Restore sadc */
|
||
|
REG_SADC_ENA = sadc;
|
||
|
|
||
|
/* Resume on-board modules */
|
||
|
jz_board_do_resume(sleep_gpio_save);
|
||
|
|
||
|
/* Restore Oscillator and Power Control Register */
|
||
|
REG_CPM_OPCR = opcr;
|
||
|
|
||
|
/* Restore current time */
|
||
|
xtime.tv_sec = REG_RTC_RSR + delta;
|
||
|
|
||
|
printk("Resume CPU from sleep mode.\n");
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
/* Put CPU to HIBERNATE mode */
|
||
|
void jz_pm_hibernate(void)
|
||
|
{
|
||
|
jz_pm_do_hibernate();
|
||
|
}
|
||
|
|
||
|
|
||
|
/* Put CPU to SLEEP mode */
|
||
|
int jz_pm_sleep(void)
|
||
|
{
|
||
|
return jz_pm_do_sleep();
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* valid states, only support standby(sleep) and mem(hibernate)
|
||
|
*/
|
||
|
static int jz4750_pm_valid(suspend_state_t state)
|
||
|
{
|
||
|
return state == PM_SUSPEND_MEM;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* Jz CPU enter save power mode
|
||
|
*/
|
||
|
static int jz4750_pm_enter(suspend_state_t state)
|
||
|
{
|
||
|
return jz_pm_do_sleep();
|
||
|
}
|
||
|
|
||
|
static struct platform_suspend_ops jz4750_pm_ops = {
|
||
|
.valid = jz4750_pm_valid,
|
||
|
.enter = jz4750_pm_enter,
|
||
|
};
|
||
|
|
||
|
/*
|
||
|
* Initialize power interface
|
||
|
*/
|
||
|
int __init jz_pm_init(void)
|
||
|
{
|
||
|
printk("Power Management for JZ\n");
|
||
|
|
||
|
suspend_set_ops(&jz4750_pm_ops);
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|