首页 > 代码库 > linux autosleep

linux autosleep

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();}



 

linux autosleep