1. 程式人生 > >Android 電源管理 -- wakelock機制

Android 電源管理 -- wakelock機制

PowerManager.WakeLock 有加鎖和解鎖兩種狀態,加鎖的方式有兩種:

    第一種是永久的鎖住,這樣的鎖除非顯式的放開,否則是不會解鎖的,所以這種鎖用起來要非常的小心。

    第二種鎖是超時鎖,這種鎖會在鎖住後一段時間解鎖。

在建立了 PowerManager.WakeLock 後,有兩種機制,第一種是不計數鎖機制,另一種是計數鎖機制。可以通過 setReferenceCounted(boolean value) 來指定,一般預設為計數機制。這兩種機制的區別在於,前者無論 acquire() 了多少次,只要通過一次 release()即可解鎖。而後者正真解鎖是在( --count == 0 

)的時候,同樣當 (count == 0) 的時候才會去申請加鎖。所以 PowerManager.WakeLock 的計數機制並不是正真意義上的對每次請求進行申請/釋放每一把鎖,它只是對同一把鎖被申請/釋放的次數進行了統計,然後再去操作。

原始碼 位置:frameworks/base/core/java/android/os/PowerManager.java

++++++++++++++++++++++++
講述 應用層 申請的鎖 怎麼傳到kernel下面的,來理解  整個wakelock的框架。  比如android跑起來之後 在 /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, “MyTag”);
        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"
nativeAcquireWakeLock(PARTIAL_WAKE_LOCK_ID, PARTIAL_NAME);
上面 這個函式呼叫Power類 裡面的 acquireWakeLock(),此時的PARTIAL_NAME作為引數傳遞到底層去。


public static native void nativeAcquireWakeLock(int lock, String id);
注:在PowerManagerService 類中沒有實現nativeAcquireWakeLock,其實現體在 frameworks/base/core/jni/android_os_Power.cpp中,所nativeAcquireWakeLock()方法時會呼叫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層 互動的地方。