日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

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

Android

Android Art Hook 技术方案

發布時間:2025/3/15 Android 43 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Android Art Hook 技术方案 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

Android Art Hook 技術方案

by 低端碼農 at 2015.4.13
www.im-boy.net

0x1 開始

Anddroid上的ART從5.0之后變成默認的選擇,可見ART的重要性,目前關于Dalvik Hook方面研究的文章很多,但我在網上卻找不到關于ART Hook相關的文章,甚至連鼎鼎大名的XPosed和Cydia Substrate到目前為止也不支持ART的Hook。當然我相信,技術方案他們肯定是的,估計卡在機型適配上的了。

既然網上找不到相關的資料,于是我決定自己花些時間去研究一下,終于黃天不負有心人,我找到了一個切實可行的方法,即本文所介紹的方法。

應該說明的是本文所介紹的方法肯定不是最好的,但大家看完本文之后,如果能啟發大家找到更好的ART Hook方法,那我拋磚引玉的目的就達到了。廢話不多說,我們開始吧。

  • 運行環境: 4.4.2 ART模式的模擬器
  • 開發環境: Mac OS X 10.10.3

0x2 ART類方法加載及執行

在ART中類方法的執行要比在Dalvik中要復雜得多,Dalvik如果除去JIT部分,可以理解為是一個解析執行的虛擬機,而ART則同時包含本地指令執行和解析執行兩種模式,同時所生成的oat文件也包含兩種類型,分別是portable和quick。portable和quick的主要區別是對于方法的加載機制不相同,quick大量使用了Lazy Load機制,因此應用的啟動速度更快,但加載流程更復雜。其中quick是作為默認選項,因此本文所涉及的技術分析都是基于quick類型的。

由于ART存在本地指令執行和解析執行兩種模式,因此類方法之間并不是能直接跳轉的,而是通過一些預先定義的bridge函數進行狀態和上下文的切換,這里引用一下老羅博客中的示意圖:

當執行某個方法時,如果當前是本地指令執行模式,則會執行ArtMethod::GetEntryPointFromCompiledCode()指向的函數,否則則執行ArtMethod::GetEntryPointFromInterpreter()指向的函數。因此每個方法,都有兩個入口點,分別保存在ArtMethod::entry_point_from_compiled_code_ArtMethod::entry_point_from_interpreter_。了解這一點非常重要,后面我們主要就是在這兩個入口做文章。

在講述原理之前,需要先把以下兩個流程了解清楚,這里的內容要展開是非常龐大的,我針對Hook的關鍵點,簡明扼要的描述一下,但還是強烈建議大家去老羅的博客里細讀一下其中關于ART的幾篇文章。

  • ArtMethod加載流程

這個過程發生在oat被裝載進內存并進行類方法鏈接的時候,類方法鏈接的代碼在art/runtime/class_linker.cc中的LinkCode,如下所示:

<code class="hljs lasso has-numbering">static <span class="hljs-literal">void</span> LinkCode(SirtRef<span class="hljs-subst"><</span>mirror<span class="hljs-tag">::ArtMethod</span><span class="hljs-subst">>&</span> method, const OatFile<span class="hljs-tag">::OatClass</span><span class="hljs-subst">*</span> oat_class, uint32_t method_index)SHARED_LOCKS_REQUIRED(Locks<span class="hljs-tag">::mutator_lock_</span>) {<span class="hljs-comment">// Method shouldn't have already been linked.</span>DCHECK(method<span class="hljs-subst">-></span>GetEntryPointFromCompiledCode() <span class="hljs-subst">==</span> <span class="hljs-built_in">NULL</span>);<span class="hljs-comment">// Every kind of method should at least get an invoke stub from the oat_method.</span><span class="hljs-comment">// non-abstract methods also get their code pointers.</span>const OatFile<span class="hljs-tag">::OatMethod</span> oat_method <span class="hljs-subst">=</span> oat_class<span class="hljs-subst">-></span>GetOatMethod(method_index);<span class="hljs-comment">// 這里默認會把method::entry_point_from_compiled_code_設置oatmethod的code</span>oat_method<span class="hljs-built_in">.</span>LinkMethod(method<span class="hljs-built_in">.</span>get());<span class="hljs-comment">// Install entry point from interpreter.</span>Runtime<span class="hljs-subst">*</span> runtime <span class="hljs-subst">=</span> Runtime<span class="hljs-tag">::Current</span>();bool enter_interpreter <span class="hljs-subst">=</span> NeedsInterpreter(method<span class="hljs-built_in">.</span>get(), method<span class="hljs-subst">-></span>GetEntryPointFromCompiledCode()); <span class="hljs-comment">//判斷方法是否需要解析執行</span><span class="hljs-comment">// 設置解析執行的入口點</span><span class="hljs-keyword">if</span> (enter_interpreter) {method<span class="hljs-subst">-></span>SetEntryPointFromInterpreter(interpreter<span class="hljs-tag">::artInterpreterToInterpreterBridge</span>);} <span class="hljs-keyword">else</span> {method<span class="hljs-subst">-></span>SetEntryPointFromInterpreter(artInterpreterToCompiledCodeBridge);}<span class="hljs-comment">// 下面是設置本地指令執行的入口點</span><span class="hljs-keyword">if</span> (method<span class="hljs-subst">-></span>IsAbstract()) {method<span class="hljs-subst">-></span>SetEntryPointFromCompiledCode(GetCompiledCodeToInterpreterBridge());<span class="hljs-keyword">return</span>;}<span class="hljs-comment">// 這里比較難理解,如果是靜態方法,但不是clinit,但需要把entry_point_from_compiled_code_設置為GetResolutionTrampoline的返回值</span><span class="hljs-keyword">if</span> (method<span class="hljs-subst">-></span>IsStatic() <span class="hljs-subst">&&</span> <span class="hljs-subst">!</span>method<span class="hljs-subst">-></span>IsConstructor()) {<span class="hljs-comment">// For static methods excluding the class initializer, install the trampoline.</span><span class="hljs-comment">// It will be replaced by the proper entry point by ClassLinker::FixupStaticTrampolines</span><span class="hljs-comment">// after initializing class (see ClassLinker::InitializeClass method).</span>method<span class="hljs-subst">-></span>SetEntryPointFromCompiledCode(GetResolutionTrampoline(runtime<span class="hljs-subst">-></span>GetClassLinker()));} <span class="hljs-keyword">else</span> <span class="hljs-keyword">if</span> (enter_interpreter) {<span class="hljs-comment">// Set entry point from compiled code if there's no code or in interpreter only mode.</span>method<span class="hljs-subst">-></span>SetEntryPointFromCompiledCode(GetCompiledCodeToInterpreterBridge());}<span class="hljs-keyword">if</span> (method<span class="hljs-subst">-></span>IsNative()) {<span class="hljs-comment">// Unregistering restores the dlsym lookup stub.</span>method<span class="hljs-subst">-></span>UnregisterNative(<span class="hljs-keyword">Thread</span><span class="hljs-tag">::Current</span>());}<span class="hljs-comment">// Allow instrumentation its chance to hijack code.</span>runtime<span class="hljs-subst">-></span>GetInstrumentation()<span class="hljs-subst">-></span>UpdateMethodsCode(method<span class="hljs-built_in">.</span>get(),method<span class="hljs-subst">-></span>GetEntryPointFromCompiledCode()); }</code><ul class="pre-numbering"><li>1</li><li>2</li><li>3</li><li>4</li><li>5</li><li>6</li><li>7</li><li>8</li><li>9</li><li>10</li><li>11</li><li>12</li><li>13</li><li>14</li><li>15</li><li>16</li><li>17</li><li>18</li><li>19</li><li>20</li><li>21</li><li>22</li><li>23</li><li>24</li><li>25</li><li>26</li><li>27</li><li>28</li><li>29</li><li>30</li><li>31</li><li>32</li><li>33</li><li>34</li><li>35</li><li>36</li><li>37</li><li>38</li><li>39</li><li>40</li><li>41</li><li>42</li><li>43</li><li>44</li><li>45</li><li>46</li><li>47</li><li>48</li></ul>

通過上面的代碼我們可以得到,一個ArtMethod的入口主要有以下幾種:

  • Interpreter2Interpreter對應artInterpreterToInterpreterBridge(art/runtime/interpreter/interpreter.cc);
  • Interpreter2CompledCode對應artInterpreterToCompiledCodeBridge(/art/runtime/entrypoints/interpreter/interpreter_entrypoints.cc);
  • CompliedCode2Interpreter對應art_quick_to_interpreter_bridge(art/runtime/arch/arm/quick_entrypoints_arm.S);
  • CompliedCode2ResolutionTrampoline對應art_quick_resolution_trampoline(art/runtime/arch/arm/quick_entrypoints_arm.S);
  • CompliedCode2CompliedCode這個入口是直接指向oat中的指令,詳細可見OatMethod::LinkMethod;
  • 其中調用約定主要有兩種,分別是:

  • typedef void (EntryPointFromInterpreter)(Thread* self, MethodHelper& mh, const DexFile::CodeItem* code_item, ShadowFrame* shadow_frame, JValue* result), 這種對應上述1,3兩種入口;
  • 剩下的2,4,5三種入口對應的是CompledCode的入口,代碼中并沒有直接給出,但我們通過分析ArtMethod::Invoke的方法調用,就可以知道其調用約定了。Invoke過程中會調用art_quick_invoke_stub(/art/runtime/arch/arm/quick_entrypoints_arm.S),代碼如下所示:

    <code class="hljs avrasm has-numbering"> <span class="hljs-comment">/** Quick invocation stub.* On entry:* r0 = method pointer* r1 = argument array or NULL for no argument methods* r2 = size of argument array in bytes* r3 = (managed) thread pointer* [sp] = JValue* result* [sp + 4] = result type char*/</span> ENTRY art_quick_invoke_stub <span class="hljs-keyword">push</span> {<span class="hljs-built_in">r0</span>, <span class="hljs-built_in">r4</span>, <span class="hljs-built_in">r5</span>, <span class="hljs-built_in">r9</span>, <span class="hljs-built_in">r11</span>, lr} @ spill regs <span class="hljs-preprocessor">.save</span> {<span class="hljs-built_in">r0</span>, <span class="hljs-built_in">r4</span>, <span class="hljs-built_in">r5</span>, <span class="hljs-built_in">r9</span>, <span class="hljs-built_in">r11</span>, lr} <span class="hljs-preprocessor">.pad</span> <span class="hljs-preprocessor">#24</span> <span class="hljs-preprocessor">.cfi</span>_adjust_cfa_offset <span class="hljs-number">24</span> <span class="hljs-preprocessor">.cfi</span>_rel_offset <span class="hljs-built_in">r0</span>, <span class="hljs-number">0</span> <span class="hljs-preprocessor">.cfi</span>_rel_offset <span class="hljs-built_in">r4</span>, <span class="hljs-number">4</span> <span class="hljs-preprocessor">.cfi</span>_rel_offset <span class="hljs-built_in">r5</span>, <span class="hljs-number">8</span> <span class="hljs-preprocessor">.cfi</span>_rel_offset <span class="hljs-built_in">r9</span>, <span class="hljs-number">12</span> <span class="hljs-preprocessor">.cfi</span>_rel_offset <span class="hljs-built_in">r11</span>, <span class="hljs-number">16</span> <span class="hljs-preprocessor">.cfi</span>_rel_offset lr, <span class="hljs-number">20</span> <span class="hljs-keyword">mov</span> <span class="hljs-built_in">r11</span>, sp @ save the stack pointer <span class="hljs-preprocessor">.cfi</span>_def_cfa_register <span class="hljs-built_in">r11</span> <span class="hljs-keyword">mov</span> <span class="hljs-built_in">r9</span>, <span class="hljs-built_in">r3</span> @ move managed thread pointer into <span class="hljs-built_in">r9</span> <span class="hljs-keyword">mov</span> <span class="hljs-built_in">r4</span>, <span class="hljs-preprocessor">#SUSPEND_CHECK_INTERVAL @ reset r4 to suspend check interval</span> <span class="hljs-keyword">add</span> <span class="hljs-built_in">r5</span>, <span class="hljs-built_in">r2</span>, <span class="hljs-preprocessor">#16 @ create space for method pointer in frame</span> <span class="hljs-keyword">and</span> <span class="hljs-built_in">r5</span>, <span class="hljs-preprocessor">#0xFFFFFFF0 @ align frame size to 16 bytes</span> <span class="hljs-keyword">sub</span> sp, <span class="hljs-built_in">r5</span> @ reserve stack space for argument array <span class="hljs-keyword">add</span> <span class="hljs-built_in">r0</span>, sp, <span class="hljs-preprocessor">#4 @ pass stack pointer + method ptr as dest for memcpy</span> bl memcpy @ memcpy (dest, src, bytes) ldr <span class="hljs-built_in">r0</span>, [<span class="hljs-built_in">r11</span>] @ restore method* ldr <span class="hljs-built_in">r1</span>, [sp, <span class="hljs-preprocessor">#4] @ copy arg value for r1</span> ldr <span class="hljs-built_in">r2</span>, [sp, <span class="hljs-preprocessor">#8] @ copy arg value for r2</span> ldr <span class="hljs-built_in">r3</span>, [sp, <span class="hljs-preprocessor">#12] @ copy arg value for r3</span> <span class="hljs-keyword">mov</span> ip, <span class="hljs-preprocessor">#0 @ set ip to 0</span> str ip, [sp] @ store NULL for method* at bottom of frame ldr ip, [<span class="hljs-built_in">r0</span>, <span class="hljs-preprocessor">#METHOD_CODE_OFFSET] @ get pointer to the code</span> blx ip @ <span class="hljs-keyword">call</span> the method <span class="hljs-keyword">mov</span> sp, <span class="hljs-built_in">r11</span> @ restore the stack pointer ldr ip, [sp, <span class="hljs-preprocessor">#24] @ load the result pointer</span> strd <span class="hljs-built_in">r0</span>, [ip] @ store <span class="hljs-built_in">r0</span>/<span class="hljs-built_in">r1</span> into result pointer <span class="hljs-keyword">pop</span> {<span class="hljs-built_in">r0</span>, <span class="hljs-built_in">r4</span>, <span class="hljs-built_in">r5</span>, <span class="hljs-built_in">r9</span>, <span class="hljs-built_in">r11</span>, lr} @ restore spill regs <span class="hljs-preprocessor">.cfi</span>_adjust_cfa_offset -<span class="hljs-number">24</span> bx lr END art_quick_invoke_stub</code><ul class="pre-numbering"><li>1</li><li>2</li><li>3</li><li>4</li><li>5</li><li>6</li><li>7</li><li>8</li><li>9</li><li>10</li><li>11</li><li>12</li><li>13</li><li>14</li><li>15</li><li>16</li><li>17</li><li>18</li><li>19</li><li>20</li><li>21</li><li>22</li><li>23</li><li>24</li><li>25</li><li>26</li><li>27</li><li>28</li><li>29</li><li>30</li><li>31</li><li>32</li><li>33</li><li>34</li><li>35</li><li>36</li><li>37</li><li>38</li><li>39</li><li>40</li><li>41</li><li>42</li><li>43</li><li>44</li><li>45</li></ul>
  • “ldr ip, [r0, #METHOD_CODE_OFFSET]”其實就是把ArtMethod::entry_point_from_compiled_code_賦值給ip,然后通過blx直接調用。通過這段小小的匯編代碼,我們得出如下堆棧的布局:

    <code class="hljs oxygene has-numbering"> -(low)| caller(<span class="hljs-function"><span class="hljs-keyword">Method</span> *) | <- <span class="hljs-title">sp</span> | <span class="hljs-title">arg1</span> | <- <span class="hljs-title">r1</span>| <span class="hljs-title">arg2</span> | <- <span class="hljs-title">r2</span>| <span class="hljs-title">arg3</span> | <- <span class="hljs-title">r3</span>| ... | | <span class="hljs-title">argN</span> || <span class="hljs-title">callee</span><span class="hljs-params">(<span class="hljs-keyword">Method</span> *)</span> | <- <span class="hljs-title">r0</span>+<span class="hljs-params">(high)</span></span></code><ul class="pre-numbering"><li>1</li><li>2</li><li>3</li><li>4</li><li>5</li><li>6</li><li>7</li><li>8</li><li>9</li></ul>

    這種調用約定并不是平時我們所見的調用約定,主要體現在參數當超過4時,并不是從sp開始保存,而是從sp + 20這個位置開始存儲,所以這就是為什么在代碼里entry_point_from_compiled_code_的類型是void *的原因了,因為無法用代碼表示。

    理解好這個調用約定對我們方案的實現至關重要

    • ArtMethod執行流程

    上面詳細講述了類方法加載和鏈接的過程,但在實際執行的過程中,其實還不是直接調用ArtMethod的entry_point(解析執行和本地指令執行的入口),為了加快執行速度,ART為oat文件中的每個dex創建了一個DexCache(art/runtime/mirror/dex_cache.h)結構,這個結構會按dex的結構生成一系列的數組,這里我們只分析它里面的methods字段。 DexCache初始化的方法是Init,實現如下:

    <code class="hljs r has-numbering">void DexCache::Init(const DexFile* dex_file,String* location,ObjectArray<String>* strings,ObjectArray<Class>* resolved_types,ObjectArray<ArtMethod>* resolved_methods,ObjectArray<ArtField>* resolved_fields,ObjectArray<StaticStorageBase>* initialized_static_storage) {//<span class="hljs-keyword">...</span>//<span class="hljs-keyword">...</span>Runtime* runtime = Runtime::Current();<span class="hljs-keyword">if</span> (runtime->HasResolutionMethod()) {// Initialize the resolve methods array to contain trampolines <span class="hljs-keyword">for</span> resolution.ArtMethod* trampoline = runtime->GetResolutionMethod();size_t length = resolved_methods->GetLength();<span class="hljs-keyword">for</span> (size_t i = <span class="hljs-number">0</span>; i < length; i++) {resolved_methods->SetWithoutChecks(i, trampoline);}} }</code><ul class="pre-numbering"><li>1</li><li>2</li><li>3</li><li>4</li><li>5</li><li>6</li><li>7</li><li>8</li><li>9</li><li>10</li><li>11</li><li>12</li><li>13</li><li>14</li><li>15</li><li>16</li><li>17</li><li>18</li><li>19</li></ul>

    根據dex方法的個數,產生相應長度resolved_methods數組,然后每一個都用Runtime::GetResolutionMethod()返回的結果進行填充,這個方法是由Runtime::CreateResolutionMethod產生的,代碼如下:

    <code class="hljs lasso has-numbering">mirror<span class="hljs-tag">::ArtMethod</span><span class="hljs-subst">*</span> Runtime<span class="hljs-tag">::CreateResolutionMethod</span>() {mirror<span class="hljs-tag">::Class</span><span class="hljs-subst">*</span> method_class <span class="hljs-subst">=</span> mirror<span class="hljs-tag">::ArtMethod</span><span class="hljs-tag">::GetJavaLangReflectArtMethod</span>();<span class="hljs-keyword">Thread</span><span class="hljs-subst">*</span> <span class="hljs-built_in">self</span> <span class="hljs-subst">=</span> <span class="hljs-keyword">Thread</span><span class="hljs-tag">::Current</span>();SirtRef<span class="hljs-subst"><</span>mirror<span class="hljs-tag">::ArtMethod</span><span class="hljs-subst">></span>method(<span class="hljs-built_in">self</span>, down_cast<span class="hljs-subst"><</span>mirror<span class="hljs-tag">::ArtMethod</span><span class="hljs-subst">*></span>(method_class<span class="hljs-subst">-></span>AllocObject(<span class="hljs-built_in">self</span>)));method<span class="hljs-subst">-></span>SetDeclaringClass(method_class);<span class="hljs-comment">// TODO: use a special method for resolution method saves</span>method<span class="hljs-subst">-></span>SetDexMethodIndex(DexFile<span class="hljs-tag">::kDexNoIndex</span>);<span class="hljs-comment">// When compiling, the code pointer will get set later when the image is loaded.</span>Runtime<span class="hljs-subst">*</span> r <span class="hljs-subst">=</span> Runtime<span class="hljs-tag">::Current</span>();ClassLinker<span class="hljs-subst">*</span> cl <span class="hljs-subst">=</span> r<span class="hljs-subst">-></span>GetClassLinker();method<span class="hljs-subst">-></span>SetEntryPointFromCompiledCode(r<span class="hljs-subst">-></span>IsCompiler() <span class="hljs-subst">?</span> <span class="hljs-built_in">NULL</span> : GetResolutionTrampoline(cl));<span class="hljs-keyword">return</span> method<span class="hljs-built_in">.</span>get(); }</code><ul class="pre-numbering"><li>1</li><li>2</li><li>3</li><li>4</li><li>5</li><li>6</li><li>7</li><li>8</li><li>9</li><li>10</li><li>11</li><li>12</li><li>13</li><li>14</li></ul>

    從method->SetDexMethodIndex(DexFile::kDexNoIndex)這句得知,所有的ResolutionMethod的methodIndexDexFile::kDexNoIndex。而ResolutionMethod的entrypoint就是我們上面入口分析中的第4種情況,GetResolutionTrampoline最終返回的入口為art_quick_resolution_trampoline(art/runtime/arch/arm/quick_entrypoints_arm.S)。我們看一下其實現代碼:

    <code class="hljs avrasm has-numbering"> <span class="hljs-preprocessor">.extern</span> artQuickResolutionTrampoline ENTRY art_quick_resolution_trampolineSETUP_REF_AND_ARGS_CALLEE_SAVE_FRAME<span class="hljs-keyword">mov</span> <span class="hljs-built_in">r2</span>, <span class="hljs-built_in">r9</span> @ pass Thread::Current<span class="hljs-keyword">mov</span> <span class="hljs-built_in">r3</span>, sp @ pass SPblx artQuickResolutionTrampoline @ (Method* called, receiver, Thread*, SP)cbz <span class="hljs-built_in">r0</span>, <span class="hljs-number">1</span>f @ is code pointer null? goto exception<span class="hljs-keyword">mov</span> <span class="hljs-built_in">r12</span>, <span class="hljs-built_in">r0</span>ldr <span class="hljs-built_in">r0</span>, [sp, <span class="hljs-preprocessor">#0] @ load resolved method in r0</span>ldr <span class="hljs-built_in">r1</span>, [sp, <span class="hljs-preprocessor">#8] @ restore non-callee save r1</span>ldrd <span class="hljs-built_in">r2</span>, [sp, <span class="hljs-preprocessor">#12] @ restore non-callee saves r2-r3</span>ldr lr, [sp, <span class="hljs-preprocessor">#44] @ restore lr</span><span class="hljs-keyword">add</span> sp, <span class="hljs-preprocessor">#48 @ rewind sp</span><span class="hljs-preprocessor">.cfi</span>_adjust_cfa_offset -<span class="hljs-number">48</span>bx <span class="hljs-built_in">r12</span> @ tail-<span class="hljs-keyword">call</span> into actual code <span class="hljs-number">1</span>:RESTORE_REF_AND_ARGS_CALLEE_SAVE_FRAMEDELIVER_PENDING_EXCEPTION END art_quick_resolution_trampoline </code><ul class="pre-numbering"><li>1</li><li>2</li><li>3</li><li>4</li><li>5</li><li>6</li><li>7</li><li>8</li><li>9</li><li>10</li><li>11</li><li>12</li><li>13</li><li>14</li><li>15</li><li>16</li><li>17</li><li>18</li><li>19</li><li>20</li></ul>

    調整好寄存器后,直接跳轉至artQuickResolutionTrampoline(art/runtime/entrypoints/quick/quick_trampoline_entrypoints.cc),接下來我們分析這個方法的實現(大家不要暈了。。。,我會把無關緊要的代碼去掉):

    <code class="hljs r has-numbering">// Lazily resolve a method <span class="hljs-keyword">for</span> quick. Called by stub code. extern <span class="hljs-string">"C"</span> const void* artQuickResolutionTrampoline(mirror::ArtMethod* called,mirror::Object* receiver,Thread* thread, mirror::ArtMethod** sp)SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) {FinishCalleeSaveFrameSetup(thread, sp, Runtime::kRefsAndArgs);// Start new JNI local reference stateJNIEnvExt* env = thread->GetJniEnv();ScopedObjectAccessUnchecked soa(env);ScopedJniEnvLocalRefState env_state(env);const char* old_cause = thread->StartAssertNoThreadSuspension(<span class="hljs-string">"Quick method resolution set up"</span>);// Compute details about the called method (avoid GCs)ClassLinker* linker = Runtime::Current()->GetClassLinker();mirror::ArtMethod* caller = QuickArgumentVisitor::GetCallingMethod(sp);InvokeType invoke_type;const DexFile* dex_file;uint32_t dex_method_idx;<span class="hljs-keyword">if</span> (called->IsRuntimeMethod()) {//<span class="hljs-keyword">...</span>//<span class="hljs-keyword">...</span>} <span class="hljs-keyword">else</span> {invoke_type = kStatic;dex_file = &MethodHelper(called).GetDexFile();dex_method_idx = called->GetDexMethodIndex();}//<span class="hljs-keyword">...</span>// Resolve method filling <span class="hljs-keyword">in</span> dex cache.<span class="hljs-keyword">if</span> (called->IsRuntimeMethod()) {called = linker->ResolveMethod(dex_method_idx, caller, invoke_type);}const void* code = <span class="hljs-literal">NULL</span>;<span class="hljs-keyword">if</span> (LIKELY(!thread->IsExceptionPending())) {//<span class="hljs-keyword">...</span>linker->EnsureInitialized(called_class, true, true);//<span class="hljs-keyword">...</span>}// <span class="hljs-keyword">...</span><span class="hljs-keyword">return</span> code; } </code><ul class="pre-numbering"><li>1</li><li>2</li><li>3</li><li>4</li><li>5</li><li>6</li><li>7</li><li>8</li><li>9</li><li>10</li><li>11</li><li>12</li><li>13</li><li>14</li><li>15</li><li>16</li><li>17</li><li>18</li><li>19</li><li>20</li><li>21</li><li>22</li><li>23</li><li>24</li><li>25</li><li>26</li><li>27</li><li>28</li><li>29</li><li>30</li><li>31</li><li>32</li><li>33</li><li>34</li><li>35</li><li>36</li><li>37</li><li>38</li><li>39</li><li>40</li><li>41</li><li>42</li><li>43</li><li>44</li><li>45</li><li>46</li></ul><code class="hljs objectivec has-numbering"><span class="hljs-keyword">inline</span> <span class="hljs-keyword">bool</span> ArtMethod::IsRuntimeMethod() <span class="hljs-keyword">const</span> {<span class="hljs-keyword">return</span> GetDexMethodIndex() == DexFile::kDexNoIndex; }</code><ul class="pre-numbering"><li>1</li><li>2</li><li>3</li></ul>

    called->IsRuntimeMethod()用于判斷當前方法是否為ResolutionMethod。如果是,那么就走ClassLinker::ResolveMethod流程去獲取真正的方法,見代碼:

    <code class="hljs lasso has-numbering">mirror<span class="hljs-tag">::ArtMethod</span><span class="hljs-subst">*</span> ClassLinker<span class="hljs-tag">::ResolveMethod</span>(const DexFile<span class="hljs-subst">&</span> dex_file,uint32_t method_idx,mirror<span class="hljs-tag">::DexCache</span><span class="hljs-subst">*</span> dex_cache,mirror<span class="hljs-tag">::ClassLoader</span><span class="hljs-subst">*</span> class_loader,const mirror<span class="hljs-tag">::ArtMethod</span><span class="hljs-subst">*</span> <span class="hljs-keyword">referrer</span>,InvokeType <span class="hljs-keyword">type</span>) {DCHECK(dex_cache <span class="hljs-subst">!=</span> <span class="hljs-built_in">NULL</span>);<span class="hljs-comment">// Check for hit in the dex cache.</span>mirror<span class="hljs-tag">::ArtMethod</span><span class="hljs-subst">*</span> resolved <span class="hljs-subst">=</span> dex_cache<span class="hljs-subst">-></span>GetResolvedMethod(method_idx);<span class="hljs-keyword">if</span> (resolved <span class="hljs-subst">!=</span> <span class="hljs-built_in">NULL</span>) {<span class="hljs-keyword">return</span> resolved;}<span class="hljs-comment">// Fail, get the declaring class.</span>const DexFile<span class="hljs-tag">::MethodId</span><span class="hljs-subst">&</span> method_id <span class="hljs-subst">=</span> dex_file<span class="hljs-built_in">.</span>GetMethodId(method_idx);mirror<span class="hljs-tag">::Class</span><span class="hljs-subst">*</span> klass <span class="hljs-subst">=</span> ResolveType(dex_file, method_id<span class="hljs-built_in">.</span>class_idx_, dex_cache, class_loader);<span class="hljs-keyword">if</span> (klass <span class="hljs-subst">==</span> <span class="hljs-built_in">NULL</span>) {DCHECK(<span class="hljs-keyword">Thread</span><span class="hljs-tag">::Current</span>()<span class="hljs-subst">-></span>IsExceptionPending());<span class="hljs-keyword">return</span> <span class="hljs-built_in">NULL</span>;}<span class="hljs-comment">// Scan using method_idx, this saves string compares but will only hit for matching dex</span><span class="hljs-comment">// caches/files.</span>switch (<span class="hljs-keyword">type</span>) {<span class="hljs-keyword">case</span> kDirect: <span class="hljs-comment">// Fall-through.</span><span class="hljs-keyword">case</span> kStatic:resolved <span class="hljs-subst">=</span> klass<span class="hljs-subst">-></span>FindDirectMethod(dex_cache, method_idx);break;<span class="hljs-keyword">case</span> kInterface:resolved <span class="hljs-subst">=</span> klass<span class="hljs-subst">-></span>FindInterfaceMethod(dex_cache, method_idx);DCHECK(resolved <span class="hljs-subst">==</span> <span class="hljs-built_in">NULL</span> <span class="hljs-subst">||</span> resolved<span class="hljs-subst">-></span>GetDeclaringClass()<span class="hljs-subst">-></span>IsInterface());break;<span class="hljs-keyword">case</span> kSuper: <span class="hljs-comment">// Fall-through.</span><span class="hljs-keyword">case</span> kVirtual:resolved <span class="hljs-subst">=</span> klass<span class="hljs-subst">-></span>FindVirtualMethod(dex_cache, method_idx);break;default:<span class="hljs-keyword">LOG</span>(FATAL) <span class="hljs-subst"><<</span> <span class="hljs-string">"Unreachable - invocation type: "</span> <span class="hljs-subst"><<</span> <span class="hljs-keyword">type</span>;}<span class="hljs-keyword">if</span> (resolved <span class="hljs-subst">==</span> <span class="hljs-built_in">NULL</span>) {<span class="hljs-comment">// Search by name, which works across dex files.</span>const char<span class="hljs-subst">*</span> name <span class="hljs-subst">=</span> dex_file<span class="hljs-built_in">.</span>StringDataByIdx(method_id<span class="hljs-built_in">.</span>name_idx_);std<span class="hljs-tag">::string</span> signature(dex_file<span class="hljs-built_in">.</span>CreateMethodSignature(method_id<span class="hljs-built_in">.</span>proto_idx_, <span class="hljs-built_in">NULL</span>));switch (<span class="hljs-keyword">type</span>) {<span class="hljs-keyword">case</span> kDirect: <span class="hljs-comment">// Fall-through.</span><span class="hljs-keyword">case</span> kStatic:resolved <span class="hljs-subst">=</span> klass<span class="hljs-subst">-></span>FindDirectMethod(name, signature);break;<span class="hljs-keyword">case</span> kInterface:resolved <span class="hljs-subst">=</span> klass<span class="hljs-subst">-></span>FindInterfaceMethod(name, signature);DCHECK(resolved <span class="hljs-subst">==</span> <span class="hljs-built_in">NULL</span> <span class="hljs-subst">||</span> resolved<span class="hljs-subst">-></span>GetDeclaringClass()<span class="hljs-subst">-></span>IsInterface());break;<span class="hljs-keyword">case</span> kSuper: <span class="hljs-comment">// Fall-through.</span><span class="hljs-keyword">case</span> kVirtual:resolved <span class="hljs-subst">=</span> klass<span class="hljs-subst">-></span>FindVirtualMethod(name, signature);break;}}<span class="hljs-keyword">if</span> (resolved <span class="hljs-subst">!=</span> <span class="hljs-built_in">NULL</span>) {<span class="hljs-comment">// Be a good citizen and update the dex cache to speed subsequent calls.</span>dex_cache<span class="hljs-subst">-></span>SetResolvedMethod(method_idx, resolved);<span class="hljs-keyword">return</span> resolved;} <span class="hljs-keyword">else</span> {<span class="hljs-comment">// ...</span>} } </code><ul class="pre-numbering"><li>1</li><li>2</li><li>3</li><li>4</li><li>5</li><li>6</li><li>7</li><li>8</li><li>9</li><li>10</li><li>11</li><li>12</li><li>13</li><li>14</li><li>15</li><li>16</li><li>17</li><li>18</li><li>19</li><li>20</li><li>21</li><li>22</li><li>23</li><li>24</li><li>25</li><li>26</li><li>27</li><li>28</li><li>29</li><li>30</li><li>31</li><li>32</li><li>33</li><li>34</li><li>35</li><li>36</li><li>37</li><li>38</li><li>39</li><li>40</li><li>41</li><li>42</li><li>43</li><li>44</li><li>45</li><li>46</li><li>47</li><li>48</li><li>49</li><li>50</li><li>51</li><li>52</li><li>53</li><li>54</li><li>55</li><li>56</li><li>57</li><li>58</li><li>59</li><li>60</li><li>61</li><li>62</li><li>63</li><li>64</li><li>65</li><li>66</li><li>67</li><li>68</li><li>69</li><li>70</li></ul>

    其實這里發生了“連鎖反應”,ClassLinker::ResolveType走的流程,跟ResolveMethod是非常類似的,有興趣的朋友可以跟一下。
    找到解析后的klass,再經過一輪瘋狂的搜索,把找到的resolved通過DexCache::SetResolvedMethod覆蓋掉之前的“替身”。當再下次再通過ResolveMethod解析方法時,就可以直接把該方法返回,不需要再解析了。

    我們回過頭來再重新“復現”一下這個過程,當我們首次調用某個類方法,其過程如下所示:

  • 調用ResolutionMethod的entrypoint,進入art_quick_resolution_trampoline;
  • art_quick_resolution_trampoline跳轉到artQuickResolutionTrampoline;
  • artQuickResolutionTrampoline調用ClassLinker::ResolveMethod解析類方法;
  • ClassLinker::ResolveMethod調用ClassLinkder::ResolveType解析類,再從解析好的類尋找真正的方法;
  • 調用DexCache::SetResolvedMethod,用真正的方法覆蓋掉“替身”方法;
  • 調用真正方法的entrypoint代碼;
  • 也許你會問,為什么要把過程搞得這么繞? 一切都是為了延遲加載,提高啟動速度,這個過程跟ELF Linker的PLT/GOT符號重定向的過程是何其相似啊,所以技術都是想通的,一通百明。

    0x3 Hook ArtMethod

    通過上述ArtMethod加載和執行兩個流程的分析,對于如何Hook ArtMethod,我想到了兩個方案,分別

  • 修改DexCach里的methods,把里面的entrypoint修改為自己的,做一個中轉處理;
  • 直接修改加載后的ArtMethod的entrypoint,同樣做一個中轉處理;
  • 上面兩個方法都是可行的,但由于我希望整個項目可以在NDK環境(而不是在源碼下)下編譯,因為就采用了方案2,因為通過JNI的接口就可以直接獲取解析之后的ArtMethod,可以減少很多文件依賴。

    回到前面的調用約定,每個ArtMethod都有兩個約定,按道理我們應該準備兩個中轉函數的,但這里我們不考慮強制解析模式執行,所以只要處理好entry_point_from_compiled_code的中轉即可。

    首先,我們找到對應的方法,先保存其entrypoint,然后再把我們的中轉函數art_quick_dispatcher覆蓋,代碼如下所示:

    <code class="hljs cpp has-numbering"><span class="hljs-keyword">extern</span> <span class="hljs-keyword">int</span> __attribute__ ((visibility (<span class="hljs-string">"hidden"</span>))) art_java_method_hook(JNIEnv* env, HookInfo *info) {<span class="hljs-keyword">const</span> <span class="hljs-keyword">char</span>* classDesc = info->classDesc;<span class="hljs-keyword">const</span> <span class="hljs-keyword">char</span>* methodName = info->methodName;<span class="hljs-keyword">const</span> <span class="hljs-keyword">char</span>* methodSig = info->methodSig;<span class="hljs-keyword">const</span> <span class="hljs-keyword">bool</span> isStaticMethod = info->isStaticMethod;<span class="hljs-comment">// TODO we can find class by special classloader what do just like dvm</span>jclass claxx = env->FindClass(classDesc);<span class="hljs-keyword">if</span>(claxx == NULL){LOGE(<span class="hljs-string">"[-] %s class not found"</span>, classDesc);<span class="hljs-keyword">return</span> -<span class="hljs-number">1</span>;}jmethodID methid = isStaticMethod ?env->GetStaticMethodID(claxx, methodName, methodSig) :env->GetMethodID(claxx, methodName, methodSig);<span class="hljs-keyword">if</span>(methid == NULL){LOGE(<span class="hljs-string">"[-] %s->%s method not found"</span>, classDesc, methodName);<span class="hljs-keyword">return</span> -<span class="hljs-number">1</span>;}ArtMethod *artmeth = <span class="hljs-keyword">reinterpret_cast</span><ArtMethod *>(methid);<span class="hljs-keyword">if</span>(art_quick_dispatcher != artmeth->GetEntryPointFromCompiledCode()){uint64_t (*entrypoint)(ArtMethod* method, Object *thiz, u4 *arg1, u4 *arg2);entrypoint = (uint64_t (*)(ArtMethod*, Object *, u4 *, u4 *))artmeth->GetEntryPointFromCompiledCode();info->entrypoint = (<span class="hljs-keyword">const</span> <span class="hljs-keyword">void</span> *)entrypoint;info->nativecode = artmeth->GetNativeMethod();artmeth->SetEntryPointFromCompiledCode((<span class="hljs-keyword">const</span> <span class="hljs-keyword">void</span> *)art_quick_dispatcher);<span class="hljs-comment">// save info to nativecode :)</span>artmeth->SetNativeMethod((<span class="hljs-keyword">const</span> <span class="hljs-keyword">void</span> *)info);LOGI(<span class="hljs-string">"[+] %s->%s was hooked\n"</span>, classDesc, methodName);}<span class="hljs-keyword">else</span>{LOGW(<span class="hljs-string">"[*] %s->%s method had been hooked"</span>, classDesc, methodName);}<span class="hljs-keyword">return</span> <span class="hljs-number">0</span>; }</code><ul class="pre-numbering"><li>1</li><li>2</li><li>3</li><li>4</li><li>5</li><li>6</li><li>7</li><li>8</li><li>9</li><li>10</li><li>11</li><li>12</li><li>13</li><li>14</li><li>15</li><li>16</li><li>17</li><li>18</li><li>19</li><li>20</li><li>21</li><li>22</li><li>23</li><li>24</li><li>25</li><li>26</li><li>27</li><li>28</li><li>29</li><li>30</li><li>31</li><li>32</li><li>33</li><li>34</li><li>35</li><li>36</li><li>37</li><li>38</li><li>39</li><li>40</li><li>41</li><li>42</li><li>43</li></ul>

    我們關鍵的信息通過ArtMethod::SetNativeMethod保存起來了。

    考慮到ART特殊的調用約定,art_quick_dispatcher只能用匯編實現了,把寄存器適當的調整一下,再跳轉到另一個函數artQuickToDispatcher,這樣就可以很方便用c/c++訪問參數了。

    先看一下art_quick_dispatcher函數的實現如下:

    <code class="hljs avrasm has-numbering"><span class="hljs-comment">/** Art Quick Dispatcher.* On entry:* r0 = method pointer* r1 = arg1* r2 = arg2* r3 = arg3* [sp] = method pointer* [sp + 4] = addr of thiz* [sp + 8] = addr of arg1* [sp + 12] = addr of arg2* [sp + 16] = addr of arg3* and so on*/</span><span class="hljs-preprocessor">.extern</span> artQuickToDispatcher ENTRY art_quick_dispatcher<span class="hljs-keyword">push</span> {<span class="hljs-built_in">r4</span>, <span class="hljs-built_in">r5</span>, lr} @ sp - <span class="hljs-number">12</span><span class="hljs-keyword">mov</span> <span class="hljs-built_in">r0</span>, <span class="hljs-built_in">r0</span> @ pass <span class="hljs-built_in">r0</span> to methodstr <span class="hljs-built_in">r1</span>, [sp, <span class="hljs-preprocessor">#(12 + 4)]</span>str <span class="hljs-built_in">r2</span>, [sp, <span class="hljs-preprocessor">#(12 + 8)]</span>str <span class="hljs-built_in">r3</span>, [sp, <span class="hljs-preprocessor">#(12 + 12)]</span><span class="hljs-keyword">mov</span> <span class="hljs-built_in">r1</span>, <span class="hljs-built_in">r9</span> @ pass <span class="hljs-built_in">r1</span> to thread<span class="hljs-keyword">add</span> <span class="hljs-built_in">r2</span>, sp, <span class="hljs-preprocessor">#(12 + 4) @ pass r2 to args array</span><span class="hljs-keyword">add</span> <span class="hljs-built_in">r3</span>, sp, <span class="hljs-preprocessor">#12 @ pass r3 to old SP</span>blx artQuickToDispatcher @ (Method* method, Thread*, u4 **, u4 **)<span class="hljs-keyword">pop</span> {<span class="hljs-built_in">r4</span>, <span class="hljs-built_in">r5</span>, pc} @ return on success, <span class="hljs-built_in">r0</span> <span class="hljs-keyword">and</span> <span class="hljs-built_in">r1</span> hold the result END art_quick_dispatcher</code><ul class="pre-numbering"><li>1</li><li>2</li><li>3</li><li>4</li><li>5</li><li>6</li><li>7</li><li>8</li><li>9</li><li>10</li><li>11</li><li>12</li><li>13</li><li>14</li><li>15</li><li>16</li><li>17</li><li>18</li><li>19</li><li>20</li><li>21</li><li>22</li><li>23</li><li>24</li><li>25</li><li>26</li><li>27</li></ul>

    我把r2指向參數數組,這樣就我們就可以非常方便的訪問所有參數了。另外,我用r3保存了舊的sp地址,這樣是為后面調用原來的entrypoint做準備的。我們先看看artQuickToDispatcher的實現:

    <code class="hljs coffeescript has-numbering">extern <span class="hljs-string">"C"</span> uint64_t artQuickToDispatcher(ArtMethod* method, Thread *self, u4 **args, u4 **old_sp){HookInfo *info = (HookInfo *)method->GetNativeMethod();LOGI(<span class="hljs-string">"[+] entry ArtHandler %s->%s"</span>, info->classDesc, info->methodName);<span class="hljs-regexp">//</span> If it <span class="hljs-keyword">not</span> <span class="hljs-keyword">is</span> static method, <span class="hljs-keyword">then</span> args[<span class="hljs-number">0</span>] was pointing to <span class="hljs-keyword">this</span><span class="hljs-keyword">if</span>(!info->isStaticMethod){Object *thiz = reinterpret_cast<Object *>(args[<span class="hljs-number">0</span>]);<span class="hljs-keyword">if</span>(thiz != NULL){char *bytes = get_chars_from_utf16<span class="hljs-function"><span class="hljs-params">(thiz->GetClass()->GetName())</span>;<span class="hljs-title">LOGI</span><span class="hljs-params">(<span class="hljs-string">"[+] thiz class is %s"</span>, bytes)</span>;<span class="hljs-title">delete</span> <span class="hljs-title">bytes</span>;}}<span class="hljs-title">const</span> <span class="hljs-title">void</span> *<span class="hljs-title">entrypoint</span> = <span class="hljs-title">info</span>-></span>entrypoint;method->SetNativeMethod(info->nativecode); <span class="hljs-regexp">//</span>restore nativecode <span class="hljs-keyword">for</span> JNI methoduint64_t res = art_quick_call_entrypoint(method, self, args, old_sp, entrypoint);JValue* result = (JValue* )&res;<span class="hljs-keyword">if</span>(result != NULL){Object *obj = result->l;char *raw_class_name = get_chars_from_utf16<span class="hljs-function"><span class="hljs-params">(obj->GetClass()->GetName())</span>;<span class="hljs-title">if</span><span class="hljs-params">(strcmp(raw_class_name, <span class="hljs-string">"java.lang.String"</span>) == <span class="hljs-number">0</span>)</span>{<span class="hljs-title">char</span> *<span class="hljs-title">raw_string_value</span> = <span class="hljs-title">get_chars_from_utf16</span><span class="hljs-params">((String *)obj)</span>;<span class="hljs-title">LOGI</span><span class="hljs-params">(<span class="hljs-string">"result-class %s, result-value \"%s\""</span>, raw_class_name, raw_string_value)</span>;<span class="hljs-title">free</span><span class="hljs-params">(raw_string_value)</span>;}<span class="hljs-title">else</span>{<span class="hljs-title">LOGI</span><span class="hljs-params">(<span class="hljs-string">"result-class %s"</span>, raw_class_name)</span>;}<span class="hljs-title">free</span><span class="hljs-params">(raw_class_name)</span>;}// <span class="hljs-title">entrypoid</span> <span class="hljs-title">may</span> <span class="hljs-title">be</span> <span class="hljs-title">replaced</span> <span class="hljs-title">by</span> <span class="hljs-title">trampoline</span>, <span class="hljs-title">only</span> <span class="hljs-title">once</span>. // <span class="hljs-title">if</span><span class="hljs-params">(method->IsStatic() && !method->IsConstructor())</span>{<span class="hljs-title">entrypoint</span> = <span class="hljs-title">method</span>-></span>GetEntryPointFromCompiledCode();<span class="hljs-keyword">if</span>(entrypoint != (<span class="hljs-reserved">const</span> <span class="hljs-reserved">void</span> *)art_quick_dispatcher){LOGW(<span class="hljs-string">"[*] entrypoint was replaced. %s->%s"</span>, info->classDesc, info->methodName);method->SetEntryPointFromCompiledCode((<span class="hljs-reserved">const</span> <span class="hljs-reserved">void</span> *)art_quick_dispatcher);info->entrypoint = entrypoint;info->nativecode = method->GetNativeMethod();}method->SetNativeMethod((<span class="hljs-reserved">const</span> <span class="hljs-reserved">void</span> *)info);<span class="hljs-regexp">//</span> }<span class="hljs-keyword">return</span> res; } </code><ul class="pre-numbering"><li>1</li><li>2</li><li>3</li><li>4</li><li>5</li><li>6</li><li>7</li><li>8</li><li>9</li><li>10</li><li>11</li><li>12</li><li>13</li><li>14</li><li>15</li><li>16</li><li>17</li><li>18</li><li>19</li><li>20</li><li>21</li><li>22</li><li>23</li><li>24</li><li>25</li><li>26</li><li>27</li><li>28</li><li>29</li><li>30</li><li>31</li><li>32</li><li>33</li><li>34</li><li>35</li><li>36</li><li>37</li><li>38</li><li>39</li><li>40</li><li>41</li><li>42</li><li>43</li><li>44</li><li>45</li><li>46</li><li>47</li><li>48</li><li>49</li><li>50</li><li>51</li><li>52</li><li>53</li></ul>

    這里參數解析就不詳細說了,接下來是最棘手的問題——如何重新調回原來的entrypoint。

    這里的關鍵點是要還原之前的堆棧布局,art_quick_call_entrypoint就是負責完成這個工作的,其實現如下所示:

    <code class="hljs avrasm has-numbering"><span class="hljs-comment">/*** Art Quick Call Entrypoint* On entry:* r0 = method pointer* r1 = thread pointer* r2 = args arrays pointer* r3 = old_sp* [sp] = entrypoint*/</span> ENTRY art_quick_call_entrypoint<span class="hljs-keyword">push</span> {<span class="hljs-built_in">r4</span>, <span class="hljs-built_in">r5</span>, lr} @ sp - <span class="hljs-number">12</span><span class="hljs-keyword">sub</span> sp, <span class="hljs-preprocessor">#(40 + 20) @ sp - 40 - 20</span>str <span class="hljs-built_in">r0</span>, [sp, <span class="hljs-preprocessor">#(40 + 0)] @ var_40_0 = method_pointer</span>str <span class="hljs-built_in">r1</span>, [sp, <span class="hljs-preprocessor">#(40 + 4)] @ var_40_4 = thread_pointer</span>str <span class="hljs-built_in">r2</span>, [sp, <span class="hljs-preprocessor">#(40 + 8)] @ var_40_8 = args_array</span>str <span class="hljs-built_in">r3</span>, [sp, <span class="hljs-preprocessor">#(40 + 12)] @ var_40_12 = old_sp</span><span class="hljs-keyword">mov</span> <span class="hljs-built_in">r0</span>, sp<span class="hljs-keyword">mov</span> <span class="hljs-built_in">r1</span>, <span class="hljs-built_in">r3</span>ldr <span class="hljs-built_in">r2</span>, =<span class="hljs-number">40</span>blx memcpy @ memcpy(dest, src, size_of_byte)ldr <span class="hljs-built_in">r0</span>, [sp, <span class="hljs-preprocessor">#(40 + 0)] @ restore method to r0</span>ldr <span class="hljs-built_in">r1</span>, [sp, <span class="hljs-preprocessor">#(40 + 4)]</span><span class="hljs-keyword">mov</span> <span class="hljs-built_in">r9</span>, <span class="hljs-built_in">r1</span> @ restore thread to <span class="hljs-built_in">r9</span>ldr <span class="hljs-built_in">r5</span>, [sp, <span class="hljs-preprocessor">#(40 + 8)] @ pass r5 to args_array</span>ldr <span class="hljs-built_in">r1</span>, [<span class="hljs-built_in">r5</span>] @ restore arg1ldr <span class="hljs-built_in">r2</span>, [<span class="hljs-built_in">r5</span>, <span class="hljs-preprocessor">#4] @ restore arg2</span>ldr <span class="hljs-built_in">r3</span>, [<span class="hljs-built_in">r5</span>, <span class="hljs-preprocessor">#8] @ restore arg3</span>ldr <span class="hljs-built_in">r5</span>, [sp, <span class="hljs-preprocessor">#(40 + 20 + 12)] @ pass ip to entrypoint</span>blx <span class="hljs-built_in">r5</span><span class="hljs-keyword">add</span> sp, <span class="hljs-preprocessor">#(40 + 20)</span><span class="hljs-keyword">pop</span> {<span class="hljs-built_in">r4</span>, <span class="hljs-built_in">r5</span>, pc} @ return on success, <span class="hljs-built_in">r0</span> <span class="hljs-keyword">and</span> <span class="hljs-built_in">r1</span> hold the result END art_quick_call_entrypoint</code><ul class="pre-numbering"><li>1</li><li>2</li><li>3</li><li>4</li><li>5</li><li>6</li><li>7</li><li>8</li><li>9</li><li>10</li><li>11</li><li>12</li><li>13</li><li>14</li><li>15</li><li>16</li><li>17</li><li>18</li><li>19</li><li>20</li><li>21</li><li>22</li><li>23</li><li>24</li><li>25</li><li>26</li><li>27</li><li>28</li><li>29</li><li>30</li><li>31</li><li>32</li><li>33</li></ul>

    這里我偷懶了,直接申請了10個參數的空間,再使用之前傳進入來的old_sp進行恢復,使用memcpy直接復制40字節。之后就是還原r0, r1, r2, r3, r9的值了。調用entrypoint完后,結果保存在r0和r1,再返回給artQuickToDispatcher。

    至此,整個ART Hook就分析完畢了。

    0x4 4.4與5.X上實現的區別

    我的整個方案都是在4.4上測試的,主要是因為我只有4.4的源碼,而且硬盤空間不足,實在裝不下5.x的源碼了。但整個思路,是完全可以套用用5.X上。另外,5.X的實現代碼比4.4上復雜了很多,否能像我這樣在NDK下編譯完成就不知道了。

    正常的4.4模擬器是以dalvik啟動的,要到設置里改為art,這里會要求進行重啟,但一般無效,我們手動關閉再重新打開就OK了,但需要等上一段時間才可以。

    0x5 結束

    雖然這篇文章只是介紹了Art Hook的技術方案,但其中的技術原理,對于如何在ART上進行代碼加固、動態代碼還原等等也是很有啟發性。

    老樣子,整個項目的代碼,我已經提交到https://github.com/boyliang/AllHookInOne,大家遇到什么問題,歡迎提問,有問題記得反饋。

    對了,請用https://github.com/boyliang/ndk-patch給你的NDK打一下patch。


    原文地址: http://blog.csdn.net/l173864930/article/details/45035521

    總結

    以上是生活随笔為你收集整理的Android Art Hook 技术方案的全部內容,希望文章能夠幫你解決所遇到的問題。

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

    欧美精品国产综合久久 | 四季av综合网站 | 国产精品久久久久久999 | 日批网站免费观看 | 黄色免费大片 | 国产在线观看不卡 | 国产精品电影一区 | 国产成人精品一区二区三区 | 久久久久免费看 | 日本在线视频网址 | 免费美女久久99 | 久久免费视频99 | 国内精品免费久久影院 | 亚洲欧美日本一区二区三区 | 欧洲精品久久久久毛片完整版 | 欧美日韩国产综合一区二区 | 高清久久久久久 | 国产精品第 | 97香蕉超级碰碰久久免费软件 | 精品99在线 | 天天色天天干天天 | 六月婷婷网 | 91精品国产成 | 日韩精品一区二区三区三炮视频 | 又黄又爽又湿又无遮挡的在线视频 | 又湿又紧又大又爽a视频国产 | 日韩a在线 | 国产精品一区二区三区99 | 日韩在线播放欧美字幕 | 天天干国产 | 在线播放亚洲激情 | 国产一级黄色av | 91久色蝌蚪 | 亚洲精品一区二区三区在线观看 | 中文字幕在线看视频国产 | 97综合视频 | 久草在线费播放视频 | 久久久精品成人 | 国产精品免费久久久久影院仙踪林 | 日韩极品在线 | 久久免费黄色 | 久久久久国产精品www | 亚洲一区不卡视频 | 午夜影视av | 久青草影院 | 日韩精品一区二区免费 | 麻豆视频www | 色中射 | 91成人免费看| 成人精品视频久久久久 | 中文字幕av全部资源www中文字幕在线观看 | 久久伊99综合婷婷久久伊 | 精品视频999| 五月天久久激情 | 久久夜夜爽 | 激情小说网站亚洲综合网 | 爱av在线网 | 日p视频| 天天射天天射天天 | 国产一区二区中文字幕 | 国产成人福利 | 欧美一区二区三区在线看 | 国产精品影音先锋 | 亚洲精品视频免费 | 色开心| 91黄色成人| 高清av免费一区中文字幕 | 日韩字幕 | 99精品国自产在线 | 欧美91成人网 | 精品久久久免费视频 | 日韩av一卡二卡三卡 | 欧美精品v国产精品v日韩精品 | 国产九九热视频 | 成人在线免费av | 不卡av在线| 精品在线视频一区 | 免费观看日韩 | 韩国一区二区三区视频 | 天海翼一区二区三区免费 | 欧美一级电影在线观看 | 国产精品久久亚洲 | 亚洲精品一区二区18漫画 | 欧美成年性 | 91在线视频免费观看 | 国产精品久久久久久久久久久久午夜 | 91免费日韩| 狠狠色狠狠色综合日日小说 | 99国内精品久久久久久久 | 国产精品h在线观看 | 国产视频手机在线 | 国产成人性色生活片 | 大胆欧美gogo免费视频一二区 | 日本性生活一级片 | 欧美成人播放 | 国产精品短视频 | 亚洲成a人片在线观看中文 中文字幕在线视频第一页 狠狠色丁香婷婷综合 | 天天色天天草天天射 | 成人动漫一区二区 | 欧美日韩中文在线观看 | 天天草天天摸 | av电影中文字幕在线观看 | 成人av一区二区三区 | 久久免费高清 | 日韩在线观看视频在线 | 在线观看蜜桃视频 | 亚洲 欧美 另类人妖 | 四虎欧美 | 国产精品欧美久久久久三级 | 亚洲精品乱码白浆高清久久久久久 | 色婷婷成人 | av成人黄色 | 日韩成片| 九九色在线观看 | 日韩精品中文字幕在线观看 | 国产黄色理论片 | 99人成在线观看视频 | 亚洲国产人午在线一二区 | 黄色小网站在线 | 缴情综合网五月天 | 欧美日韩免费观看一区二区三区 | 黄色最新网址 | 黄色软件在线观看免费 | 久久国产精品电影 | 麻豆视频在线 | 97精品国产91久久久久久久 | 久久再线视频 | 精品久久久久一区二区国产 | 欧美日韩在线精品 | 亚洲伊人av | 免费看黄电影 | 97视频网址 | 久久久精品二区 | 欧美午夜久久久 | 日本在线视频一区二区三区 | 国产在线精品区 | 成年人在线免费看片 | 久久观看最新视频 | 午夜精品一区二区三区可下载 | av电影 一区二区 | 国产精品视频在线看 | 一区免费视频 | 亚洲国产成人在线观看 | 精品在线观看一区二区 | 亚洲精品高清视频 | 亚洲伊人网在线观看 | 久久精品香蕉 | wwwwwww色| 国产精品一区二区三区久久久 | 久久久久久高清 | 在线免费黄| 亚洲精品在线观看不卡 | 亚洲激情中文 | 欧美一级性生活视频 | 69av在线视频 | 亚洲精品网页 | 国产精品久久久久久久久费观看 | 九九国产精品视频 | 九九视频在线播放 | av三级在线免费观看 | 亚洲成人蜜桃 | 18网站在线观看 | 色婷婷亚洲 | 黄色一级免费电影 | 最近中文字幕大全 | 亚洲精品久久久蜜臀下载官网 | 麻豆首页 | 欧美另类xxxxx | av丝袜制服 | 美女激情影院 | 国产精品乱码久久 | 97国产超碰在线 | 在线观看91精品国产网站 | 久久天堂精品视频 | 一区视频在线 | 正在播放国产一区二区 | 久久美女高清视频 | 中文字幕在线观看一区 | 国产黄色大全 | 在线观看 亚洲 | 欧美性黄网官网 | 国产91精品一区二区 | 人人舔人人干 | 亚洲成人av在线电影 | 成人av网站在线播放 | 欧美久久久久久久久久久久 | 日韩在线高清 | 综合国产在线观看 | 麻豆一精品传二传媒短视频 | 亚洲欧美国内爽妇网 | 国产一级片在线播放 | 国产一级电影免费观看 | 成人国产网址 | 国产精品久久艹 | 国产九九九九九 | 在线天堂中文在线资源网 | 国产黄色特级片 | 欧美日韩国产网站 | 黄色软件大全网站 | 国产中文字幕一区 | 911香蕉视频 | 99久e精品热线免费 99国产精品久久久久久久久久 | 成人黄大片视频在线观看 | 青青河边草免费直播 | 国产高清日韩欧美 | 在线免费观看麻豆视频 | 中文不卡视频 | 四虎影院在线观看av | 亚洲夜夜综合 | 国产 日韩 欧美 中文 在线播放 | 91亚洲夫妻 | 91视频久久久| 国产精品黄色av | 日韩女同av| 特黄特黄的视频 | 国产在线精品播放 | 国产成人精品一区二区三区 | 亚洲国产欧美一区二区三区丁香婷 | 91视频啊啊啊 | 国产精品一区二区av麻豆 | 激情网色 | 99这里只有| 又黄又爽又刺激视频 | 久久国产精品免费看 | 国产99色| 国产亚洲欧美精品久久久久久 | 黄色在线免费观看网站 | 国产美女主播精品一区二区三区 | 亚洲精品www| 亚洲每日更新 | 97超碰福利久久精品 | 亚洲精品视频大全 | 深爱激情五月网 | 六月丁香综合网 | 中文字幕亚洲不卡 | 国产中文字幕网 | 成人黄色毛片 | 国产精品99久久久久 | 奇米先锋 | 91mv.cool在线观看 | 欧亚日韩精品一区二区在线 | 婷婷在线精品视频 | 射射射av| 国产破处精品 | 精品国产乱码久久久久久三级人 | 狠狠干狠狠久久 | 开心色插| 在线观看日韩视频 | 2019中文字幕网站 | 成人欧美一区二区三区黑人麻豆 | 成人欧美亚洲 | 国产一区二区高清视频 | 国产精品成人自产拍在线观看 | av成人免费观看 | 婷婷色中文字幕 | 99精品欧美一区二区三区黑人哦 | 综合久久精品 | 久久久亚洲电影 | 色综合天天在线 | 黄色的网站免费看 | 91xav| 91麻豆精品国产自产在线游戏 | 国产女做a爱免费视频 | 国产日韩精品一区二区在线观看播放 | 亚洲天天在线日亚洲洲精 | 日本中文乱码卡一卡二新区 | av电影中文 | 97精品国自产拍在线观看 | 99精品热视频只有精品10 | 一本一本久久aa综合精品 | 亚洲成人欧美 | 日韩久久电影 | 在线看小早川怜子av | av在线播放网址 | 一区二区三区精品久久久 | 在线观看91久久久久久 | aaa毛片视频| 久草影视在线观看 | 精品中文字幕在线 | 日韩高清观看 | 99热 精品在线 | 午夜在线日韩 | 日本久久免费电影 | 五月视频| 97在线影视| 国产在线日韩 | 97视频人人澡人人爽 | 免费一级片在线观看 | 综合激情婷婷 | 九九爱免费视频在线观看 | 久久久久久久久久久久影院 | 福利视频一二区 | 国产精品美女久久久免费 | 免费97视频| 国产亚洲情侣一区二区无 | 亚洲va欧美va国产va黑人 | 亚洲一区视频免费观看 | 中文字幕在线观看的网站 | 爱情影院aqdy鲁丝片二区 | 久久66热这里只有精品 | 免费av在线网站 | 99久久国产免费,99久久国产免费大片 | 99久久精品免费看国产一区二区三区 | 四虎影视成人精品 | 91av视频免费观看 | 日韩一级电影在线 | 国产精品久久久久999 | 婷婷色婷婷 | 永久免费精品视频网站 | 9999精品| 色的网站在线观看 | 免费福利视频导航 | 99热国内精品 | 亚洲永久字幕 | 欧美一级艳片视频免费观看 | 超碰在线观看99 | 中文字幕一区二区三区精华液 | 久久精品国产精品亚洲 | 欧美日韩国产精品一区二区 | 亚洲国产精久久久久久久 | 中文字幕在线一区二区三区 | 米奇影视7777 | 亚洲一区精品人人爽人人躁 | 奇米影视四色8888 | 十八岁以下禁止观看的1000个网站 | 91成人精品视频 | 操操综合 | 久久一线 | 国产色拍拍拍拍在线精品 | 色就是色综合 | 国产精品高潮久久av | 正在播放亚洲精品 | 伊人久久电影网 | 午夜精品一区二区三区可下载 | 国产亚洲一区二区三区 | 最近中文字幕大全中文字幕免费 | 日韩精品国产一区 | 99热超碰 | 国产高清免费在线播放 | 91av99| 91免费观看网站 | 午夜婷婷在线播放 | 久久永久免费视频 | 国产精品婷婷午夜在线观看 | 日韩啪视频 | 亚洲成人免费观看 | 久久久久久久久久久久99 | 麻花豆传媒mv在线观看网站 | 欧美色图亚洲图片 | 免费在线色 | 91亚洲影院 | 国产免费成人 | 日本三级久久久 | 97香蕉超级碰碰久久免费软件 | 国产免费又黄又爽 | 久久九九国产精品 | 91精品国产三级a在线观看 | 九九九热精品免费视频观看网站 | 久久视频免费观看 | 日韩在线观看一区二区 | 日本中文字幕视频 | 欧美性受极品xxxx喷水 | 在线观看国产成人av片 | 亚洲精品乱码久久久久久 | 国产精品岛国久久久久久久久红粉 | 一级欧美黄 | 中文字幕欧美日韩va免费视频 | 欧美成年人在线观看 | 综合国产在线 | 二区视频在线观看 | 97网站| 久草干| 精品久久久久一区二区国产 | 午夜美女福利直播 | 黄色国产在线观看 | 亚洲电影毛片 | av 在线观看 | 国产手机av在线 | 久久三级视频 | 91在线看免费 | 玖玖玖精品 | 久久深夜福利免费观看 | 免费看的国产视频网站 | 免费三级在线 | 99久久精品免费看国产一区二区三区 | 欧产日产国产69 | 激情在线网站 | 免费看片亚洲 | 久久久久综合精品福利啪啪 | 久久久亚洲电影 | 一级理论片在线观看 | 国内精品视频一区二区三区八戒 | 亚洲韩国一区二区三区 | 精品国产91亚洲一区二区三区www | 亚洲码国产日韩欧美高潮在线播放 | 91在线欧美| 国产精品毛片一区二区 | 国产精品毛片一区视频播 | 成人黄色中文字幕 | 国产一区二区在线免费播放 | 夜添久久精品亚洲国产精品 | 一本—道久久a久久精品蜜桃 | 成人av免费电影 | 国产r级在线观看 | 日韩一区二区三区高清免费看看 | 国产精品伦一区二区三区视频 | 91精品国产综合久久婷婷香蕉 | 国产精品久久综合 | 国产精品一级在线 | 黄毛片在线观看 | 国产一区二区视频在线播放 | 中文在线8资源库 | 亚洲清纯国产 | 四虎在线免费观看 | 国产美女被啪进深处喷白浆视频 | 亚洲精品女人久久久 | 九九日韩 | 国产a级片免费观看 | 天天干 天天摸 天天操 | 日韩激情片在线观看 | 亚洲性少妇性猛交wwww乱大交 | 国产精品久久久久久久久久免费 | 91免费日韩| 精品国产乱码久久久久久天美 | 精品国产一区二区三区久久 | 午夜久久精品 | 日韩高清不卡一区二区三区 | 少妇性色午夜淫片aaaze | 亚洲有 在线 | 极品嫩模被强到高潮呻吟91 | a在线观看免费视频 | 一区二区不卡视频在线观看 | 亚洲成av人片在线观看www | 日韩精品在线看 | 久久久久国产精品一区二区 | 福利视频一区二区 | 97av超碰 | 国产精品1000 | 在线观看的黄色 | 色婷婷久久 | 最近2019中文免费高清视频观看www99 | 日韩精品极品视频 | 欧美日韩在线视频观看 | 久久撸在线视频 | 免费热情视频 | 日韩极品视频在线观看 | 欧美激情精品久久久久久免费 | 国产精品免费一区二区 | 亚洲午夜精品一区 | 在线免费精品视频 | 色资源在线| 九九在线高清精品视频 | 成人动漫精品一区二区 | 久久黄色影视 | 久久综合久久久久88 | 亚洲精品理论片 | 日韩中文字幕免费电影 | 日韩久久网站 | 亚洲精品黄 | av中文在线播放 | 日本特黄特色aaa大片免费 | 91视频免费看网站 | 日韩欧美在线国产 | 国产黄色免费观看 | 一本到视频在线观看 | 久久免费99精品久久久久久 | 91精品国产入口 | 久久久久久黄 | 久久久久国产精品午夜一区 | 天天操夜夜操 | 日韩大片免费在线观看 | 美女网色| 国内99视频 | 久久久国产精品亚洲一区 | 99精品欧美一区二区三区黑人哦 | 精品在线观看一区二区 | 日韩xxxx视频 | 国产视频97 | 97视频资源 | 久久久久久久久久久久影院 | 天堂在线视频免费观看 | 欧美日韩免费在线视频 | 日日爱影视 | 国产99一区 | 人人澡超碰碰97碰碰碰软件 | 久久精品79国产精品 | 亚洲美女精品区人人人人 | 伊人色播| 在线 影视 一区 | 狠狠色丁香婷婷综合 | 91精品国产91久久久久久三级 | 国产91全国探花系列在线播放 | 粉嫩av一区二区三区四区在线观看 | 黄色网在线免费观看 | 欧美做受高潮 | 久久有精品 | 婷婷六月在线 | 91大神dom调教在线观看 | 麻豆系列在线观看 | 91九色视频在线观看 | 91精品国产高清自在线观看 | 日韩99热| 九色91在线视频 | 亚洲精品综合在线观看 | 欧美巨乳波霸 | 日韩高清成人在线 | 日韩在线视频免费播放 | 丁香导航| 九九热久久免费视频 | 国产色资源 | 五月婷婷久久丁香 | 亚洲国产成人精品在线观看 | 久久视频在线免费观看 | 婷婷精品在线 | 国产一级黄色片免费看 | 狠狠躁日日躁狂躁夜夜躁av | 91精品在线看 | 高清不卡一区二区在线 | 国产在线第三页 | 91超级碰碰 | 狠狠的干狠狠的操 | 91精品一区二区三区久久久久久 | 久久大片 | 亚洲午夜精品福利 | 亚洲精品乱码久久久久 | 久一久久 | 波多野结衣在线中文字幕 | 深爱激情五月婷婷 | 欧美日韩在线电影 | 国产a级片免费观看 | 99精品免费久久久久久久久 | 美女免费黄视频网站 | 国产精品久久久久久久久婷婷 | 成人免费观看在线视频 | 久久免费视频99 | av一区在线 | 国产精品欧美激情在线观看 | 日韩久久久久久久久 | 国产成人精品久久亚洲高清不卡 | 久久久精品 | 欧美日韩p片 | 成人三级视频 | 蜜桃视频色 | 日韩激情免费视频 | 欧美性久久久久久 | 国产免码va在线观看免费 | 国产精品手机在线 | 99久久99| 国产一区二区不卡视频 | 精品一二三区视频 | 97人人艹 | 91麻豆精品国产91久久久使用方法 | 久久电影日韩 | 久久国产精品成人免费浪潮 | 久久一视频 | www.黄色小说.com| 狠狠躁日日躁狂躁夜夜躁 | 久久精品99国产精品酒店日本 | 久久综合久久88 | 91一区二区三区在线观看 | 九九热有精品 | 一区二区伦理电影 | 色婷五月天 | 日韩理论在线观看 | 国产免费观看高清完整版 | 中文字幕高清av | 麻豆免费视频观看 | 色天天综合久久久久综合片 | 日日干天天爽 | 亚洲日本va中文字幕 | 精品毛片久久久久久 | 国产成人精品午夜在线播放 | 午夜久久影视 | 成人黄色小视频 | 欧美日韩一级久久久久久免费看 | 免费视频一二三区 | 日韩二区在线 | 国产精品视频你懂的 | www.久草.com| 色五月成人| 国产123区在线观看 国产精品麻豆91 | 欧美日韩高清在线观看 | 高清av免费看 | 日韩免费视频网站 | 性色在线视频 | 欧美色伊人 | 国产精品爽爽久久久久久蜜臀 | 国产一二三区在线观看 | 日日干激情五月 | 亚洲美女精品区人人人人 | 天天操天天射天天插 | 精品国产乱码久久久久久浪潮 | 蜜桃av观看 | bbbbb女女女女女bbbbb国产 | 国产精品毛片一区二区在线看 | 亚洲aⅴ在线观看 | 伊人影院99 | 国产精品免费观看久久 | www中文在线| 国产亚洲婷婷免费 | 国产一区二区不卡在线 | 午夜私人影院 | 日韩在线观看视频免费 | 久久综合久久综合九色 | 国精产品永久999 | 久久这里有 | 国产精品美女久久久久久 | 国产一在线精品一区在线观看 | 午夜手机看片 | 欧美另类色图 | 国产看片 色 | 一区二区视 | 国产成人一级电影 | 91精品国产亚洲 | 麻豆传媒视频在线 | 一区二区三区在线视频观看58 | 日本公妇色中文字幕 | 国产精品岛国久久久久久久久红粉 | 亚洲aⅴ乱码精品成人区 | 亚洲精品动漫在线 | 久久亚洲精品电影 | 亚洲 欧美日韩 国产 中文 | 五月婷婷一级片 | 国产精品99久久久久久久久 | 日本三级久久久 | 国产高清视频在线免费观看 | 麻豆久久精品 | 久久艹国产| 久久欧美在线电影 | 成人a在线观看高清电影 | 亚洲欧美国产精品久久久久 | 久久综合久久综合这里只有精品 | 香蕉视频一级 | 国产精品久久久久永久免费看 | 午夜影视一区 | 天天色综合久久 | 亚洲在线网址 | 午夜精品99久久免费 | 看黄色.com | 日韩大片在线看 | 黄av免费在线观看 | 99热这里只有精品在线观看 | 国产精品久久久久毛片大屁完整版 | 天天草天天草 | 91精品国产99久久久久久久 | 奇米影视8888 | 精品国产亚洲一区二区麻豆 | 日本精品视频一区二区 | 成人毛片一区二区三区 | 国内精品久久久久久中文字幕 | av中文资源在线 | 99视频国产精品免费观看 | 日韩中文字幕第一页 | 国产精品免费在线播放 | 粉嫩av一区二区三区入口 | 伊人天堂久久 | 午夜在线免费观看 | av日韩中文 | 青青草国产成人99久久 | 在线看污网站 | 久久电影色 | 午夜国产福利在线 | 在线视频 影院 | 亚洲一区精品二人人爽久久 | 最新国产在线 | 六月丁香综合 | 91人人干 | 亚洲一级电影 | 久草在线欧美 | 国产在线不卡视频 | 黄色免费视频在线观看 | www.色国产| 日日草天天干 | 超碰在线97国产 | 欧美aⅴ在线观看 | 久章草在线观看 | 国产黄影院色大全免费 | 美女视频黄免费 | 亚洲精品乱码久久久久久蜜桃欧美 | 一区三区视频在线观看 | 91丨九色丨91啦蝌蚪老版 | 日本黄色大片儿 | 日本巨乳在线 | 美女网站视频久久 | 国产首页| 国产婷婷在线观看 | 欧美日韩裸体免费视频 | 2019中文在线观看 | 国产又粗又猛又爽又黄的视频先 | 99国产成+人+综合+亚洲 欧美 | 99精品视频免费全部在线 | 天天干天天做天天爱 | 国产精品欧美日韩在线观看 | 亚洲乱码中文字幕综合 | 天天躁日日躁狠狠 | 99产精品成人啪免费网站 | 四虎在线免费 | 精品久久在线 | 国产精品成人国产乱 | 丁香婷婷综合激情 | 精品久久久久久久久久 | 亚洲欧美国产精品va在线观看 | 中文字幕在线不卡国产视频 | 日本一区二区三区免费观看 | 国产精品视频在线观看 | 日韩精品久久久久久 | 久久免费精品国产 | 欧美日本日韩aⅴ在线视频 插插插色综合 | 国产福利免费看 | 亚洲精品美女视频 | 91免费在线看片 | 国产一级做a | 中文av网 | av电影不卡在线 | 色噜噜噜噜 | 亚洲精品在线观看网站 | 国产精品久久久久婷婷 | 午夜黄色一级片 | 激情丁香综合五月 | 国产三级视频在线 | 99精品国产福利在线观看免费 | 亚州国产视频 | 国内精品久久久久久 | 天天操天天射天天爽 | 激情丁香婷婷 | 激情综合五月天 | 久久精品亚洲综合专区 | 国产高清成人 | 午夜精品一区二区三区视频免费看 | 不卡的一区二区三区 | 国产欧美日韩一区 | 欧美一级特黄aaaaaa大片在线观看 | 99久高清在线观看视频99精品热在线观看视频 | 五月天激情开心 | 91精品网站在线观看 | av高清不卡| 婷婷天天色 | 欧美一级电影免费观看 | 黄色影院在线观看 | 日韩一区视频在线 | 99国产一区 | 国产精品久久久久久久久久东京 | 超碰免费观看 | 99精品欧美一区二区三区黑人哦 | 欧美黄色免费 | 99国产在线观看 | 97天堂| 麻豆视频网址 | 在线观看av免费观看 | 激情五月婷婷综合网 | 日韩视频在线观看免费 | 最近中文字幕免费观看 | av成人免费在线 | 国产中年夫妇高潮精品视频 | 深爱激情站 | 激情欧美国产 | 国产精选视频 | 午夜久草 | 亚洲综合色丁香婷婷六月图片 | 国产欧美综合在线观看 | 亚洲人av免费网站 | 国产精品久久电影观看 | 国产精品欧美在线 | 最近高清中文在线字幕在线观看 | 香蕉影视在线观看 | 国产一区二区在线影院 | 久久婷婷国产 | 久久精品国产亚洲aⅴ | 亚洲天堂网视频在线观看 | 亚洲视频 视频在线 | 国产精品va在线观看入 | 国产中文视频 | 中文字幕在线观看一区二区 | 国产色视频一区二区三区qq号 | 天天干天天射天天操 | 国产黄色特级片 | 免费日韩| 99在线免费观看视频 | 亚洲播播 | 99久久婷婷国产综合精品 | 在线免费av播放 | 粉嫩av一区二区三区四区五区 | 久久久久国产成人精品亚洲午夜 | 丁香六月在线 | 天天操天天操天天操天天操天天操天天操 | 九九爱免费视频 | 亚洲欧美在线综合 | 日韩理论影院 | 激情综合站 | 日本福利视频在线 | 免费视频二区 | 国产精品综合久久久 | 精品主播网红福利资源观看 | 久久这里只精品 | 正在播放日韩 | 日韩精品久久久免费观看夜色 | 精品亚洲国产视频 | 97成人精品视频在线观看 | 狠狠躁日日躁夜夜躁av | 免费成人看片 | 久久美女高清视频 | 日韩精品在线看 | 一级黄色片在线免费观看 | 国产日韩精品一区二区在线观看播放 | 黄色在线看网站 | 欧美日韩国产伦理 | 欧美视频一区二 | 在线免费av网| 国产精品久久久一区二区三区网站 | 亚洲资源片| 麻豆免费视频观看 | 日韩高清不卡在线 | 精品99999 | 在线中文字幕播放 | 人人添人人澡人人澡人人人爽 | 午夜精品久久久99热福利 | 国产区av在线 | 亚洲精品免费在线观看 | 中文字幕日韩无 | 91精品导航| 国产精品成人在线 | 中文视频在线看 | av中文字幕亚洲 | 狠狠干中文字幕 | 91看成人| 久久久久久视频 | 欧美成人手机版 | 黄色大片免费网站 | 免费在线观看av片 | 成人一区二区在线观看 | 亚洲精区二区三区四区麻豆 | 很黄很色很污的网站 | 精品久久精品久久 | 亚洲第一av在线 | 久久伊人色综合 | 国产在线高清 | 97精品电影院 | 最近中文字幕免费 | 国产黄网在线 | 九九爱免费视频 | 日韩精品一区二区三区免费观看视频 | 一级性av| 久久久久 免费视频 | 天天天天干 | 日韩av影视在线观看 | 日韩电影一区二区在线观看 | 国产a精品 | 国产精品男女啪啪 | 国产精品99久久久久久有的能看 | 91porny九色91啦中文 | 日本公妇色中文字幕 | 亚洲精品456在线播放乱码 | 国产一区二区免费在线观看 | 国产精品21区| 蜜桃av观看 | 久久久久美女 | 中文字幕亚洲精品日韩 | 国产 精品 资源 | 日本 在线 视频 中文 有码 | 成人一区二区三区中文字幕 | 欧亚日韩精品一区二区在线 | 99精品久久久久久久久久综合 | 午夜久久网| 免费福利片2019潦草影视午夜 | 蜜臀久久99精品久久久酒店新书 | 97在线精品视频 | 日韩久久精品一区 | 高清不卡免费视频 | 久久视奸 | 欧美aa在线| a级片韩国 | 天天天天色射综合 | 夜夜高潮夜夜爽国产伦精品 | 日韩两性视频 | 成人av中文字幕在线观看 | 国产日韩欧美综合在线 | 亚洲成人资源在线观看 | 免费h视频 | 日韩美女av在线 | 99热这里只有精品在线观看 | 国产黄色美女 | 在线黄频| 天天操天天射天天舔 | 国产精品99久久久久的智能播放 | 国产视频在线观看一区二区 | 99爱精品视频 | 日韩精品视频在线观看网址 | 国内免费的中文字幕 | 91黄色免费看 | 国产人在线成免费视频 | 免费a网站| 欧美视频二区 | 免费在线观看不卡av | 成人蜜桃网 | 六月色丁香 | 成人h在线观看 | 久久精品视频免费播放 | 亚洲一区二区视频 | 亚洲成熟女人毛片在线 | av免费播放 | 免费成人在线观看视频 | 国产性天天综合网 | 成年人电影毛片 | 午夜久久福利影院 | 婷婷久久综合九色综合 | 亚洲一区 av | 日本一区二区不卡高清 | 久久午夜精品影院一区 | 久久亚洲二区 | 国产精品99免视看9 国产精品毛片一区视频 | 成人午夜精品久久久久久久3d | 国产成人黄色在线 | 国产精品视屏 | 免费av 在线 | 日韩久久精品一区二区 | 天天操夜夜操夜夜操 | 人人澡人人爽欧一区 | 狠狠干成人综合网 | 麻豆视频国产 | 久久伦理电影网 | av片无限看| 欧美激情视频免费看 | 日韩av在线高清 | 开心激情五月婷婷 | 久久激情婷婷 | 欧美污污网站 | 中文字幕字幕中文 | 久久99精品久久久久婷婷 | 日韩色中色| 国产精品欧美久久久久久 | 在线观看一区视频 | 久久婷婷精品视频 | 精品在线你懂的 | 久久精品国产亚洲精品2020 | 国产精品免费一区二区三区 | www.av小说 | 久久久精品午夜 | 97网在线观看 | 69视频在线播放 | 激情久久综合网 | 亚洲涩涩涩涩涩涩 | 456免费视频 | 狠狠狠的干 | 婷婷在线不卡 | av黄色av| 久久久国产精品免费 | 一区二区三区四区五区在线 | 99久久99久国产黄毛片 | 麻豆一区在线观看 | 国产精品永久久久久久久久久 | 91污在线观看 | 91麻豆精品久久久久久 | av成人免费在线看 | 亚洲欧洲日韩 | 久久九九免费视频 | 色婷婷激情四射 | 韩日色视频 | 久久久免费毛片 | 国产麻豆精品久久 | 精品一区欧美 | 97超级碰碰碰视频在线观看 | 九热精品| 亚洲第一av在线播放 | 韩日电影在线 | 超级碰碰免费视频 | 欧美大片大全 | 国产成人99av超碰超爽 | 深夜免费小视频 | 成年人黄色大片在线 | 亚洲国产中文字幕在线视频综合 | 国产区在线 | 99国产精品一区二区 | www.五月婷婷 | 麻豆视屏| 精品国产aⅴ麻豆 | 色多视频在线观看 | 综合国产在线 | 国产在线精品一区二区 | 18性欧美xxxⅹ性满足 | 99这里有精品 | 日韩理论电影在线观看 | 久久综合亚洲鲁鲁五月久久 | 在线免费观看国产视频 | 97超级碰| 91麻豆精品91久久久久同性 | 日韩网站一区 | 日韩丝袜在线 | 成人看片| 中文字幕一区二区三区在线播放 | 综合网五月天 | 狠狠操狠狠操 | 人交video另类hd| 国产一区二区精品 | 天天射天天艹 | 视频福利在线 | 午夜视频二区 | 久久免费视频7 | 国产高清在线a视频大全 |