吾爱破解 - 52pojie.cn

 找回密码
 注册[Register]

QQ登录

只需一步,快速开始

查看: 2277|回复: 16
收起左侧

[Android 原创] 神庙逃亡(Temple Run)IL2CPP 逆向实战:从 APK 到 Frida 实现角色无敌

  [复制链接]
jet58 发表于 2026-1-20 14:56

Unity IL2CPP 逆向实战:从 APK 到 Frida 无敌 Hook 的完整流程

  • Unity IL2CPP 游戏  
  • 无符号 / 无字符串 / 全是 sub_xxx 的 SO  
  • 目标:定位死亡逻辑 / 无敌 / 关键状态变量  

本文以一个跑酷类神庙逃亡(Temple Run) Unity 游戏为例。


游戏截图

在这里插入图片描述

一、准备工具 & 我使用的设备 & 工具版本

  • apktool
  • Il2CppDumper
  • IDA Pro(ARM64)
  • Ubuntu / Linux 环境(本文以 Linux localhost 4.19.95-perf-g79255ac #1 SMP PREEMPT Thu Sep 2 03:29:24 CST 2021 aarch64 为例)
  • Android 10 真机(arm64)
  • Frida版本:17.5.2  
  • Frida Server版本frida-server-17.5.1-android-arm64
  • Python版本:Python 3.12.3

二、解包 APK,提取关键文件

apktool d game.apk -o game_apk

可以用 adb 查看手机 CPU 架构:

adb shell getprop ro.product.cpu.abi

常见输出:

  • arm64-v8a ✅(64 位 ARM)
  • armeabi-v7a ❌(32 位)
  • x86 / x86_64(模拟器或少数设备)

    如果同时存在,可以使用上面的命令查看自己对应的库目录:

    • arm64-v8a
    • armeabi-v7a

    优先选择 arm64-v8a


重点关注两个位置libil2cpp.so和global-metadata.dat这两个文件缺一不可。

1️⃣ libil2cpp.so

game_apk/lib/arm64-v8a/libil2cpp.so

2️⃣ global-metadata.dat

game_apk/assets/bin/Data/Managed/Metadata/global-metadata.dat

三、使用 Il2CppDumper工具还原 IL2CPP 结构

./Il2CppDumper libil2cpp.so global-metadata.dat il2cpp_dump

生成目录结构如下:

il2cpp_dump/
├── dump.cs
├── script.json
├── stringliteral.json
├── DummyDll/
│   ├── Assembly-CSharp.dll
│   └── ...

四、从 dump.cs 找“真正有用的类”

dump.cs 中搜索查看关键信息 游戏核心角色类,例如:

public class CharacterPlayer : CommonPlayer
{
    public bool IsDead;              // 0xD0
    public float TimeSinceDeath;     // 0xD4
    public DeathTypes DeathType;     // 0xD8

    public void Kill(DeathTypes type);
    public void Update();
    public void StartInvcibility(float duration);
}

这一步非常重要:

字段偏移 = 后续 Hook / 内存验证的锚点


五、利用 script.json 精准定位 SO 中的函数

script.json 中可以找到方法对应关系:

{
  "Address": 17287964,
  "Name": "CharacterPlayer$$Kill",
  "Signature": "void CharacterPlayer__Kill (CharacterPlayer_o* __this, int32_t deathType, const MethodInfo* method);"
}

注意:

  • Address十进制
  • 表示 RVA(相对于 libil2cpp.so 基址)

十进制 → 十六进制

17287964 (decimal)
= 0x107CB1C (hex)

六、在 IDA 中精准跳转函数

1️⃣ 用 IDA 打开 libil2cpp.so
2️⃣ 按下键盘 G
3️⃣ 输入:

0x107CB1C

4️⃣ 回车

👉 IDA 会直接跳转到 CharacterPlayer$$Kill 的真实函数实现


七、验证逻辑是否匹配 dump.cs

IDA 反编译中可以看到:

*(_BYTE *)(a1 + 208) = 1;   // IsDead = true
*(_DWORD *)(a1 + 216) = a2;// DeathType

而 dump.cs 中:

public bool IsDead;      // 0xD0 (208)
public DeathTypes DeathType; // 0xD8 (216)

字段偏移完全一致

说明:

  • metadata
  • dump.cs
  • IDA 反编译

三者已经完全对齐。


八、为什么“字符串搜索 / 函数名搜索”基本没用?

在 IL2CPP 游戏中通常存在:

  • SO 全 strip
  • 函数名被抽掉
  • 字符串清空
  • Update / LateUpdate 刷屏

👉 靠猜函数 ≈ 大海捞针

正确思路是:

metadata 给你“地图”,SO 只是实现


九、用 Frida Hook 而不是硬改 SO

目标:让角色始终处于无敌状态。

在script.json中发现:

{
  "Address": 17291184,
  "Name": "CharacterPlayer$$StartInvcibility",
  "Signature": "void CharacterPlayer__StartInvcibility (CharacterPlayer_o* __this, float duration, const MethodInfo* method);",
  "TypeSignature": "vifi"
}

或 dump.cs中发现:

// RVA: 0x107D7B0 Offset: 0x107C7B0 VA: 0x107D7B0
public void StartInvcibility(float duration);

在 script.json 中查到的"Address": 17291184和dump.cs中查到的RVA: 0x107D7B0,两种都可以:

"Address": 0x107D7B0

十、最终 Frida Hook 实现(实战)

在script.json中发现:

{
  "Address": 17285176,
  "Name": "CharacterPlayer$$Update",
  "Signature": "void CharacterPlayer__Update (CharacterPlayer_o* __this, const MethodInfo* method);",
  "TypeSignature": "vii"
}

或 dump.cs中发现:

 // RVA: 0x107C038 Offset: 0x107B038 VA: 0x107C038 Slot: 4
public override void Update();

在 script.json 中查到的"Address": 17285176和dump.cs中查到的RVA: 0x107C038,两种都可以:
函数Update游戏开始后会一直调用,每次调用就调用函数StartInvcibility(player, 9999.0, ptr(0));设置无敌

神庙逃亡(Temple Run)的进程Frida Attach

frida -U -p 19480 -l hook.js

hook.js 脚本(核心代码)

const base = Process.findModuleByName("libil2cpp.so").base;

// CharacterPlayer::StartInvcibility
const StartInvAddr = base.add(0x107D7B0);

// CharacterPlayer::Update
const UpdateAddr = base.add(0x107C038);

const StartInvcibility = new NativeFunction(
    StartInvAddr,
    'void',
    ['pointer', 'float', 'pointer']
);

Interceptor.attach(UpdateAddr, {
    onEnter(args) {
        const player = args[0];
        // 每一帧刷新无敌时间
        StartInvcibility(player, 9999.0, ptr(0));
    }
});

console.log("[+] Invincibility hook installed");

效果说明:

  • 普通碰撞、陷阱:不会死亡
  • 游戏本身设计的“强制死亡”(掉深渊、被追上):仍然会死
    👉 说明这是调用游戏原生无敌逻辑,不是暴力 patch

免费评分

参与人数 5威望 +1 吾爱币 +26 热心值 +5 收起 理由
qtfreet00 + 1 + 20 + 1 感谢发布原创作品,吾爱破解论坛因你更精彩!
imfly + 1 + 1 我很赞同!
allspark + 1 + 1 用心讨论,共获提升!
dryzh + 3 + 1 欢迎分析讨论交流,吾爱破解论坛有你更精彩!
zfzzqlx + 1 + 1 热心回复!

查看全部评分

发帖前要善用论坛搜索功能,那里可能会有你要找的答案或者已经有人发布过相同内容了,请勿重复发帖。

Hmily 发表于 2026-1-21 11:56
图片盗链无法显示,上传到本地编辑下吧。
liyicha 发表于 2026-1-23 09:15
太厉害了,那这样会不会出现问题呀,就比如:跑的时候,前面是悬崖需要转弯但是没有转,会不会卡住。
qq66 发表于 2026-1-21 12:09
dante9394 发表于 2026-1-21 14:06
哈哈哈,这么多年,跑到终点了吗
ClarkWong 发表于 2026-1-21 17:06
没有成品 我不玩
zhanglei1371 发表于 2026-1-21 19:31
不能停下来打怪,不是我喜欢的风格
我喜欢停下来狠狠地揍后面的怪
ChengjinY 发表于 2026-1-22 19:01
哈哈哈,跑到终点了吗
alimbeg 发表于 2026-1-22 23:22
小时候有这个想法,现在可以试试了。
hxddzgf 发表于 2026-1-23 08:57
感觉好久远的游戏啊!多年前玩过
您需要登录后才可以回帖 登录 | 注册[Register]

本版积分规则

返回列表

RSS订阅|小黑屋|处罚记录|联系我们|吾爱破解 - 52pojie.cn ( 京ICP备16042023号 | 京公网安备 11010502030087号 )

GMT+8, 2026-2-7 20:44

Powered by Discuz!

Copyright © 2001-2020, Tencent Cloud.

快速回复 返回顶部 返回列表