电源管理之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


  • 3
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值