转自:http://wenku.baidu.com/view/61ee1da7284ac850ad02422a.html
Android下有三种省电方式:
Early_suspend::主要是屏幕省电需要。进入early_suspend后,注册了early_suspend的外设进入休眠,大部分进程都还在运行,CPU处于正常工作状态。外设可以通过register_early_suspend注册为early_suspend设备。
LPAUDION_MODE:系统进入early_suspend模式之后,通过has_audio_wake_lock判断是否进入LPAUDION_MODE模式,进入LPAUDION_MODE后,音乐正常播放,CPU进入省电模式。
Suspend:系统运行状态保存到内存,CPU进入省电模式,所有外设电源关闭,只有内存保持供电和工作。
跟休眠唤醒相关的文件:
/kernel/power/main.c
/kernel/power/earlysuspend.c
/kernel/power/wakelock.c
/kernel/power/suspend.c
/drivers/base/power/main.c
/arch/arm/plat-samsung/pm.c
/arch/arm/mach-s5pv210
/arch/arm/plat-s5p/sleep.s
/arch/arm/mach-s5pv210/cpuidle.c
Android linux休眠流程:
休眠的入口在/kernel/power/main.c的state_store函数。State_store函数通过 power_attr(state)宏,在sysfs中定义了一个属性。用户空间可以通过向state写入MEM,就可以使系统进入休眠状态。也可以通过cat state查看支持哪些休眠方式
在state_store函数中,调用request_suspend_state函数请求休眠。在request_suspend_state中,如果判断到系统不处于休眠状态,则调用early_suspend。这里是通过将early_suspend插入到休眠工作队列来实现调用的。
在early_suspend中,通过early_suspend设备链表依次调用设备各自的suspend函数(early_suspend设备通过register_early_suspend注册)。接下来判断是否申请了audio_wake_lock,如果是,进入LPAUDIO_MODE模式。最后,调用wake_unlock(&main_wake_lock)
在wake_unlock中,系统判断lock是否为0,若不为0,则系统处于early_suspend模式。因此,在kernel层,我们可以使用wake_lock去申请使系统不进入休眠;若为0则将suspend插入工作队列(wakelock.c)
Suspend函数中,调用pm_suspend,pm_suspend调用enter_state,准备进入休眠模式。到此为止,休眠进入标准的linux模式的休眠。在enter_state中,经过三步进入休眠
1. 调用suspend_prepare 冻结用户程序。
2. 调用dpm_suspend_start 分别调用设备的suspend函数
3. 调用suspend_enter 进入suspend模式
在suspend_enter中,是通过suspend_ops调用进入suspend的。Suspend_ops是一个数据结构,通过s3c_pm_init函数初始化。s3c_pm_ops数据结构的enter成员函数初始化为s3c_pm_enter。跟硬件相关的睡眠动作,通过pm_cpu_prep和pm_cpu_sleep来实现。这两个变量在arch/arm/mach-s5pv210中定义,被赋值为s5pv210_pm_prepare和s5pv210_cpu_suspend。s5pv210_cpu_suspend是suspend过程中最后运行的地方。