autosleep.c这个文件代码量总共100行出头,但是其作为Linux低功耗的入口,在linux低功耗流程中却占据着举足轻重的作用。
首先,代码一,在初始化函数里,会做两件事情,一件是创建一个wake source ,供本文件其他函数为防止进入低功耗时使用;另一件就是创建一个名字为autosleep的工作队列,用于低功耗work对象附着、调度进入低功耗。
代码一:
int __init pm_autosleep_init(void) { autosleep_ws = wakeup_source_register("autosleep"); if (!autosleep_ws) return -ENOMEM; autosleep_wq = alloc_ordered_workqueue("autosleep", 0); if (autosleep_wq) return 0; wakeup_source_unregister(autosleep_ws); return -ENOMEM; }
代码二,而对于函数pm_autosleep_set_state来讲,则是触发低功耗进入条件的第一个接口,我们可以通过往init.rc里添加write /sys/power/autosleep mem来触发(或者控制台输入echo mem > /sys/power/autosleep),曾经在调试linux低功耗时,系统一直不进入睡眠,后来才打听到原来是没有配置此命令。在此函数中,我们可以看到这里使用了初始化中注册的wake source来防止进入睡眠(__pm_stay_awake(autosleep_ws));当我们在init.rc里加入write
/sys/power/autosleep mem这条语句后,系统启动到一定阶段,就会调用到该接口,该函数会在if (state > PM_SUSPEND_ON)满足条件进入到if分支,调用queue_up_suspend_work();来把创建的work对象加入到初始化时创建的工作队列中
代码二:
int pm_autosleep_set_state(suspend_state_t state) { #ifndef CONFIG_HIBERNATION if (state >= PM_SUSPEND_MAX) return -EINVAL; #endif __pm_stay_awake(autosleep_ws); mutex_lock(&autosleep_lock); autosleep_state = state; __pm_relax(autosleep_ws); if (state > PM_SUSPEND_ON) { pm_wakep_autosleep_enabled(true); queue_up_suspend_work(); } else { pm_wakep_autosleep_enabled(false); } mutex_unlock(&autosleep_lock); return 0; }
代码三:
static DECLARE_WORK(suspend_work, try_to_suspend); void queue_up_suspend_work(void) { if (!work_pending(&suspend_work) && autosleep_state > PM_SUSPEND_ON) queue_work(autosleep_wq, &suspend_work); }
代码四,而创建的work对象的回调函数try_to_suspend则是本文件的另一个重要接口,该函数会检查当前wake lock锁的持有状态,如果还有组件持有反对锁,则停止睡眠,重新触发工作队列;该文件又引出了其他两个相关组件,即pm_suspend()接口所在的suspend.c和pm_get_wakeup_count接口所在的文件Wakeup.c。suspend.c主要就是进入真正的睡眠流程的内容,而wakeup.c则是wake lock锁的相关内容,这两部分稍后我们再做介绍。
代码四:
static void try_to_suspend(struct work_struct *work) { unsigned int initial_count, final_count; if (!pm_get_wakeup_count(&initial_count, true)) goto out; mutex_lock(&autosleep_lock); if (!pm_save_wakeup_count(initial_count)) { mutex_unlock(&autosleep_lock); goto out; } if (autosleep_state == PM_SUSPEND_ON) { mutex_unlock(&autosleep_lock); return; } if (autosleep_state >= PM_SUSPEND_MAX) hibernate(); else pm_suspend(autosleep_state); mutex_unlock(&autosleep_lock); if (!pm_get_wakeup_count(&final_count, false)) goto out; /* * If the wakeup occured for an unknown reason, wait to prevent the * system from trying to suspend and waking up in a tight loop. */ if (final_count == initial_count) schedule_timeout_uninterruptible(HZ / 2); out: queue_up_suspend_work(); }