【Frida】【Android】02_JAVA层HOOK

2024-03-26 23:28
文章标签 java android 02 hook frida

本文主要是介绍【Frida】【Android】02_JAVA层HOOK,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

🛫 系列文章导航

  • 【Frida】【Android】01_手把手教你环境搭建 https://blog.csdn.net/kinghzking/article/details/136986950
  • 【Frida】【Android】02_JAVA层HOOK https://blog.csdn.net/kinghzking/article/details/137008446
  • 【Frida】【Android】03_RPC https://blog.csdn.net/kinghzking/article/details/137050967

▒ 目录 ▒

    • 🛫 系列文章导航
    • 🛫 导读
      • 开发环境
    • 1️⃣ Android App 代码介绍
      • 中文内部类和方法
      • 重载函数
      • 成员变量及静态函数
    • 2️⃣ 普通函数HOOK
      • Java.perform
      • 操作步骤
    • 3️⃣ 重载函数HOOK
    • 4️⃣ 构造函数HOOK
    • 5️⃣ 函数主动调用
    • 🛬 文章小结
      • frida api释义
    • 📖 参考资料

🛫 导读

开发环境

版本号描述
文章日期2024-03-24
操作系统Win11 - 22H222621.2715
node -vv20.10.0
npm -v10.2.3
夜神模拟器7.0.5.8
Android9
python3.9.9
frida16.2.1
frida-tools12.3.0
objection1.11.0

1️⃣ Android App 代码介绍

在本小节中,我们将开始学习Frida在Android Java中的Hook功能。为了清楚地了解App的运行内容,这里先编写一个简单的App用于练习。其主要使用逻辑为,通过点击测试按钮,触发各种函数调用。通过adb logcat | findstr yemao查看运行结果。
在这里插入图片描述

中文内部类和方法

该类逻辑相对简单,包含一个中文方法,内部打印一句话。
除此之外,编译器会生成一个默认的构造函数

在这里插入图片描述

设计这个类的原因有以下几点:

  • 演示普通函数HOOK
  • 演示构造函数HOOK
  • 为了测试中文相关名称处理方法
  • 演示子类Smali命名规则

重载函数

在入口类MainActivity中添加了两个相同名称的函数(重载函数),接受不同的参数,用于演示重载函数HOOK功能。

public class MainActivity extends AppCompatActivity {void fun(int x , int y ){Log.d("yemao.sum" , String.valueOf(x+y));}String fun(String x){return x.toLowerCase();}
}

成员变量及静态函数

除此之外,还预留了成员变量及静态函数,用于下面几个测试:

  • 修改成员变量
  • 主动调用静态函数
  • frida rpc测试
public class MainActivity extends AppCompatActivity {private String total = "hello";void secret(){total += " secretFunc";Log.d("yemao.secret" , "this is secret func");}static void staticSecret(){Log.d("yemao.secret" , "this is static secret func");}}

2️⃣ 普通函数HOOK

这里以com.yemao.demo.MainActivity.非重载测试类的方法普通函数为例,介绍普通函数HOOK,其它hook在此基础上进行解释。

Java.perform

用Frida的API函数Java.perform()将脚本中的内容注入到Java运行库,这个API的参数是一个函数,函数内容是监控和修改Java函数逻辑的主体内容。
注意,这里的Java.perform()函数非常重要,任何对App中Java层的操作都必须包裹在这个函数中,否则Frida运行起来后就会报错。


function main() {普通函数()重载函数()构造函数()函数主动调用()
}Java.perform(main)

操作步骤

  • 首先调用了Frida的API函数Java.use(),这个函数的参数是Hook的函数所在类的类名,参数的类型是一个字符串类型,比如Hook的中文方法()函数所在类的全名为com.roysue.demo02.MainActivity$非重载测试类,那么传递给这个函数的参数就是com.roysue.demo02.MainActivity$非重载测试类。这个函数的返回值动态地为相应Java类获取一个JavaScript Wrapper,可以通俗地理解为一个JavaScript对象。
    这里特别注意的是,内部类命名需要使用$连接!!!
  • 接着,通过JavaScript Wrapper获取其方法函数对象,有以下两种方法:
    • 如果是英文,直接通过.即可,例如cls.fun。(也可以使用cls['fun']
    • 对于中文,则需要使用[]获取方法对象:cls['中文方法']
  • 然后加上implementation 关键词表示实现,通过对其进行赋值,替换原有函数逻辑。
    通过this['中文方法']()函数再次调用原函数,并把原本的参数传递给这个函数。简而言之,就是重新执行原函数的内容,最后将这个函数的返回值直接通过return指令返回。

function 普通函数() {var cls = Java.use('com.yemao.demo.MainActivity$非重载测试类')console.log("Java.Use.Successfully!") //定位类成功!cls['中文方法'].implementation = function(x: number, y: number) {console.log("[普通函数] x => ",x,", y => ",y)// this['中文方法'] 可以获得Hook之前的函数,可以直接调用之前的函数// 这时候,可以根据自己的需求,修改参数等!!!var ret_value = this['中文方法'](x, y);return ret_value}
}

ps: 在Hook一个函数时,还有一个地方需要注意,那就是最好不要修改被Hook的函数的返回值类型,否则可能会引起程序崩溃等问题,比如直接通过调用原函数将原函数的返回值返回。

3️⃣ 重载函数HOOK

重载函数HOOK逻辑如果使用普通函数HOOK中的代码,会报错Error: fun(): has more than one overload。这是函数的重载导致Frida不知道具体应该Hook哪个函数而出现的问 题。 其 实 Frida 已 经 提 供 了 解 决 方 案 ( use .overload(<signature>), 就 是 指 定 函 数 签 名 , 将 报 错 中的.overload(‘java.lang.String’)者.overload(‘int’, ‘int’)添加到要Hook的函数名后、关键词implementation之前。


function 普通函数() {var MainAcitivity = Java.use('com.yemao.demo.MainActivity')console.log("Java.Use.Successfully!") //定位类成功!MainAcitivity.fun.overload('int', 'int').implementation = function(x: number, y: number) {console.log("[重载函数] x => ",x,", y => ",y)var ret_value = this.fun(x, y);return ret_value}
}

4️⃣ 构造函数HOOK

构造函数比普通函数,只有函数命名上特殊一点,frida采用了$init代表构造函数。
本例子使用的是内部类,相对于普通类,会多传递一个参数外部类对象,这是java语法特性,可以通过查看Smali代码看出该参数的传递,所以

  • 必须增加该参数mainAcitivity,否则程序会崩溃的!!!
  • 必须增加该参数mainAcitivity,否则程序会崩溃的!!!
  • 必须增加该参数mainAcitivity,否则程序会崩溃的!!!

function 普通函数() {var cls = Java.use('com.yemao.demo.MainActivity$非重载测试类')console.log("Java.Use.Successfully!") //定位类成功!cls.$init.implementation = function(mainAcitivity) {console.log("[构造函数] ===")var ret_value = this.$init(mainAcitivity);return ret_value}
}

5️⃣ 函数主动调用

在Java中,类中的函数可分为两种:类函数实例方法。通俗地讲,就是静态的方法和动态的方法。类函数使用关键字static修饰,和对应类是绑定的,如果类函数还被public关键词修饰着,在外部就可以直接通过类去调用;实例方法则没有关键字static修饰,在外部只能通过创建对应类的实例再通过这个实例去调用。
如果是类函数的主动调用,直接使用Java.use()函数找到类进行调用即可;
如果是实例方法的主动调用,则需要在找到对应的实例后对方法进行调用。这里用到了Frida中非常重要的一个API函数Java.choose(),这个函数可以在Java的堆中寻找指定类的实例。


function 函数主动调用() {var MainAcitivity = Java.use('com.yemao.demo.MainActivity')console.log("Java.Use.Successfully! 函数主动调用") //定位类成功!// 静态函数主动调用MainAcitivity.staticSecret();// Error: secret: cannot call instance method without an instance// MainAcitivity.secret();// 动态函数主动调用Java.choose('com.yemao.demo.MainActivity',{onMatch: function(instance){console.log('instance found',instance)instance.secret()},onComplete: function(){console.log('search Complete')}})
}

🛬 文章小结

frida api释义

函数名参数返回值含义
Java.perform函数任何对App中Java层的操作都必须包裹在这个函数
中,否则Frida运行起来后就会报错
Java.useclassNameJavaScript wrapper动态获取类实例对象
MainAcitivity.fun.implementation函数实现
MainAcitivity.fun.overload参数数组重载函数中,区分不同的函数
Java.chooseclassName, callbacks枚举className的实例对象,调用callbacks处理回调逻辑

📖 参考资料

  • 《安卓Frida逆向与抓包实战》
  • 【Frida】 00_简单介绍和使用 https://blog.csdn.net/kinghzking/article/details/123225580
  • 本节源码地址 https://gitcode.com/android8/AndroidFridaBeginnersBook

ps: 文章中内容仅用于技术交流,请勿用于违规违法行为。

这篇关于【Frida】【Android】02_JAVA层HOOK的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/850208

相关文章

JVM 的类初始化机制

前言 当你在 Java 程序中new对象时,有没有考虑过 JVM 是如何把静态的字节码(byte code)转化为运行时对象的呢,这个问题看似简单,但清楚的同学相信也不会太多,这篇文章首先介绍 JVM 类初始化的机制,然后给出几个易出错的实例来分析,帮助大家更好理解这个知识点。 JVM 将字节码转化为运行时对象分为三个阶段,分别是:loading 、Linking、initialization

Spring Security 基于表达式的权限控制

前言 spring security 3.0已经可以使用spring el表达式来控制授权,允许在表达式中使用复杂的布尔逻辑来控制访问的权限。 常见的表达式 Spring Security可用表达式对象的基类是SecurityExpressionRoot。 表达式描述hasRole([role])用户拥有制定的角色时返回true (Spring security默认会带有ROLE_前缀),去

浅析Spring Security认证过程

类图 为了方便理解Spring Security认证流程,特意画了如下的类图,包含相关的核心认证类 概述 核心验证器 AuthenticationManager 该对象提供了认证方法的入口,接收一个Authentiaton对象作为参数; public interface AuthenticationManager {Authentication authenticate(Authenti

Spring Security--Architecture Overview

1 核心组件 这一节主要介绍一些在Spring Security中常见且核心的Java类,它们之间的依赖,构建起了整个框架。想要理解整个架构,最起码得对这些类眼熟。 1.1 SecurityContextHolder SecurityContextHolder用于存储安全上下文(security context)的信息。当前操作的用户是谁,该用户是否已经被认证,他拥有哪些角色权限…这些都被保

Spring Security基于数据库验证流程详解

Spring Security 校验流程图 相关解释说明(认真看哦) AbstractAuthenticationProcessingFilter 抽象类 /*** 调用 #requiresAuthentication(HttpServletRequest, HttpServletResponse) 决定是否需要进行验证操作。* 如果需要验证,则会调用 #attemptAuthentica

Spring Security 从入门到进阶系列教程

Spring Security 入门系列 《保护 Web 应用的安全》 《Spring-Security-入门(一):登录与退出》 《Spring-Security-入门(二):基于数据库验证》 《Spring-Security-入门(三):密码加密》 《Spring-Security-入门(四):自定义-Filter》 《Spring-Security-入门(五):在 Sprin

Java架构师知识体认识

源码分析 常用设计模式 Proxy代理模式Factory工厂模式Singleton单例模式Delegate委派模式Strategy策略模式Prototype原型模式Template模板模式 Spring5 beans 接口实例化代理Bean操作 Context Ioc容器设计原理及高级特性Aop设计原理Factorybean与Beanfactory Transaction 声明式事物

Java进阶13讲__第12讲_1/2

多线程、线程池 1.  线程概念 1.1  什么是线程 1.2  线程的好处 2.   创建线程的三种方式 注意事项 2.1  继承Thread类 2.1.1 认识  2.1.2  编码实现  package cn.hdc.oop10.Thread;import org.slf4j.Logger;import org.slf4j.LoggerFactory

JAVA智听未来一站式有声阅读平台听书系统小程序源码

智听未来,一站式有声阅读平台听书系统 🌟&nbsp;开篇:遇见未来,从“智听”开始 在这个快节奏的时代,你是否渴望在忙碌的间隙,找到一片属于自己的宁静角落?是否梦想着能随时随地,沉浸在知识的海洋,或是故事的奇幻世界里?今天,就让我带你一起探索“智听未来”——这一站式有声阅读平台听书系统,它正悄悄改变着我们的阅读方式,让未来触手可及! 📚&nbsp;第一站:海量资源,应有尽有 走进“智听

Android实现任意版本设置默认的锁屏壁纸和桌面壁纸(两张壁纸可不一致)

客户有些需求需要设置默认壁纸和锁屏壁纸  在默认情况下 这两个壁纸是相同的  如果需要默认的锁屏壁纸和桌面壁纸不一样 需要额外修改 Android13实现 替换默认桌面壁纸: 将图片文件替换frameworks/base/core/res/res/drawable-nodpi/default_wallpaper.*  (注意不能是bmp格式) 替换默认锁屏壁纸: 将图片资源放入vendo