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层交互的地方。