在前文的分析中,我們略去了CBS中與權限管理相關的部分,本節將集中討論這個問題。先來回顧一下CBS中和權限管理相關的函數調用。
~~~
//copy方設置ClipData在CBS的setPrimaryClip函數中進行:
checkDataOwnerLocked(clip,Binder.getCallingUid());
clearActiveOwnersLocked();
//paste方獲取ClipData在CBS的getPrimaryClip函數中進行:
addActiveOwnerLocked(Binder.getCallingUid(), pkg);
~~~
在分析這3個函數之前,不妨先介紹一下Android系統中的URI權限管理。
1. URI權限管理介紹
Android系統的權限管理中有一類是專門針對URI的,先來看一個示例,該例來自package/providers/ContactsProvider,在它的AndroidManifest.xml中有如下聲明:
**AndroidManifest.xml**
~~~
<providerandroid:name="ContactsProvider2"
......
android:readPermission="android.permission.READ_CONTACTS"
android:writePermission="android.permission.WRITE_CONTACTS">
......
<grant-uri-permission android:pathPattern=".*" />
</provider>
~~~
這里聲明了一個名為“ContactsProvider2“的ContentProvider,并定義了幾個權限聲明,下面對其進行解釋。
- readPermission:要求調用query函數的客戶端必須聲明use-permission為READ_CONTACTS。
- writePermission:要求調用update或insert函數的客戶端必須聲明use-permission為WRITE_CONTACTS。
- grant-uri-permission:和授權有關。
初識grant-uri-permission時,會覺得比較難以理解,現在通過舉例分析幫助讀者加深認識。
- Contacts和ContactProvider這兩個APP都是由系統提供的程序,而且二者關系緊密,所以Contacts一定會聲明READ_CONTACTS和WRITE_CONTACT的權限。如此,Contacts就可以毫無阻礙地通過ContactsProvider來查詢或更新數據庫了。
- 假設Contacts新增一個功能,將ContactsProvider中的某條數據copy到剪切板。根據前面已介紹過的知識可以知道,Contacts會向剪切板中復制一個URI類型的數據。
- 另外一個程序從剪切板中paste這條數據,由于是URI類型的,所以此程序會通過ContentResolver來query該URI所指向的數據。但是這個程序卻并未聲明READ_CONTACTS的權限,所以它query數據時必然會失敗。
或許有人會問,為什么第三個程序不聲明相應權限呢?原因很簡單,第三個程序不知道自己該申明怎樣的權限(除非這兩個程序的開發者能互通信息)。本例ContactsProvider設置的讀權限是READ_CONTACTS,以后可能換成READ_CONTACTS_EXTEND。為了解決類似問題,Android提供了一套專門針對URI的權限管理機制。以這套機制解決示例中權限聲明問題的方法是這樣的:當第三個程序從剪切板中paste數據時,系統會判斷是否需要為這個程序授權。當然,系統不會隨意授權,而是需要考慮ContactsProvider的情況。因為ContactsProvider聲明了grant-uri-permission,并且所paste的URI匹配其中的pathPattern,所以授權就能成功。倘若ContactsProvider沒有聲明grant-uri-permission,或者URI不匹配指定的pathPattern,則授權失敗。
有了前面介紹的權限管理機制,相信下面CBS中的權限管理理解起來就比較簡單了。
感興趣的讀者可閱讀SDK安裝目錄下/docs/guide/topics/security/security.html中關于URI Permission的說明部分。
2. checkDataOwnerLocked函數分析
checkDataOwnerLocked函數的代碼如下:
**ClipboardService.java**
~~~
private final void checkDataOwnerLocked(ClipDatadata, int uid) {
//第二個參數uid為copy方進程的uid
final int N = data.getItemCount();
for(int i=0; i<N; i++) {
//為每一個item調用checkItemOwnerLocked
checkItemOwnerLocked(data.getItemAt(i), uid);
}
}
// checkItemOwnerLocked函數分析
private final voidcheckItemOwnerLocked(ClipData.Item item, int uid) {
if(item.getUri() != null) {//檢查Uri
checkUriOwnerLocked(item.getUri(), uid);
}
Intent intent = item.getIntent();
//getData函數返回的也是一個URI,因此這里實際上檢查的也是URI
if(intent != null && intent.getData() != null) {
checkUriOwnerLocked(intent.getData(), uid);
}
}
~~~
權限檢查就是針對URI的,因為URI所指向的數據可能是系統內部使用或私密的。例如Setting數據中的Secure表,這里的數據不能隨意訪問。雖然直接使用ContentResolver訪問這些數據時系統會進行權限檢查,但是由于目前的剪切板服務也支持URI數據類型,所以這里也需要做檢查,否則惡意程序就能輕松讀取私密信息。
下邊來分析checkUriOwnerLocked函數,其代碼如下:
**ClipboardService.java**
~~~
private final void checkUriOwnerLocked(Uri uri,int uid) {
......
longident = Binder.clearCallingIdentity();
boolean allowed = false;
try{
/*
調用ActivityManagerService的checkGrantUriPermission函數,
該函數內部將檢查copy方是否能被賦予URI_READ權限。如果不允許,
該函數會拋SecurityException異常
*/
mAm.checkGrantUriPermission(uid, null, uri,
Intent.FLAG_GRANT_READ_URI_PERMISSION);
}catch (RemoteException e) {
}finally {
Binder.restoreCallingIdentity(ident);
}
}
~~~
根據前面的知識,這里先要檢查copy方是否有讀取URI的權限。下面來分析paste方的權限管理
3. clearActiveOwnersLocked函數分析
clearActiveOwnersLocked函數的代碼如下:
**ClipboardService.java**
~~~
private final void addActiveOwnerLocked(int uid,String pkg) {
PackageInfo pi;
try{
/*
調用PackageManagerService的getPackageInfo函數得到相關信息
然后做一次安全檢查,如果PacakgeInfo的uid信息和當前調用的uid不一致,
則拋SecurityException。這個很好理解,因為paste方可以傳遞虛假的
packagename,但uid是沒法造假的
*/
pi = mPm.getPackageInfo(pkg, 0);
if (pi.applicationInfo.uid != uid) {
throw new SecurityException("Calling uid " + uid
+ " does not ownpackage " + pkg);
}
} ......
}
//mActivePermissionOwners用來保存已經通過安全檢查的package
if(mPrimaryClip != null && !mActivePermissionOwners.contains(pkg)) {
//針對ClipData中的每一個Item,都需要調用grantItemLocked來檢查權限
final int N = mPrimaryClip.getItemCount();
for (int i=0; i<N; i++) {
grantItemLocked(mPrimaryClip.getItemAt(i), pkg);
}//保存package信息到mActivePermissionOwners
mActivePermissionOwners.add(pkg);
}
}
//grantItemLocked分析
private final void grantItemLocked(ClipData.Itemitem, String pkg) {
if(item.getUri() != null) {
grantUriLocked(item.getUri(), pkg);
} //和copy方一樣,這里僅檢查URI的情況
Intent intent = item.getIntent();
if(intent != null && intent.getData() != null) {
grantUriLocked(intent.getData(), pkg);
}
}
~~~
再來看grantUriLocked的代碼:
**ClipboardService.java**
~~~
private final void grantUriLocked(Uri uri, Stringpkg) {
longident = Binder.clearCallingIdentity();
try{
/*
調用ActivityManagerService的grantUriPermissionFromOwner函數,
注意第二個參數傳遞的是CBS所在進程的uid。該函數內部也會檢查權限。
該函數調用成功后,paste方就被授予了對應URI的讀權限
*/
mAm.grantUriPermissionFromOwner(mPermissionOwner,
Process.myUid(),pkg, uri,
Intent.FLAG_GRANT_READ_URI_PERMISSION);
}catch (RemoteException e) {
} finally {
Binder.restoreCallingIdentity(ident);
}
}
~~~
既然有授權,那么客戶端使用完畢后就需要撤銷授權,這個工作是在setPrimaryClip函數的clearActiveOwnersLocked中完成的。當為剪切板設置新的ClipData時,自然需要將與舊ClipData相關的權限撤銷。讀者可自行分析clearActiveOwnersLocked函數。
- 前言
- 第1章 搭建Android源碼工作環境
- 1.1 Android系統架構
- 1.2 搭建開發環境
- 1.2.1 下載源碼
- 1.2.2 編譯源碼
- 1.2.3 利用Eclipse調試system_process
- 1.3 本章小結
- 第2章 深入理解Java Binder和MessageQueue
- 2.1 概述
- 2.2 Java層中的Binder架構分析
- 2.2.1 Binder架構總覽
- 2.2.2 初始化Java層Binder框架
- 2.2.3 addService實例分析
- 2.2.4 Java層Binder架構總結
- 2.3 心系兩界的MessageQueue
- 2.3.1 MessageQueue的創建
- 2.3.2 提取消息
- 2.3.3 nativePollOnce函數分析
- 2.3.4 MessageQueue總結
- 2.4 本章小結
- 第3章 深入理解SystemServer
- 3.1 概述
- 3.2 SystemServer分析
- 3.2.1 main函數分析
- 3.2.2 Service群英會
- 3.3 EntropyService分析
- 3.4 DropBoxManagerService分析
- 3.4.1 DBMS構造函數分析
- 3.4.2 dropbox日志文件的添加
- 3.4.3 DBMS和settings數據庫
- 3.5 DiskStatsService和DeviceStorageMonitorService分析
- 3.5.1 DiskStatsService分析
- 3.5.2 DeviceStorageManagerService分析
- 3.6 SamplingProfilerService分析
- 3.6.1 SamplingProfilerService構造函數分析
- 3.6.2 SamplingProfilerIntegration分析
- 3.7 ClipboardService分析
- 3.7.1 復制數據到剪貼板
- 3.7.2 從剪切板粘貼數據
- 3.7.3 CBS中的權限管理
- 3.8 本章小結
- 第4章 深入理解PackageManagerService
- 4.1 概述
- 4.2 初識PackageManagerService
- 4.3 PKMS的main函數分析
- 4.3.1 構造函數分析之前期準備工作
- 4.3.2 構造函數分析之掃描Package
- 4.3.3 構造函數分析之掃尾工作
- 4.3.4 PKMS構造函數總結
- 4.4 APK Installation分析
- 4.4.1 adb install分析
- 4.4.2 pm分析
- 4.4.3 installPackageWithVerification函數分析
- 4.4.4 APK 安裝流程總結
- 4.4.5 Verification介紹
- 4.5 queryIntentActivities分析
- 4.5.1 Intent及IntentFilter介紹
- 4.5.2 Activity信息的管理
- 4.5.3 Intent 匹配查詢分析
- 4.5.4 queryIntentActivities總結
- 4.6 installd及UserManager介紹
- 4.6.1 installd介紹
- 4.6.2 UserManager介紹
- 4.7 本章學習指導
- 4.8 本章小結
- 第5章 深入理解PowerManagerService
- 5.1 概述
- 5.2 初識PowerManagerService
- 5.2.1 PMS構造函數分析
- 5.2.2 init分析
- 5.2.3 systemReady分析
- 5.2.4 BootComplete處理
- 5.2.5 初識PowerManagerService總結
- 5.3 PMS WakeLock分析
- 5.3.1 WakeLock客戶端分析
- 5.3.2 PMS acquireWakeLock分析
- 5.3.3 Power類及LightService類介紹
- 5.3.4 WakeLock總結
- 5.4 userActivity及Power按鍵處理分析
- 5.4.1 userActivity分析
- 5.4.2 Power按鍵處理分析
- 5.5 BatteryService及BatteryStatsService分析
- 5.5.1 BatteryService分析
- 5.5.2 BatteryStatsService分析
- 5.5.3 BatteryService及BatteryStatsService總結
- 5.6 本章學習指導
- 5.7 本章小結
- 第6章 深入理解ActivityManagerService
- 6.1 概述
- 6.2 初識ActivityManagerService
- 6.2.1 ActivityManagerService的main函數分析
- 6.2.2 AMS的 setSystemProcess分析
- 6.2.3 AMS的 installSystemProviders函數分析
- 6.2.4 AMS的 systemReady分析
- 6.2.5 初識ActivityManagerService總結
- 6.3 startActivity分析
- 6.3.1 從am說起
- 6.3.2 AMS的startActivityAndWait函數分析
- 6.3.3 startActivityLocked分析
- 6.4 Broadcast和BroadcastReceiver分析
- 6.4.1 registerReceiver流程分析
- 6.4.2 sendBroadcast流程分析
- 6.4.3 BROADCAST_INTENT_MSG消息處理函數
- 6.4.4 應用進程處理廣播分析
- 6.4.5 廣播處理總結
- 6.5 startService之按圖索驥
- 6.5.1 Service知識介紹
- 6.5.2 startService流程圖
- 6.6 AMS中的進程管理
- 6.6.1 Linux進程管理介紹
- 6.6.2 關于Android中的進程管理的介紹
- 6.6.3 AMS進程管理函數分析
- 6.6.4 AMS進程管理總結
- 6.7 App的 Crash處理
- 6.7.1 應用進程的Crash處理
- 6.7.2 AMS的handleApplicationCrash分析
- 6.7.3 AppDeathRecipient binderDied分析
- 6.7.4 App的Crash處理總結
- 6.8 本章學習指導
- 6.9 本章小結
- 第7章 深入理解ContentProvider
- 7.1 概述
- 7.2 MediaProvider的啟動及創建
- 7.2.1 Context的getContentResolver函數分析
- 7.2.2 MediaStore.Image.Media的query函數分析
- 7.2.3 MediaProvider的啟動及創建總結
- 7.3 SQLite創建數據庫分析
- 7.3.1 SQLite及SQLiteDatabase家族
- 7.3.2 MediaProvider創建數據庫分析
- 7.3.3 SQLiteDatabase創建數據庫的分析總結
- 7.4 Cursor 的query函數的實現分析
- 7.4.1 提取query關鍵點
- 7.4.2 MediaProvider 的query分析
- 7.4.3 query關鍵點分析
- 7.4.4 Cursor query實現分析總結
- 7.5 Cursor close函數實現分析
- 7.5.1 客戶端close的分析
- 7.5.2 服務端close的分析
- 7.5.3 finalize函數分析
- 7.5.4 Cursor close函數總結
- 7.6 ContentResolver openAssetFileDescriptor函數分析
- 7.6.1 openAssetFileDescriptor之客戶端調用分析
- 7.6.2 ContentProvider的 openTypedAssetFile函數分析
- 7.6.3 跨進程傳遞文件描述符的探討
- 7.6.4 openAssetFileDescriptor函數分析總結
- 7.7 本章學習指導
- 7.8 本章小結
- 第8章 深入理解ContentService和AccountManagerService
- 8.1 概述
- 8.2 數據更新通知機制分析
- 8.2.1 初識ContentService
- 8.2.2 ContentResovler 的registerContentObserver分析
- 8.2.3 ContentResolver的 notifyChange分析
- 8.2.4 數據更新通知機制總結和深入探討
- 8.3 AccountManagerService分析
- 8.3.1 初識AccountManagerService
- 8.3.2 AccountManager addAccount分析
- 8.3.3 AccountManagerService的分析總結
- 8.4 數據同步管理SyncManager分析
- 8.4.1 初識SyncManager
- 8.4.2 ContentResolver 的requestSync分析
- 8.4.3 數據同步管理SyncManager分析總結
- 8.5 本章學習指導
- 8.6 本章小結