Android WakeLock
【转:】http://blog.csdn.net/to_cm/archive/2010/08/01/5780773.aspx
在Android中,申请WakeLock可以让你的进程持续执行即使手机进入睡眠模式,
比较实用的是比如:
后台有网络功能,可以保证操作持续进行.
方法:在操作之前加入
PowerManagerpm=(PowerManager)mContext.getSystemService(Context.POWER_SERVICE);
wakeLock=pm.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK,Constants.TAG);
wakeLock.acquire();
别忘了在操作完毕之后释放掉
if(wakeLock!=null){
wakeLock.release();
wakeLock=null;
}
例子:
publicvoidsetWakeMode(Contextcontext,intmode){
booleanwasheld=false;
if(mWakeLock!=null){
if(mWakeLock.isHeld()){
washeld=true;
mWakeLock.release();
}
mWakeLock=null;
}
PowerManagerpm=(PowerManager)context
.getSystemService(Context.POWER_SERVICE);
mWakeLock=pm.newWakeLock(mode|PowerManager.ON_AFTER_RELEASE,
CloudGamePlayer.class.getName());
mWakeLock.setReferenceCounted(false);
if(washeld){
mWakeLock.acquire();
}
}
WakeLock是一种锁的机制,只要有人拿着这个锁,系统就无法进入休眠,
可以被用户态程序和内核获得.这个锁可以是有超时的或者是没有超时的,
超时的锁会在时间过去以后自动解锁.如果没有锁了或者超时了,内核就
会启动休眠的那套机制来进入休眠.
(一).内核维护了:
1).两个链表,active_wake_locks[WAKE_LOCK_TYPE_COUNT]
active_wake_locks[0]维护的是suspendlock.
active_wake_locks[1]维护的是idlelock.
2).一个链表,inactive_locks来记录所有处于inactive状态的锁.
(二).下面讲述应用层申请的锁怎么传到kernel下面的,来理解
整个wakelock的框架。
比如/sys/power/wake_lock下面的PowerManagerService
的生成过程。
1).Android提供了现成android.os.PowerManager类,类中
提供newWakeLock(intflags,Stringtag)方法来取得相应
层次的锁,此函数的定义
frameworks/base/core/java/android/os/PowerManager.java
下面,应用程序在申请wake_lock时都会有调用。
实例:
PowerManagerpm=(PowerManager)getSystemService(Context.POWER_SERVICE);
PowerManager.WakeLockwl=pm.newWakeLock
(PowerManager.SCREEN_DIM_WAKE_LOCK,“MyTag”);
wl.acquire();//申请锁这个里面会调用PowerManagerService里面acquireWakeLock()
***********************
wl.release();//释放锁,显示的释放,如果申请的锁不在此释放系统就不会进入休眠。
2).frameworks层
/frameworks/base/services/java/com/android/server/
PowerManagerService.java
这个类是来管理所有的应用程序申请的wakelock。比如音视
频播放器,camera等申请的wakelock都是通过这个类来管理的。
staticfinalStringPARTIAL_name="PowerManagerService"
Power.acquireWakeLock(Power.PARTIAL_WAKE_LOCK,
PARTIAL_NAME);
这个函数调用Power类里面的acquireWakeLock(),此时的
PARTIAL_NAME作为参数传递到底层去。
/frameworks/base/core/java/android/os/Power.java
publicstaticnativevoidacquireWakeLock(intlock,Stringid);
注:native申明的方法在Power类中没有实现,其实现体在
frameworks/base/core/jni/android_os_Power.cpp中,所
以调用Power类的acquireWakeLock()方法时会调用JNI
下的实现方法。
3).JNI层的实现
路径:frameworks/base/core/jni/android_os_Power.cpp
staticvoidacquireWakeLock(JNIEnv*env,jobjectclazz,
jintlock,jstringidObj)
{
**************
constchar*id=env->GetStringUTFChars(idObj,NULL);
acquire_wake_lock(lock,id);
env->ReleaseStringUTFChars(idObj,id);
}
注:在acquireWakeLock()中调用了
路径下hardware/libhardware_legacy/power/power.c下面
的acquire_wake_lock(lock,id)
4).与kernel层的交互
在power.c下的acquire_wake_lock(lock,id)函数如下:
intacquire_wake_lock(intlock,constchar*id)
{
**************
returnwrite(fd,id,strlen(id));
}
注:fd就是文件描述符,在此表示”/sys/power/wake_lock”
id就是从PowerManagerService类中传下来的参数即:
PARTIAL_name="PowerManagerService"
到此就是通过文件系统来与kernel层交互的地方。