肥宅钓鱼网
当前位置: 首页 钓鱼百科

hook深度剖析(Hook技术简介)

时间:2023-06-08 作者: 小编 阅读量: 3 栏目名: 钓鱼百科

在Android操作系统中系统维护着自己的一套事件分发机制。应用程序,包括应用触发事件和后台逻辑处理,也是根据事件流程一步步地向下执行。APIHook技术是一种用于改变API执行结果的技术,能够将系统的API函数执行重定向。在Android系统中使用了沙箱机制,普通用户程序的进程空间都是独立的,程序的运行互不干扰。当然,根据Hook对象与Hook后处理的事件方式不同,Hook还分为不同的种类,比如消息Hook、APIHook等。

1. 什么是 Hook

Hook 英文翻译过来就是「钩子」的意思,那我们在什么时候使用这个「钩子」呢?在 Android 操作系统中系统维护着自己的一套事件分发机制。应用程序,包括应用触发事件和后台逻辑处理,也是根据事件流程一步步地向下执行。而「钩子」的意思,就是在事件传送到终点前截获并监控事件的传输,像个钩子钩上事件一样,并且能够在钩上事件时,处理一些自己特定的事件。

Hook 的这个本领,使它能够将自身的代码「融入」被勾住(Hook)的程序的进程中,成为目标进程的一个部分。API Hook 技术是一种用于改变 API 执行结果的技术,能够将系统的 API 函数执行重定向。在 Android 系统中使用了沙箱机制,普通用户程序的进程空间都是独立的,程序的运行互不干扰。这就使我们希望通过一个程序改变其他程序的某些行为的想法不能直接实现,但是 Hook 的出现给我们开拓了解决此类问题的道路。当然,根据 Hook 对象与 Hook 后处理的事件方式不同,Hook 还分为不同的种类,比如消息 Hook、API Hook 等。

  1. 使用 Java 反射实现 API Hook
  2. 通过对 Android 平台的虚拟机注入与 Java 反射的方式,来改变 Android 虚拟机调用函数的方式(ClassLoader),从而达到 Java 函数重定向的目的,在这里我们将此类操作称为 Java API Hook。

下面通过 Hook View 的 OnClickListener 来说明 Hook 的使用方法。

首先进入 View 的 setOnClickListener 方法,我们看到 OnClickListener 对象被保存在了一个叫做 ListenerInfo 内部类里,其中 mListenerInfo 是 View 的成员变量。ListeneInfo 里面保存了 View 的各种监听事件,比如 OnClickListener、OnLongClickListener、OnKeyListener 等等。

public void setOnClickListener(@Nullable OnClickListener l) {if (!isClickable()) {setClickable(true);}getListenerInfo().mOnClickListener = l;}ListenerInfo getListenerInfo() {if (mListenerInfo != null) {return mListenerInfo;}mListenerInfo = new ListenerInfo();return mListenerInfo;}

我们的目标是 Hook OnClickListener,所以就要在给 View 设置监听事件后,替换 OnClickListener 对象,注入自定义的操作。

private void hookOnClickListener(View view) {try {// 得到 View 的 ListenerInfo 对象Method getListenerInfo = View.class.getDeclaredMethod("getListenerInfo");getListenerInfo.setAccessible(true);Object listenerInfo = getListenerInfo.invoke(view);// 得到 原始的 OnClickListener 对象Class<?> listenerInfoClz = Class.forName("android.view.View$ListenerInfo");Field mOnClickListener = listenerInfoClz.getDeclaredField("mOnClickListener");mOnClickListener.setAccessible(true);View.OnClickListener originOnClickListener = (View.OnClickListener) mOnClickListener.get(listenerInfo);// 用自定义的 OnClickListener 替换原始的 OnClickListenerView.OnClickListener hookedOnClickListener = new HookedOnClickListener(originOnClickListener);mOnClickListener.set(listenerInfo, hookedOnClickListener);} catch (Exception e) {log.warn("hook clickListener failed!", e);}}class HookedOnClickListener implements View.OnClickListener {private View.OnClickListener origin;HookedOnClickListener(View.OnClickListener origin) {this.origin = origin;}@Overridepublic void onClick(View v) {Toast.makeText(Mainactivity.this, "hook click", Toast.LENGTH_SHORT).show();log.info("Before click, do what you want to to.");if (origin != null) {origin.onClick(v);}log.info("After click, do what you want to to.");}}

到这里,我们成功 Hook 了 OnClickListener,在点击之前和点击之后可以执行某些操作,达到了我们的目的。下面是调用的部分,再给 Button 设置 OnClickListener 后,执行 Hook 操作。点击按钮后,日志的打印结果是:Before click → onClick → After click。

Button btnSend = (Button) findViewById(R.id.btn_send);btnSend.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {log.info("onClick");}});hookOnClickListener(btnSend);

我们再来看一个很常见的例子 startActivity

下面我们Hook掉 startActivity 这个方法,使得每次调用这个方法之前输出一条日志;(当然,这个输入日志有点点弱,只是为了展示原理,如果你想可以替换参数,拦截这个 startActivity 过程,使得调用它导致启动某个别的Activity,指鹿为马!)

我们知道对于Context.startActivity,Context的实现实际上是ContextImpl;我们看ConetxtImpl类的startActivity方法:

@Overridepublic void startActivity(Intent intent, Bundle options) {warnIfCallingFromSystemProcess();if ((intent.getFlags()&Intent.FLAG_ACTIVITY_NEW_TASK) == 0) {throw new AndroidRuntimeException("Calling startActivity() from outside of an Activity "" context requires the FLAG_ACTIVITY_NEW_TASK flag."" Is this really what you want?");}mMainThread.getInstrumentation().execStartActivity(getOuterContext(), mMainThread.getApplicationThread(), null,(Activity)null, intent, -1, options);}

这里,实际上使用了 ActivityThread 类的 mInstrumentation 成员的 execStartActivity 方法;注意到, ActivityThread 实际上是主线程,而主线程一个进程只有一个,因此这里是一个良好的Hook点。

接下来就是想要Hook掉我们的主线程对象,也就是把这个主线程对象里面的 mInstrumentation 被替换成我们修改过的代理对象;要替换主线程对象里面的字段,首先我们得拿到主线程对象的引用,如何获取呢? ActivityThread 类里面有一个静态方法 currentActivityThread 可以帮助我们拿到这个对象类;但是 ActivityThread 是一个隐藏类,我们需要用反射去获取,代码如下:

// 先获取到当前的ActivityThread对象Class<?> activityThreadClass = Class.forName("android.app.ActivityThread");Method currentActivityThreadMethod = activityThreadClass.getDeclaredMethod("currentActivityThread");currentActivityThreadMethod.setAccessible(true);Object currentActivityThread = currentActivityThreadMethod.invoke(null);

拿到这个 currentActivityThread 之后,我们需要修改它的 mInstrumentation 这个字段为我们的代理对象,我们先实现这个代理对象,由于JDK动态代理只支持接口,而这个 Instrumentation 是一个类,没办法,我们只有手动写静态代理类,覆盖掉原始的方法即可。( cglib 可以做到基于类的动态代理,这里先不介绍)

public class EvilInstrumentation extends Instrumentation {private static final String TAG = "EvilInstrumentation";// ActivityThread中原始的对象, 保存起来Instrumentation mBase;public EvilInstrumentation(Instrumentation base) {mBase = base;}public ActivityResult execStartActivity(Context who, IBinder contextThread, IBinder token, Activity target,Intent intent, int requestCode, Bundle options) {// Hook之前, XXX到此一游!Log.d(TAG, "\n执行了startActivity, 参数如下: \n""who = ["who"], ""\ncontextThread = ["contextThread"], \ntoken = ["token"], ""\ntarget = ["target"], \nintent = ["intent"], \nrequestCode = ["requestCode"], \noptions = ["options"]");// 开始调用原始的方法, 调不调用随你,但是不调用的话, 所有的startActivity都失效了.// 由于这个方法是隐藏的,因此需要使用反射调用;首先找到这个方法try {Method execStartActivity = Instrumentation.class.getDeclaredMethod("execStartActivity",Context.class, IBinder.class, IBinder.class, Activity.class, Intent.class, int.class, Bundle.class);execStartActivity.setAccessible(true);return (ActivityResult) execStartActivity.invoke(mBase, who, contextThread, token, target, intent, requestCode, options);} catch (Exception e) {// 某该死的rom修改了需要手动适配throw new RuntimeException("do not support!!! pls adapt it");}}}

Ok,有了代理对象,我们要做的就是偷梁换柱!代码比较简单,采用反射直接修改:

public static void attactContext() throws Exception{// 先获取到当前的ActivityThread对象Class<?> activityThreadClass = Class.forName("android.app.ActivityThread");Field currentActivityThreadField = activityThreadClass.getDeclaredField("sCurrentActivityThread");currentActivityThreadField.setAccessible(true);Object currentActivityThread = currentActivityThreadField.get(null);// 拿到原始的 mInstrumentation字段Field mInstrumentationField = activityThreadClass.getField("mInstrumentation");mInstrumentationField.setAccessible(true);Instrumentation mInstrumentation = (Instrumentation) mInstrumentationField.get(currentActivityThread);// 创建代理对象Instrumentation evilInstrumentation = new EvilInstrumentation(mInstrumentation);// 偷梁换柱mInstrumentationField.set(currentActivityThread, evilInstrumentation);}

好了,我们启动一个Activity测试一下,结果如下:

总结一下:

Hook 过程:

寻找 Hook 点,原则是静态变量或者单例对象,尽量 Hook public 的对象和方法。

选择合适的代理方式,如果是接口可以用动态代理。

偷梁换柱——用代理对象替换原始对象。

Android 的 API 版本比较多,方法和类可能不一样,所以要做好 API 的兼容工作。

举个例子

Android10后添加了 ActivityTaskManager

int result = ActivityTaskManager.getService().startActivity(whoThread,who.getBasePackageName(), who.getAttributionTag(), intent,intent.resolveTypeIfNeeded(who.getContentResolver()), token,target != null ? target.mEmbeddedID : null, requestCode, 0, null, options);

int result = ActivityManagerNative.getDefault().startActivity(whoThread, who.getBasePackageName(), intent,intent.resolveTypeIfNeeded(who.getContentResolver()),token, target != null ? target.mEmbeddedID : null,requestCode, 0, null, null, options);

2. Xposed

通过替换 /system/bin/app_process 程序控制 Zygote 进程,使得 app_process 在启动过程中会加载 XposedBridge.jar 这个 Jar 包,从而完成对 Zygote 进程及其创建的 Dalvik 虚拟机的劫持。

Xposed 在开机的时候完成对所有的 Hook Function 的劫持,在原 Function 执行的前后加上自定义代码。

现在安装Xposed比较方便,因为Xposed作者开发了一个Xposed Installer App,下载后按照提示傻瓜式安装(前提是root手机)。其实它的安装过程是这个样子的:首先探测手机型号,然后按照手机版本下载不同的刷机包,最后把Xposed刷机包刷入手机重启就好。刷机包下载 里面有所有版本的刷机包。

刷机包解压打开里面的问件构成是这个样子的:

META-INF/里面有文件配置脚本 flash-script.sh 配置各个文件安装位置。system/bin/替换zygote进程等文件system/framework/XposedBridge.jar jar包位置system/lib system/lib64 一些so文件所在位置xposed.prop xposed版本说明文件

所以安装Xposed的过程就上把上面这些文件放到手机里相同文件路径下。

通过查看文件安装脚本发现:

system/bin/下面的文件替换了app_process等文件,app_process就是zygote进程文件。所以Xposed通过替换zygote进程实现了控制手机上所有app进程。因为所有app进程都是由Zygote fork出来的。

Xposed的基本原理是修改了ART/Davilk虚拟机,将需要hook的函数注册为Native层函数。当执行到这一函数是虚拟机会优先执行Native层函数,然后再去执行Java层函数,这样完成函数的hook。如下图:

通过读Xposed源码发现其启动过程:

  1. 手机启动时init进程会启动zygote这个进程。由于zygote进程文件app_process已被替换,所以启动的时Xposed版的zygote进程。
  2. Xposed_zygote进程启动后会初始化一些so文件(system/lib system/lib64),然后进入XposedBridge.jar中的XposedBridge.main中初始化jar包完成对一些关键Android系统函数的hook。
  3. Hook则是利用修改过的虚拟机将函数注册为native函数。
  4. 然后再返回zygote中完成原本zygote需要做的工作。
  5. 这只是在宏观层面稍微介绍了下Xposed,要想详细了解需要读它的源码了。
    推荐阅读
  • 网址哪个好用又实用(有用网址留着以后用)

    有用网址,留着以后用只要是学生,看一眼,你就会爱上它!5、论文帝国http://www.papersempire.com/二、教育类1、教研论文交流中心http://www.k12.com.cn/teacher/resource/lunwen/以中小学教育为主,基础教育、英语教学文章居多。

  • 不要骗局(不要上当)

    “预售类”骗局“预售”是近几年“618”期间各购物平台常用的营销方式,很多平台商家为了赢得先机,会提前通过短信或微信,将打折预售信息发给顾客。“快递类”骗局“618”过后,诈骗分子利用非法获取的快递信息,以快递损坏或者丢失为由群发短信,让受害人添加其QQ或微信好友,随后拉入一个满是“托”的群聊中,诱骗受害人在群聊“刷单”。

  • 安徽风俗(安徽省有哪些风俗)

    安徽风俗阜阳剪纸。安徽省阜阳地区的剪纸,应用范围很广。徽州宗祠祭祀是徽州宗族的一项重要的礼仪活动,曾广泛流传于古徽州的祁门县、歙县、黟县、绩溪等地。特别是箬坑乡马山村整个春祭活动环境热烈、过程繁杂、组织完整、程序规范、仪式隆重,有较强的宗族色彩。为研究徽州宗族的祭祀礼仪、以及民间音乐有着较好的参考价值。

  • 最正确的养柯基(新手如何养柯基)

    选狗粮的首要原则是,天然粮>商品粮。生煎现在基本在渴望的六种鱼、雪山的甜薯鸭肉和wellness三者之间轮换。商品粮方面,可以去旗舰店买皇家的狗粮,算是顶级商品粮,也是生煎狗粮的最低配。建议根据狗子的体型选择不倒翁的尺寸。可以参考生煎的女儿ruby每天欺负她家的猫咪榛子的故事。

  • 曝黄色iPhone 黄色iphone xr

    有消息称,苹果将在本周发布全新的iPhone14和iPhone14Plus的特殊配色,目前已经确定了是黄色版本。

  • 怎么让面条爽滑劲道(如何让面条爽滑劲道)

    以下内容大家不妨参考一二希望能帮到您!怎么让面条爽滑劲道取面粉500g。在面粉中加入2g盐。再在面粉中打入一个鸡蛋。和面时水可以少量多次的加。将面粉和成一块光滑、较硬的面团,饧30分钟。将面团揉匀分成几块,先压制面片,再压制出粗细适中的面条。制好的面条表面撒一些干面防止粘连,在案板上晾干几分钟,就可以煮制各种你喜爱的口味的面条,劲道的口感绝对令人满意。

  • 梦幻西游三维版普陀特技选择(梦幻西游三维版)

    很多少侠都会选择“力劈华山”这个技能在PVP中打满输出,100%重击在PVP中优势十分大,那么召唤兽的其他技能该选择哪些才能最好的搭配“力劈华山”打出高伤害呢?也许有小伙伴就要问了,力劈华山已经是100%重击了,还需要重击干什么?这就要回到前文提到的召唤兽的输出组成以及装备力劈华山这个技能的初衷了。

  • 弱混和强混的区别(弱混和强混的区别声乐)

    弱混和强混的区别,主要就是对电能的依赖程度不同。也可以说,弱混与强混的区别,在于电动机和蓄电池组的威力不同。弱混的电机不直接参与驱动,主要用于启动和回收制动能量。弱混车型的电动机和电池组功率都很小,不能支持车辆作纯电动的行驶,电动机主要在引擎加速时帮一把,起到提高引擎效率的作用。而强混车型电动机的功率更为强大,除了有弱混的优点,还可以完全满足车辆在起步和低速时的动力要求。

  • 考试前送什么礼物(预祝考试成功送什么礼物最好)

    不管对于男孩子还是女孩子来说,送他们衣服都是非常好的选择,比如红色的衣服,不但代表着红红火火的意思,同时适合他们的穿搭更会带给他们一种不同的心境。当然也有家长会选择送孩子钢笔,在一定程度上也是代表节节高升的意思。礼物在一定程度上只是表达自己心意的一种方式,想要更好的表达自己的爱,就要送一些孩子喜欢又需要的礼物才好。

  • 等1289台汽车所有人缴纳罚款的公告(等1289台汽车所有人缴纳罚款的公告)

    等1289台汽车所有人缴纳罚款的公告豫U78222等1289台汽车所有人:你名下的车辆因交通违法被电子监控抓拍后,交警部门先后通过邮寄违法行为处理通知书等方式督促你接受处理,消除安全隐患。截至目前,你名下车辆的交通违法行为已处理未交款。