电源管理之android系统suspend to disk的实现(四)
3、Android的suspend执行流程
函数的流程如下所示:
应用程序通过对/sys/power/state的写入操作可以使系统进行休眠的状态,会调用/kernel/power/main.c中的state_store函数。pm_states包括:PM_SUSPEND_ON,PM_SUSPEND_STANDBY,PM_SUSPEND_MEM满足的状态。
1)当状态位PM_SUSPEND_ON的状态的时候,request_suspend_state();当满足休眠的状态的时候,调用request_suspend_state在suspend_work_queue工作线程上创建early_suspend_work队列,queue_work(suspend_work_queue,&early_suspend_work)。
2)然后通过DECLARE_WORK(early_suspend_work,early_suspend);在early_suspend_work工作队列中添加工作任务调用early_suspend,所以early_suspend函数会被调用。
3)early_suspend函数中通过list_for_each_entry(pos,&early_suspend_handlers,link) {
if(pos->suspend != NULL)pos->suspend(pos);在链表中找注册的suspend函数,这个suspend是early的。early_suspend后面调用wake_unlock函数。语句:wake_unlock(&main_wake_lock);
4)wake_unlock()中调用mod_timer启动expire_timer定时器,当定时时间到了,则执行expire_wake_locks函数,将suspend_work加入到suspend_work_queue队列中,分析到这里就可以知道了early_suspend_work和suspend_work这两个队列的先后顺序了(先执行early,定义一段时间后才执行suspend_work),然后会在suspend_work队列中加入suspend的工作任务,所以wakelock.c中的suspend函数会被调用。
5)suspend调用了pm_suspend,通过判断当前的状态,选择enter_state(),在enter_state中,经过了suspend_prepare,suspend_test和suspend_device_and_enter(),在suspend_device_and_enter中调用dpm_suspend_start(),然后调用dpm_suspend()。
6)dpm_suspend中利用while循环在dpm_list链表查找所有devic,然后调用device_suspend来保存状态和结束系统的设备。到了这里,我们就又可以看见在初始化的时候所看到的队列dpm_list。
dpm_list链表的添加是在device_pm_add中完成。
4、Android的wake Lock执行流程
我们接下来看一看wakelock的机制是怎么运行和起作用的,主要关注wakelock.c文件就可以了。
wake lock有加锁和解锁两种状态,加锁的方式有两种,一种是永久的锁住,这样的锁除非显示的放开,是不会解锁的,所以这种锁的使用是非常小心的.第二种是超时锁,这种锁会锁定系统唤醒一段时间,如果这个时间过去了,这个锁会自动解除.
锁有两种类型:
WAKE_LOCK_SUSPEND这种锁会防止系统进入睡眠
WAKE_LOCK_IDLE这种锁不会影响系统的休眠,作用我不是很清楚.
在wakelock中,会有3个地方让系统直接开始suspend(),分别是:
1)在wake_unlock()中,如果发现解锁以后没有任何其他的wakelock了,就开始休眠
2)在定时器都到时间以后,定时器的回调函数会查看是否有其他的wakelock,如果没有,就在这里让系统进入睡眠.
3)在wake_lock()中,对一个wakelock加锁以后,会再次检查一下有没有锁,我想这里的检查是没有必要的,更好的方法是使加锁的这个操作原子化,而 不是繁冗的检查.而且这样的检查也有可能漏掉.
5、Android于标准Linux休眠的区别
pm_suspend()虽然会调用enter_state()来进入标准的Linux休眠流程,但是还是有一些区别:
当进入冻结进程的时候,android首先会检查有没有wakelock,如果没有,才会停止这些进程,因为在开始suspend和冻结进程期间有可能有人申请了wake lock,如果是这样,冻结进程会被中断.
在suspend_late()中,会最后检查一次有没有wakelock,这有可能是某种快速申请wakelock,并且快速释放这个锁的进程导致的,如果有这种情况,这里会返回错误,整个suspend就会全部放弃.如果pm_suspend()成功了,LOG的输出可以通过在kernelcmd里面增加"no_console_suspend"来看到suspend和resume过程中的log输出。
个人观点,有问题请斧正!!
转载请注明出处:http://blog.csdn.net/wang_zheng_kai
郑重声明:本站内容如果来自互联网及其他传播媒体,其版权均属原媒体及文章作者所有。转载目的在于传递更多信息及用于网络分享,并不代表本站赞同其观点和对其真实性负责,也不构成任何其他建议。