Android wakelock 自上而下浅析
生活随笔
收集整理的這篇文章主要介紹了
Android wakelock 自上而下浅析
小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.
Wake Lock是一種鎖的機(jī)制, 只要有人拿著這個(gè)鎖,系統(tǒng)就無法進(jìn)入休眠,可以被用戶態(tài)程序和內(nèi)核獲得. 這個(gè)鎖可以是有超時(shí)的或者是沒有超時(shí)的,超時(shí)的鎖會(huì)在時(shí)間過去以后自動(dòng)解鎖.?如果沒有鎖了或者超時(shí)了, 內(nèi)核就會(huì)啟動(dòng)休眠的那套機(jī)制來進(jìn)入休眠.
PowerManager.WakeLock 有加鎖和解鎖兩種狀態(tài),加鎖的方式有兩種,一種是永久的鎖住,這樣的鎖除非顯式的放開,是不會(huì)解鎖的,所以這種鎖用起來要非常的小心。第二種鎖是超時(shí)鎖,這種鎖會(huì)在鎖住后一段時(shí)間解鎖。在創(chuàng)建了 PowerManager.WakeLock 后,有兩種機(jī)制,第一種是不計(jì)數(shù)鎖機(jī)制,另一種是計(jì)數(shù)鎖機(jī)制。可以通過 setReferenceCounted(boolean value) 來指定,一般默認(rèn)為計(jì)數(shù)機(jī)制。這兩種機(jī)制的區(qū)別在于,前者無論 acquire() 了多少次,只要通過一次 release()即可解鎖。而后者正真解鎖是在( --count == 0 )的時(shí)候,同樣當(dāng) (count == 0) 的時(shí)候才會(huì)去申請(qǐng)加鎖,其他情況 isHeld 狀態(tài)是不會(huì)改變的。所以 PowerManager.WakeLock 的計(jì)數(shù)機(jī)制并不是正真意義上的對(duì)每次請(qǐng)求進(jìn)行申請(qǐng)/釋放每一把鎖,它只是對(duì)同一把鎖被申請(qǐng)/釋放的次數(shù)進(jìn)行了統(tǒng)計(jì)再正真意義上的去操作。一下進(jìn)行了永久鎖的測(cè)試: 從測(cè)試我們可以看到使用計(jì)數(shù)和計(jì)數(shù)鎖的區(qū)別。
(一).內(nèi)核維護(hù)了:
1).兩個(gè)鏈表,active_wake_locks[WAKE_LOCK_TYPE_COUNT]
?? ?active_wake_locks[0]維護(hù)的是suspend lock.
? ? active_wake_locks[1]維護(hù)的是idle lock.
2).一個(gè)鏈表,inactive_locks來記錄所有處于inactive狀態(tài)的鎖.
??
(二). 下面講述應(yīng)用層申請(qǐng)的鎖怎么傳到kernel下面的,來理解整個(gè)wakelock的框架。
比如/sys/power/wake_lock 下面的PowerManagerService的生成過程。
? ? ?
1). Android 提供了現(xiàn)成 android.os.PowerManager 類 ,類中提供 newWakeLock(int flags, String tag)方法來取得相應(yīng)層次的鎖, 此函數(shù)的定義?frameworks/base/core/java/android/os/PowerManager.java
? ? ? 下面,應(yīng)用程序在申請(qǐng)wake_lock時(shí)都會(huì)有調(diào)用。
? ? ? ?實(shí)例:
? ? ? ? PowerManager pm = (PowerManager)getSystemService(Context.POWER_SERVICE);
? ? ? ? PowerManager.WakeLock wl = pm.newWakeLock
? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?(PowerManager.SCREEN_DIM_WAKE_LOCK, “My Tag”);
? ? ? ? wl.acquire();//申請(qǐng)鎖這個(gè)里面會(huì)調(diào)用PowerManagerService里面acquireWakeLock()
***********************
? ? ? ? wl.release(); //釋放鎖,顯示的釋放,如果申請(qǐng)的鎖不在此釋放系統(tǒng)就不會(huì)進(jìn)入休眠。
?
2). frameworks層
/frameworks/base/services/java/com/android/server/PowerManagerService.java
這個(gè)類是來管理所有的應(yīng)用程序申請(qǐng)的wakelock。比如音視頻播放器,camera等申請(qǐng)的wakelock都是通過這個(gè)類來管理的。
static final String PARTIAL_NAME = "PowerManagerService"
Power.acquireWakeLock(Power.PARTIAL_WAKE_LOCK,PARTIAL_NAME);
這個(gè)函數(shù)調(diào)用Power類里面的acquireWakeLock(),此時(shí)的PARTIAL_NAME作為參數(shù)傳遞到底層去。
/frameworks/base/core/java/android/os/Power.java
public static native void acquireWakeLock(int lock, String id);
注:native申明的方法在Power類中沒有實(shí)現(xiàn),其實(shí)現(xiàn)體在frameworks/base/core/jni/android_os_Power.cpp中,所?以調(diào)用Power類的acquireWakeLock()方法時(shí)會(huì)調(diào)用JNI下的實(shí)現(xiàn)方法。
3).JNI層的實(shí)現(xiàn)
路徑: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()中調(diào)用了?路徑下hardware/libhardware_legacy/power/power.c下面的acquire_wake_lock(lock, id)
4).與kernel層的交互
在power.c下的acquire_wake_lock(lock, id)函數(shù)如下:
int acquire_wake_lock(int lock, const char* id)
{ ??
? ? ?return write(fd, id, strlen(id));
}
注: fd就是文件描述符,在此表示”/sys/power/wake_lock”
? ? ? ? ?id就是從PowerManagerService類中傳下來的參數(shù)即:
static?long?has_wake_lock_locked(int?type)?? {?? ????struct?wake_lock?*lock,?*n;?? ????long?max_timeout?=?0;?? ?? ?? ????BUG_ON(type?>=?WAKE_LOCK_TYPE_COUNT);?? ????list_for_each_entry_safe(lock,?n,?&active_wake_locks[type],?link)?{?? ????????if?(lock->flags?&?WAKE_LOCK_AUTO_EXPIRE)?{?? ????????????long?timeout?=?lock->expires?-?jiffies;?? ????????????if?(timeout?<=?0)?? ????????????????expire_wake_lock(lock);?? ????????????else?if?(timeout?>?max_timeout)?? ????????????????max_timeout?=?timeout;?? ????????}?else?? ????????????printk(">>%s?lock.name?==?%s\n",__FUNCTION__,lock->name);?? ????????????return?-1;?? ????}?? ????return?max_timeout;?? }??
PowerManager.WakeLock 有加鎖和解鎖兩種狀態(tài),加鎖的方式有兩種,一種是永久的鎖住,這樣的鎖除非顯式的放開,是不會(huì)解鎖的,所以這種鎖用起來要非常的小心。第二種鎖是超時(shí)鎖,這種鎖會(huì)在鎖住后一段時(shí)間解鎖。在創(chuàng)建了 PowerManager.WakeLock 后,有兩種機(jī)制,第一種是不計(jì)數(shù)鎖機(jī)制,另一種是計(jì)數(shù)鎖機(jī)制。可以通過 setReferenceCounted(boolean value) 來指定,一般默認(rèn)為計(jì)數(shù)機(jī)制。這兩種機(jī)制的區(qū)別在于,前者無論 acquire() 了多少次,只要通過一次 release()即可解鎖。而后者正真解鎖是在( --count == 0 )的時(shí)候,同樣當(dāng) (count == 0) 的時(shí)候才會(huì)去申請(qǐng)加鎖,其他情況 isHeld 狀態(tài)是不會(huì)改變的。所以 PowerManager.WakeLock 的計(jì)數(shù)機(jī)制并不是正真意義上的對(duì)每次請(qǐng)求進(jìn)行申請(qǐng)/釋放每一把鎖,它只是對(duì)同一把鎖被申請(qǐng)/釋放的次數(shù)進(jìn)行了統(tǒng)計(jì)再正真意義上的去操作。一下進(jìn)行了永久鎖的測(cè)試: 從測(cè)試我們可以看到使用計(jì)數(shù)和計(jì)數(shù)鎖的區(qū)別。
(一).內(nèi)核維護(hù)了:
1).兩個(gè)鏈表,active_wake_locks[WAKE_LOCK_TYPE_COUNT]
?? ?active_wake_locks[0]維護(hù)的是suspend lock.
? ? active_wake_locks[1]維護(hù)的是idle lock.
2).一個(gè)鏈表,inactive_locks來記錄所有處于inactive狀態(tài)的鎖.
??
(二). 下面講述應(yīng)用層申請(qǐng)的鎖怎么傳到kernel下面的,來理解整個(gè)wakelock的框架。
比如/sys/power/wake_lock 下面的PowerManagerService的生成過程。
? ? ?
1). Android 提供了現(xiàn)成 android.os.PowerManager 類 ,類中提供 newWakeLock(int flags, String tag)方法來取得相應(yīng)層次的鎖, 此函數(shù)的定義?frameworks/base/core/java/android/os/PowerManager.java
? ? ? 下面,應(yīng)用程序在申請(qǐng)wake_lock時(shí)都會(huì)有調(diào)用。
? ? ? ?實(shí)例:
? ? ? ? PowerManager pm = (PowerManager)getSystemService(Context.POWER_SERVICE);
? ? ? ? PowerManager.WakeLock wl = pm.newWakeLock
? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?(PowerManager.SCREEN_DIM_WAKE_LOCK, “My Tag”);
? ? ? ? wl.acquire();//申請(qǐng)鎖這個(gè)里面會(huì)調(diào)用PowerManagerService里面acquireWakeLock()
***********************
? ? ? ? wl.release(); //釋放鎖,顯示的釋放,如果申請(qǐng)的鎖不在此釋放系統(tǒng)就不會(huì)進(jìn)入休眠。
?
2). frameworks層
/frameworks/base/services/java/com/android/server/PowerManagerService.java
這個(gè)類是來管理所有的應(yīng)用程序申請(qǐng)的wakelock。比如音視頻播放器,camera等申請(qǐng)的wakelock都是通過這個(gè)類來管理的。
static final String PARTIAL_NAME = "PowerManagerService"
Power.acquireWakeLock(Power.PARTIAL_WAKE_LOCK,PARTIAL_NAME);
這個(gè)函數(shù)調(diào)用Power類里面的acquireWakeLock(),此時(shí)的PARTIAL_NAME作為參數(shù)傳遞到底層去。
/frameworks/base/core/java/android/os/Power.java
public static native void acquireWakeLock(int lock, String id);
注:native申明的方法在Power類中沒有實(shí)現(xiàn),其實(shí)現(xiàn)體在frameworks/base/core/jni/android_os_Power.cpp中,所?以調(diào)用Power類的acquireWakeLock()方法時(shí)會(huì)調(diào)用JNI下的實(shí)現(xiàn)方法。
3).JNI層的實(shí)現(xiàn)
路徑: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()中調(diào)用了?路徑下hardware/libhardware_legacy/power/power.c下面的acquire_wake_lock(lock, id)
4).與kernel層的交互
在power.c下的acquire_wake_lock(lock, id)函數(shù)如下:
int acquire_wake_lock(int lock, const char* id)
{ ??
? ? ?return write(fd, id, strlen(id));
}
注: fd就是文件描述符,在此表示”/sys/power/wake_lock”
? ? ? ? ?id就是從PowerManagerService類中傳下來的參數(shù)即:
? ? ? ? ?PARTIAL_NAME = "PowerManagerService"? 到此就是通過文件系統(tǒng)來與kernel層交互的地方。
(三)。wakelocks調(diào)試:
[cpp]?view plaincopy
在內(nèi)核打印休眠時(shí)沒釋放鎖的名稱。
1.要想能夠在休眠的時(shí)候喚醒,必須使用?RTC_WAKEUP,并且賦予<uses-permission android:name="android.permission.WAKE_LOCK">? 系統(tǒng)喚醒的權(quán)限。
否則,即使使用了RTC_WAKEUP也喚醒不了設(shè)備。
總結(jié)
以上是生活随笔為你收集整理的Android wakelock 自上而下浅析的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 电脑连接不上wifi,怎么办?
- 下一篇: Android studio JNI j