日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 运维知识 > Android >内容正文

Android

Android wakelock 自上而下浅析

發布時間:2023/12/20 Android 30 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Android wakelock 自上而下浅析 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.
Wake Lock是一種鎖的機制, 只要有人拿著這個鎖,系統就無法進入休眠,可以被用戶態程序和內核獲得. 這個鎖可以是有超時的或者是沒有超時的,超時的鎖會在時間過去以后自動解鎖.?如果沒有鎖了或者超時了, 內核就會啟動休眠的那套機制來進入休眠.
PowerManager.WakeLock 有加鎖和解鎖兩種狀態,加鎖的方式有兩種,一種是永久的鎖住,這樣的鎖除非顯式的放開,是不會解鎖的,所以這種鎖用起來要非常的小心。第二種鎖是超時鎖,這種鎖會在鎖住后一段時間解鎖。在創建了 PowerManager.WakeLock 后,有兩種機制,第一種是不計數鎖機制,另一種是計數鎖機制。可以通過 setReferenceCounted(boolean value) 來指定,一般默認為計數機制。這兩種機制的區別在于,前者無論 acquire() 了多少次,只要通過一次 release()即可解鎖。而后者正真解鎖是在( --count == 0 )的時候,同樣當 (count == 0) 的時候才會去申請加鎖,其他情況 isHeld 狀態是不會改變的。所以 PowerManager.WakeLock 的計數機制并不是正真意義上的對每次請求進行申請/釋放每一把鎖,它只是對同一把鎖被申請/釋放的次數進行了統計再正真意義上的去操作。一下進行了永久鎖的測試: 從測試我們可以看到使用計數和計數鎖的區別。


(一).內核維護了:
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層交互的地方。

(三)。wakelocks調試:

[cpp]?view plaincopy
  • 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;??
  • }??

  • 在內核打印休眠時沒釋放鎖的名稱。

    1.要想能夠在休眠的時候喚醒,必須使用?RTC_WAKEUP,并且賦予<uses-permission android:name="android.permission.WAKE_LOCK">? 系統喚醒的權限。

    否則,即使使用了RTC_WAKEUP也喚醒不了設備。

    總結

    以上是生活随笔為你收集整理的Android wakelock 自上而下浅析的全部內容,希望文章能夠幫你解決所遇到的問題。

    如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。