本文由嵌入式企鹅圈原创团队成员、阿里资深工程师Hao分享。java
上篇文章《Android无线开发的几种经常使用技术》咱们介绍了几种android移动应用开发中的经常使用技术,其中的热补丁正在被愈来愈多的开发团队所使用,它涉及到dalvik虚拟机和android的一些核心技术,如今就来介绍下它的一些原理。android
本篇先介绍dexposed方案:https://github.com/alibaba/dexposed,它是手机淘宝团队使用的热补丁方案,后来开源到github上,取的名字dexposed代表了本身是基于大名鼎鼎的xposed hook方案,有饮水思源、回馈开源项目的意思。与xposed不一样的是,dexposed是本身hook本身的应用,所以不须要root权限。git
它的关键点是:在native层中先找到要修复的Java函数对应的Method对象,修改它变为native方法,把它的nativeFunc指向hookedMethodCallback。这样对这个java函数的调用就转为调用hookedMethodCallback这个native函数了,而后再用这个native函数回调java层本身实现的统一接口来处理。这个统一接口是XC_MethodReplacement类,它主要有beforeHookedMethod、afterHookedMethod和replaceHookedMethod等几个方法,前两个在执行原java函数先后作一些事,replaceHookedMethod则是替换原java方法。下面来详细分析下这个hook的过程。github
1、DexposedBridge.findAndHookMethod数据结构
findAndHookMethod是hook原java方法的入口,它传入的参数是类Class和方法名,最后一个可变参数parameterTypesAndCallback,是用户实现的用于替换原方法的XC_MethodReplacement的实例。函数
先调用XposedHelpers.findMethodExact找到要hook的java方法,再用hookMethod进行真正的hook。.net
1.findMethodExact根据类名和方法名,用反射找到Method,并把它的属性改成可访问。3d
2.hookMethod先把hook成功后的callback、要hook的方法的参数和返回值类型保存到AdditionalHookInfo中,把它做为参数传给hookMethodNative。hookMethodNative是一个native方法,它的第3个参数slot表示该Method在class的方法表中所处的位置,在native的实现中会用到这个slot。指针
hookMethodNative的实现环境分dalvik和art,由于dexposed对art的支持不完善,同时art自己的原理和机制也是一个难点,因此本篇只介绍dalvik下的实现,art的有关内容之后有机会再做介绍。code
2、hookMethodNative
每个java的类在虚拟机的实现中都对应着一个C++的ClassObject。dvmDecodeIndirectRef是libdvm中的方法,它能够从java对象的间接引用得到ClassObject对象,再根据slot,用dvmSlotToMethod找到Method对象。这里的ClassObject和Method都是虚拟机内部用来表示class和Method的数据结构。
而后把原来的Method结构先备份到XposedHookInfo中,
XposedHookInfo的结构以下:
可见,它用originalMethod保存原来java方法的Method,用reflectedMethod保存原java方法在native的引用,注意这跟originalMethod中保存的Method对象不一样。originalMethod中保存的Method能够理解为执行字节码的地址,而reflectedMethod中保存的是用来描述原java方法的一个ClassObject对象。它们两个在第五部分从新调用原java方法时会用到。
additionalInfo用来保存附加信息AdditionalHookInfo。接着使用SET_METHOD_FLAG宏把该方法设为native,让nativeFunc指向hookedMethodCallback,这样对该java方法的调用就会转为对hookedMethodCallback这个native方法的调用了。Insns指向这个方法的字节码,在这里把它改成指向hookInfo,实际上也就是originalMethod的字节码的地址。
3、hookedMethodCallback
hookedMethodCallback会回调java层的方法handleHookedMethod,最终会调用到前面说过的,在findAndHookMethod中传入的XC_MethodReplacement里的before、after方法。
这里首先把在hookInfo中保存的信息做为传给java层的handleHookedMethod的参数,而后用dvmCallMethod这个dalvik的函数调用xposedHandleHookedMethod这个java的方法。xposedHandleHookedMethod在初始化时已经被设置好了。
GetStaticMethodId是dvm中用来获取静态方法地址的函数,可见在初始化时,已经把java的静态方法handleHookedMethod的地址赋给了xposedHandleHookedMethod了。这里须要注意两点,一是这个时候已经不能像没hook以前那样,经过jni从native调java的函数,或者从java调native的函数,由于原来java方法的上下文已经被改变了(已经被保存在hookInfo中),因此后面只能经过libdvm中的方法,手动修改函数的指向。二是dvmCallMethod的第5和第6个参数originalReflected和original就是第二部分中保存的方法的引用和方法的字节码地址(original被直接转成了int型),后面第五部分中这两项还会被从新传回native层用来找到原java函数的入口。
4、handleHookedMethod
前面说到从native中调回java的方法handleHookedMethod,handleHookedMethod会根据须要,选择是否还调用原来的java方法,或者只调用XC_MethodReplacement里本身实现的before、after方法。
其中beforeHookedMethod方法默认会调用replaceHookedMethod,咱们只要实现它便可替代对原方法的调用。
若是param.returnEarly为false才会调invokeOriginalMethodNative执行原来的方法。
默认的beforeHookedMethod中会调setParam,把param.returnEarly的值设为为true,这样就不会再调用原来的java方法了。
最后还要把返回值返回。
5、invokeOriginalMethodNative
若是在java层须要从新调用原java函数,那么在第二部分中把原java函数的信息备份到hookInfo中就能起到做用了。Java层的invokeOriginalMethod方法会调一个native的方法invokeOriginalMethodNative来实现这个过程。
这个native函数一样在初始化时就被设置好了:
要调用的invokeOriginalMethodNative在虚拟机中Method是dexposedInvokeOriginalMethod,这里传入了第二部分中备份的原java方法的对象引用reflectedMethod和字节码地址int型的original。
dvmSetNativeFunc的第2个参数是DalvikBridgeFunc类型的指针,这个函数会把dexposedInvokeOriginalMethod的nativeFunc指向xxx_invokeOriginalMethodNative。再次注意此时不能像日常的jni调用那样,java层的invokeOriginalMethodNative通过jni注册后能直接调到com_taobao_android_dexposed_DexposedBridge_invokeOriginalMethodNative了。
dvmInvokeMethod跟dvmCallMethod同样,都是dalvik中用来直接调Method的函数,这样就完成了对原java方法的调用。
最后一句话归纳这种hook方法,就是经过把原java方法的类型改成native来把对java函数的调用转到native层,在native层用dvm的各类函数来操做Method的指针和对象来控制函数流程。