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



 

郑重声明:本站内容如果来自互联网及其他传播媒体,其版权均属原媒体及文章作者所有。转载目的在于传递更多信息及用于网络分享,并不代表本站赞同其观点和对其真实性负责,也不构成任何其他建议。