Wake lock - wakelock在android的电源管理系统中扮演一个核心的角色,wakelock是一种锁的机制, 只要有task拿着这个锁, 系统就无法进入休眠, 可以被用户态进程和内核线程获得。这个锁可以是有超时的或者是没有超时的, 超时的锁会在时间过去以后自动解锁。如果没有锁了或者超时了, 内核就会启动标准linux的那套休眠机制机制来进入休眠。
wakelock在android的休眠唤醒机制中扮演着及其重要的角色,主要源码位于文件:
kernel/kernel/power/wakelock.c
kernel/include/linux/wakelock.h
wakelock有两种形式的锁:超时锁和非超时锁,这两种形式的锁都是使用函数wake_lock_init()来初始化,只是在上锁的时候会有一点点差别,超时锁使用函数wake_lock_timeout(),而非超时锁使用函数wake_lock(), 这个两个函数会最终调用到同一个函数wake_lock_internal(),该函数依靠传入的不同参数来选择不同的路径来工作。值得注意的是,非超时锁必须手工解锁,否则系统永远不能进入睡眠
Wakelock的机制被文件userwakelock.c中的code封装成了sys的接口sys/power/wake_lock和sys/power/wake_unlock文件,那么上层如果需要新建wakelock或者注销wakelock,或者是解锁wakelock,都是操作这两个sys接口文件。
申请与释放(如何使用)
wakelock可以被内核空间 申请和释放。如果申请的是非超时锁wake_lock,需要相应的调用wake_unlock来释放,而超时锁则不需要手工释放(当然你也可以手工释放),超时后kernel系统会自动释放锁,在内核空间可以直接调用wake_lock, wake_lock_timeout 申请锁 Android kernel为用户空间提供了申请和释放wakelock的接口,实现在kernel/power/userwakelock.c中。
实例:
1:首先申明一个wake_lock
static struct wake_lock alarm_rtc_wake_lock;
2:在init函数中对wake_lock初始化
wake_lock_init(&alarm_rtc_wake_lock, WAKE_LOCK_SUSPEND, “alarm_rtc”);//WAKE_LOCK_SUSPEND 为suspend lock还有一种idle lock
WAKE_LOCK_SUSPEND, /* Prevent suspend /
WAKE_LOCK_IDLE, / Prevent low power idle*/
3:使用wake_lock()或者wake_lock_timeout()
wake_lock_timeout(&alarm_rtc_wake_lock, 1 * HZ);
wake_lock(&alarm_rtc_wake_lock);
4:使用wake_unlock()
wake_unlock(&alarm_rtc_wake_lock);
4:在exit中调用wake_lock_destroy()
wake_lock_destroy(&alarm_rtc_wake_lock);
wake_lock_init(&alarm_rtc_wake_lock, WAKE_LOCK_SUSPEND, “alarm_rtc”);
if (is_wakeup && !suspended && head_removed)
wake_unlock(&alarm_rtc_wake_lock);
if (is_wakeup && suspended) {
pr_alarm(FLOW, “changed alarm while suspened\n”);
wake_lock_timeout(&alarm_rtc_wake_lock, 1 * HZ);
return;
}
wake_lock(&alarm_rtc_wake_lock);
wake_unlock(&alarm_rtc_wake_lock);
wake_lock_timeout(&alarm_rtc_wake_lock, 1 * HZ);
suspended = false;
wake_lock_timeout(&alarm_rtc_wake_lock, 2 * HZ);
wake_lock_destroy(&alarm_rtc_wake_lock)