Wake Lock是一种锁的机制, 只要有人拿着这个锁,系统就无法进入休眠,

可以被用户态程序和内核获得. 这个锁可以是有超时的或者是没有超时的,

超时的锁会在时间过去以后自动解锁. 如果没有锁了或者超时了, 内核就

会启动休眠的那套机制来进入休眠.

(一).内核维护了:

1).两个链表,active_wake_locks[WAKE_LOCK_TYPE_COUNT]

   active_wake_locks[0]维护的是suspend lock.

   active_wake_locks[1]维护的是idle lock.

2).一个链表,inactive_locks来记录所有处于inactive状态的锁.

 

 (二). 下面讲述应用层申请的锁怎么传到kernel下面的,来理解

       整个wakelock的框架。

       比如/sys/power/wake_lock 下面的PowerManagerService

       的生成过程。

    

     1). Android 提供了现成 android.os.PowerManager 类 ,类中

      提供 newWakeLock(int flags, String tag)方法来取得相应

      层次的锁, 此函数的定义

       

frameworks/base/core/java/android/os/PowerManager.java

      下面,应用程序在申请wake_lock时都会有调用。

       实例:

        PowerManager pm = (PowerManager)getSystemService(Context.POWER_SERVICE);

        PowerManager.WakeLock wl = pm.newWakeLock

                                                   (PowerManager.SCREEN_DIM_WAKE_LOCK, “My Tag”);

        wl.acquire();//申请锁这个里面会调用PowerManagerService里面acquireWakeLock()

 

 ***********************

        wl.release(); //释放锁,显示的释放,如果申请的锁不在此释放系统就不会进入休眠。

 

 

2). frameworks层

/frameworks/base/services/java/com/android/server/

 PowerManagerService.java

这个类是来管理所有的应用程序申请的wakelock。比如音视

频播放器,camera等申请的wakelock都是通过这个类来管理的。

static final String PARTIAL_NAME = "PowerManagerService"
 
  
Power.acquireWakeLock(Power.PARTIAL_WAKE_LOCK,
 
  
                                                                   PARTIAL_NAME);

这个函数调用Power类里面的acquireWakeLock(),此时的

PARTIAL_NAME作为参数传递到底层去。

 

 

/frameworks/base/core/java/android/os/Power.java
 
  
public static native void acquireWakeLock(int lock, String id);

注:native申明的方法在Power类中没有实现,其实现体在

       frameworks/base/core/jni/android_os_Power.cpp中,所

       以调用Power类的acquireWakeLock()方法时会调用JNI

       下的实现方法。

 

 

 

   3).JNI层的实现

     路径:

frameworks/base/core/jni/android_os_Power.cpp
 
  
    static void acquireWakeLock(JNIEnv *env, jobject clazz,
 
  
                                                         jint lock, jstring idObj)
 
  
  {
 
  
    **************
 
  
      const char *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)函数如下:

int acquire_wake_lock(int lock, const char* id)
 
  
 {   
 
  
 **************
 
  
     return write(fd, id, strlen(id));
 
  
 }

注: fd就是文件描述符,在此表示”/sys/power/wake_lock”

         id就是从PowerManagerService类中传下来的参数即:

         PARTIAL_NAME = "PowerManagerService"

         到此就是通过文件系统来与kernel层交互的地方。

 

 

wake lock具有两种类型:

WAKE_LOCK_SUSPEND– 这种锁如果被某个task持有,那么系统将无法进入休眠。

WAKE_LOCK_IDLE

 

不过常用的所类型还是WAKE_LOCK_SUSPEND,包括userwakelock.c提供给用户空间的新建wakelock的接口,都是建立的第一种锁。另外系统为了分开管理这两种不同类型的锁,建立了两个链表来统一链接不同类型的锁:active_wake_locks[],这个是具有两个链表头的数组,元素0是挂接WAKE_LOCK_SUSPEND类型的锁,而元素1就是挂接WAKE_LOCK_IDLE类型的wakelock了。

 

       接着上面说,这个初始化函数新建这些锁之后,直接将主锁(main_wake_lock)给上锁了,其余都是非锁状态。新建wakelock使用函数wake_lock_init(),该函数设置锁的名字,类型,最后将新建的锁挂接到一个专门链接这些非锁状态的链表inactive_locks上(新建的wakelock初期都是出于非锁状态的,除非显示调用函数wake_lock来上锁)。接着如果使用函数wake_lock()来给特定的wakelock上锁的话,会将该锁从链表inactive_locks上移动到对应类型的专用链表上active_wake_locks[type]上。

      

       wakelock有两种形式的锁:超时锁和非超时锁,这两种形式的锁都是使用函数wake_lock_init()来初始化,只是在上锁的时候会有一点点差别,超时锁使用函数wake_lock_timeout(),而非超时锁使用函数wake_lock(), 这个两个函数会最终调用到同一个函数wake_lock_internal(),该函数依靠传入的不同参数来选择不同的路径来工作。值得注意的是,非超时锁必须手工解锁,否则系统永远不能进入睡眠。