Android 外掛化原理解析(3):Hook 機制之 Binder Hook

weishu發表於2016-05-16

Android系統通過Binder機制給應用程式提供了一系列的系統服務,諸如ActivityManagerServiceClipboardManagerAudioManager等;這些廣泛存在系統服務給應用程式提供了諸如任務管理,音訊,視訊等異常強大的功能。

外掛框架作為各個外掛的管理者,為了使得外掛能夠無縫地使用這些系統服務,自然會對這些系統服務做出一定的改造(Hook),使得外掛的開發和使用更加方便,從而大大降低外掛的開發和維護成本。比如,Hook住ActivityManagerService可以讓外掛無縫地使用startActivity方法而不是使用特定的方式(比如that語法)來啟動外掛或者主程式的任意介面。

我們把這種Hook系統服務的機制稱之為Binder Hook,因為本質上這些服務提供者都是存在於系統各個程式的Binder物件。因此,要理解接下來的內容必須瞭解Android的Binder機制,可以參考我之前的文章Binder學習指南

閱讀本文之前,可以先clone一份 understand-plugin-framework,參考此專案的binder-hook模組。另外,外掛框架原理解析系列文章見索引

系統服務的獲取過程

我們知道系統的各個遠端service物件都是以Binder的形式存在的,而這些Binder有一個管理者,那就是ServiceManager;我們要Hook掉這些service,自然要從這個ServiceManager下手,不然星羅棋佈的Binder廣泛存在於系統的各個角落,要一個個找出來還真是大海撈針。

回想一下我們使用系統服務的時候是怎麼幹的,想必這個大家一定再熟悉不過了:通過Context物件的getSystemService方法;比如要使用ActivityManager

可是這個貌似跟ServiceManager沒有什麼關係啊?我們再檢視getSystemService方法;(Context的實現在ContextImpl裡面):

很簡單,所有的service物件都儲存在一張map裡面,我們再看這個map是怎麼初始化的:

ContextImpl的靜態初始化塊裡面,有的Service是像上面這樣初始化的;可以看到,確實使用了ServiceManager;當然還有一些service並沒有直接使用ServiceManager,而是做了一層包裝並返回了這個包裝物件,比如我們的ActivityManager,它返回的是ActivityManager這個包裝物件:

但是在ActivityManager這個類內部,也使用了ServiceManager;具體來說,因為ActivityManager裡面所有的核心操作都是使用ActivityManagerNative.getDefault()完成的。那麼這個語句幹了什麼呢?

因此,通過分析我們得知,系統Service的使用其實就分為兩步:

尋找Hook點

Android 外掛化原理解析(2):Hook 機制之動態代理裡面我們說過,Hook分為三步,最關鍵的一步就是尋找Hook點。我們現在已經搞清楚了系統服務的使用過程,那麼就需要找出在這個過程中,在哪個環節是最合適hook的。

由於系統服務的使用者都是對第二步獲取到的IXXInterface進行操作,因此如果我們要hook掉某個系統服務,只需要把第二步的asInterface方法返回的物件修改為為我們Hook過的物件就可以了。

asInterface過程

接下來我們分析asInterface方法,然後想辦法把這個方法的返回值修改為我們Hook過的系統服務物件。這裡我們以系統剪下版服務為例,原始碼位置為android.content.IClipboard,IClipboard.Stub.asInterface方法程式碼如下:

這個方法的意思就是:先檢視本程式是否存在這個Binder物件,如果有那麼直接就是本程式呼叫了;如果不存在那麼建立一個代理物件,讓代理物件委託驅動完成跨程式呼叫。

觀察這個方法,前面的那個if語句判空返回肯定動不了手腳;最後一句呼叫建構函式然後直接返回我們也是無從下手,要修改asInterface方法的返回值,我們唯一能做的就是從這一句下手:

我們可以嘗試修改這個obj物件的queryLocalInterface方法的返回值,並保證這個返回值符合接下來的if條件檢測,那麼就達到了修改asInterface方法返回值的目的。

而這個obj物件剛好是我們第一步返回的IBinder物件,接下來我們嘗試對這個IBinder物件的queryLocalInterface方法進行hook。

getService過程

上文分析得知,我們想要修改IBinder物件的queryLocalInterface方法;獲取IBinder物件的過程如下:

因此,我們希望能修改這個getService方法的返回值,讓這個方法返回一個我們偽造過的IBinder物件;這樣,我們可以在自己偽造的IBinder物件的queryLocalInterface方法作處理,進而使得asInterface方法返回在queryLocalInterface方法裡面處理過的值,最終實現hook系統服務的目的。

在跟蹤這個getService方法之前我們思考一下,由於系統服務是一系列的遠端Service,它們的本體,也就是Binder本地物件一般都存在於某個單獨的程式,在這個程式之外的其他程式存在的都是這些Binder本地物件的代理。因此在我們的程式裡面,存在的也只是這個Binder代理物件,我們也只能對這些Binder代理物件下手。(如果這一段看不懂,建議不要往下看了,先看Binder學習指南)

然後,這個getService是一個靜態方法,如果此方法什麼都不做,拿到Binder代理物件之後直接返回;那麼我們就無能為力了:我們沒有辦法攔截一個靜態方法,也沒有辦法獲取到這個靜態方法裡面的區域性變數(即我們希望修改的那個Binder代理物件)。

接下來就可以看這個getService的程式碼了:

天無絕人之路!ServiceManager為了避免每次都進行跨程式通訊,把這些Binder代理物件快取在一張map裡面。

我們可以替換這個map裡面的內容為Hook過的IBinder物件,由於系統在getService的時候每次都會優先查詢快取,因此返回給使用者的都是被我們修改過的物件,從而達到瞞天過海的目的。

總結一下,要達到修改系統服務的目的,我們需要如下兩步:

  1. 首先肯定需要偽造一個系統服務物件,接下來就要想辦法讓asInterface能夠返回我們的這個偽造物件而不是原始的系統服務物件。
  2. 通過上文分析我們知道,只要讓getService返回IBinder物件的queryLocalInterface方法直接返回我們偽造過的系統服務物件就能達到目的。所以,我們需要偽造一個IBinder物件,主要是修改它的queryLocalInterface方法,讓它返回我們偽造的系統服務物件;然後把這個偽造物件放置在ServiceManager的快取map裡面即可。

我們通過Binder機制的優先查詢本地Binder物件的這個特性達到了Hook掉系統服務物件的目的。因此queryLocalInterface也失去了它原本的意義(只查詢本地Binder物件,沒有本地物件返回null),這個方法只是一個傀儡,是我們實現hook系統物件的橋樑:我們通過這個“漏洞”讓asInterface永遠都返回我們偽造過的物件。由於我們接管了asInterface這個方法的全部,我們偽造過的這個系統服務物件不能是隻擁有本地Binder物件(原始queryLocalInterface方法返回的物件)的能力,還要有Binder代理物件操縱驅動的能力。

接下來我們就以Hook系統的剪下版服務為例,用實際程式碼來說明,如何Hook掉系統服務。

Hook系統剪下版服務

偽造剪下版服務物件

首先我們用代理的方式偽造一個剪下版服務物件,關於如何使用代理的方式進行hook以及其中的原理,可以檢視Android 外掛化原理解析(2):Hook 機制之動態代理

具體程式碼如下,我們用動態代理的方式Hook掉了hasPrimaryClip()getPrimaryClip()這兩個方法:

注意,我們拿到原始的IBinder物件之後,如果我們希望使用被Hook之前的系統服務,並不能直接使用這個IBinder物件,而是需要使用asInterface方法將它轉換為IClipboard介面;因為getService方法返回的IBinder實際上是一個裸Binder代理物件,它只有與驅動打交道的能力,但是它並不能獨立工作,需要人指揮它;asInterface方法返回的IClipboard.Stub.Proxy類的物件通過操縱這個裸BinderProxy物件從而實現了具體的IClipboard介面定義的操作。

偽造IBinder 物件

在上一步中,我們已經偽造好了系統服務物件,現在要做的就是想辦法讓asInterface方法返回我們偽造的物件了;我們偽造一個IBinder物件:

我們使用動態代理的方式偽造了一個跟原始IBinder一模一樣的物件,然後在這個偽造的IBinder物件的queryLocalInterface方法裡面返回了我們第一步建立的偽造過的系統服務物件;注意看註釋,詳細解釋可以看程式碼

替換ServiceManager的IBinder物件

現在就是萬事具備,只欠東風了;我們使用反射的方式修改ServiceManager類裡面快取的Binder物件,使得getService方法返回我們偽造的IBinder物件,進而asInterface方法使用偽造IBinder物件的queryLocalInterface方法返回了我們偽造的系統服務物件。程式碼較簡單,如下:

接下來,在app裡面使用剪下版,比如長按進行貼上之後,剪下版的內容永遠都是you are hooked了;這樣,我們Hook系統服務的目的宣告完成!詳細的程式碼參見 github

也許你會問,外掛框架會這麼hook嗎?如果不是那麼外掛框架hook這些幹什麼?外掛框架當然不會做替換文字這麼無聊的事情,DroidPlugin外掛框架管理外掛使得外掛就像是主程式一樣,因此外掛需要使用主程式的剪下版,外掛之間也會共用剪下版;其他的一些系統服務也類似,這樣就可以達到外掛和宿主程式之間的天衣服縫,水乳交融!另外,ActivityManager以及PackageManager這兩個系統服務雖然也可以通過這種方式hook,但是由於它們的重要性和特殊性,DroidPlugin使用了另外一種方式,我們會單獨講解。

喜歡就點個贊吧~持續更新,請關注github專案 understand-plugin-framework 和我的 部落格!

打賞支援我寫出更多好文章,謝謝!

打賞作者

打賞支援我寫出更多好文章,謝謝!

任選一種支付方式

Android 外掛化原理解析(3):Hook 機制之 Binder Hook Android 外掛化原理解析(3):Hook 機制之 Binder Hook

相關文章