通过Xposed+Substrate实现非侵入性的Unity代码注入

别看标题这么厉害... 其实就是通过 hook 的方式实现在运行时替换 Unity 游戏中的 Assembly-CSharp.dll 来实现代码注入的功能。

为什么要做到这个运行时替换呢?原因如下:

  • 目标游戏安装包有一个多G,每次重新打包和安装耗时将十分巨大。
  • 目标游戏是付费游戏,所以应该在底层对签名做了校验。即使没有也要以防万一(迷惑行为

步入正题。

使用 Xposed 使目标游戏执行我们的代码

Xposed 就不用说了... 很常见的一个工具。为了实现标题的功能,我们只需要简单地 hook 掉 ContextWrapper 的 attachBaseContext 方法,在目标游戏启动时根据拿到的 Context 去得到我们自己的插件的 apk 路径,然后再用 System.load 去加载我们自己的库

需要注意的是,System.load 的源码如下:

@CallerSensitive
public static void load(String filename) {
    Runtime.getRuntime().load0(Reflection.getCallerClass(), filename);
}

也就是说 System.load 是根据调用者的 Class 来获取用于搜索动态库的 ClassLoader 的,而 Xposed 在加载我们的插件类时并没有根据 Context.getPackageContext 之类来创建 ClassLoader ,而是!直接!用 PathClassLoader!!因此我们插件类的 ClassLoader 里面并不会有 nativeLibraries 的搜索路径(一般是在 /data/data/包名/lib)。因此!不能用 System.loadLibrary!!!

也正因为如此,我们如果直接加载我们的动态库,系统将无法自动找到 libsubstrate.so,所以我们也需要把 libsubstrate.so 的加载放在 Java 层上来,并且要放在加载我们自己的动态库之前

下面是可爱的加载部分的代码w:

val nativeLibDir = context.packageManager.getApplicationInfo(CYMOE_PACKAGE_NAME, 0).nativeLibraryDir
File(nativeLibDir, "libsubstrate.so").absolutePath.run {
    Log.v(T, "Loading Substrate library, path: $this")
    System.load(this)
}
File(nativeLibDir, "libxxx.so").absolutePath.run {
    Log.v(T, "Loading 我的 library, path: $this")
    System.load(this)
}

完毕!接下来就来看看我们的 native 部分吧~

替换对方mono库的加载函数

众所周知,Unity 加载游戏代码是通过 mono 库(以前是 libmono.so,现在我这个游戏里面是 libmonobdwgc-2.0.so)来加载游戏中的 assets/bin/Data/Managed/Assembly-CSharp.dll 的。所以我们只需要 hook 掉 mono 库的对应方法就好啦~

问题来了:我们怎么获取到 mono 库的句柄?

我这边的方法是直接 hook 掉 dlopen 方法,在加载 mono 库时 hook,因为不知道为什么自己加载 mono 库和游戏真正加载的句柄不同... 按理来说一个进程打开的同一个动态库句柄应该是不变的。如果有知道的可以在下面的评论区给其他读者和我答疑解惑。

如何在 Android 高版本 hook dlopen 方法可以看 这篇文章

问题又来了,mono 库加载 dll 的函数是谁呢?我最初尝试 hook fopen 并打印堆栈,看读取 apk 安装包的有哪些调用栈,不过实在是有太多了... 很难分析。没办法,最后偷了个懒,在 这里 找到了 mono 加载库的函数,也就是 mono_image_open_from_data_with_name ,我们只需要对传入这个函数的参数进行修改再调用原函数就好了。

注:由于我是把更改后的 Assembly-CSharp.dll 放在我自己软件的安装包里,所以还需要 Java 层额外向本地传一个 AAssetManager 来读取我自己软件的 Asset,大家也可以用其他方法,这里就不赘述。

上代码:

char assmbly_replacement[PATH_MAX];
AAsset *asset;

void* (*mono_image_open_from_data_with_name_old)(char *data, guint32 data_len, gboolean need_copy, void* status, gboolean refonly, const char *name);
void* (*__loader_dlopen_old)(const char* filename, int flags, const void* caller_addr);

void* mono_image_open_from_data_with_name_fake(char *data, guint32 data_len, gboolean need_copy, void* status, gboolean refonly, const char *name) {
    LOGI("image open: %s", name);
    if (endsWith(name, "Assembly-CSharp.dll")) {
        LOGI("Got Assembly-CSharp, replacing it");
        do {
            if (asset==nullptr) {
                LOGE("Asset is null");
                break;
            }
            off_t len = AAsset_getLength(asset);
            if (len<0) {
                LOGE("Length < 0");
                break;
            }
            LOGI("File length: %lu\n", len);
            char *file_data = new char[len];
            if (AAsset_read(asset, file_data, len)<0) {
                LOGE("Failed to read");
                delete[] file_data;
                break;
            }
            name = assmbly_replacement;
            data = file_data;
            data_len = len;
            LOGI("Replaced successfully");
        } while (false);
    }
    return mono_image_open_from_data_with_name_old(data, data_len, need_copy, status, refonly, name);
}

void* __loader_dlopen_fake(const char* filename, int flags, const void* caller_addr) {
    void *handle = __loader_dlopen_old(filename, flags, caller_addr);
    LOGI("dlopen: %s %d %p", filename, flags, handle);
    if (endsWith(filename, "libmonobdwgc-2.0.so")) {
        LOGI("Got you! libmono!");
        void *mono_image_open_from_data_with_name = dlsym(handle, "mono_image_open_from_data_with_name");
        HOOK_FUNCTION_DYNAMIC(mono_image_open_from_data_with_name);
    }
    return handle;
}

extern "C" JNIEXPORT JNICALL void Java_你_的_包名_类名_nativeSetReplacement(ARG_STATIC, jobject assetManagerObj, jstring path) {
    if (path==nullptr) return;
    AAssetManager *asset_manager = AAssetManager_fromJava(env, assetManagerObj);
    asset = AAssetManager_open(AAssetManager_fromJava(env, assetManagerObj), "Assembly-CSharp.dll", AASSET_MODE_STREAMING);
    const char* chars  = env->GetStringUTFChars(path, NULL);
    strncpy(assmbly_replacement, chars, PATH_MAX);
    env->ReleaseStringUTFChars(path, chars);
}

顺带一提,上面代码的 assembly_replacement 也是需要 Java 层传过来。通过日志可以发现 mono_image_open_from_data_with_name 的 name 参数是 安装包路径/assets/bin/Data/..... 的形式,所以我们也需要(不一定?)把 name 改成我们自己的路径。最后再把 data 参数和 data_len 参数改一下就好。(p.s. 最初我忘了改 data_len 导致出了一堆问题...)

Java 层设置的函数如下:(说是 Java 实则是 Kotlin)

// attachBaseContext时
nativeSetReplacement(
    context.createPackageContext(你自己的包名, 0).assets,
    context.packageManager.getApplicationInfo(你自己的包名, 0).sourceDir
            + "/assets/Assembly-CSharp.dll"
)
.....
// native函数定义
@JvmStatic
external fun nativeSetReplacement(assetManager: AssetManager, path: String)

这样就整好了w。

结语

没有结语。总之是项目还没做完,就先不放在 GitHub 上面啦w

有兴趣或问题的可以联系我,Q:250851048

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 202,723评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,080评论 2 379
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,604评论 0 335
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,440评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,431评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,499评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,893评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,541评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,751评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,547评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,619评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,320评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,890评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,896评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,137评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,796评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,335评论 2 342

推荐阅读更多精彩内容

  • 转自长亭知乎专栏,实习时小姐姐的约稿,已经不在那边了所以版权不归我哈 笔者一直自认玩过不少游戏,无奈水平太菜,日常...
    hyrathon阅读 1,583评论 0 0
  • 前项目的C#热更方案 小甜甜的C#热更方案 前段时间 noodle 说他把 小甜甜 项目中他做的 C#热更方案 开...
    恶毒的狗阅读 2,524评论 0 0
  • Unity3D打包android应用程序时,如果不对DLL加密,很容易被反编译,导致代码的泄露。通常的做法是通过加...
    某人在阅读 2,170评论 0 2
  • 前段时间编译了一下Unity的Mono,看了很多相关的文章,也遇到很多新坑。所以来总结一下,加深自己对Mono的理...
    李嘉的博客阅读 3,007评论 0 3
  • 现在是晚上11点,我躺在床上怎么也睡不着,翻了翻微博,感觉没意思,突然就想打开简书把名字改了,头像也换了。...
    海珍的猪阅读 252评论 0 0