commit d21de408e5f081c1ef4214e54b6d2acf87dcd919
parent c0225919e977a3e29a46737b348babc3bfeae5b8
Author: rhunk <101876869+rhunk@users.noreply.github.com>
Date:   Sat, 26 Aug 2023 12:05:58 +0200

feat(native): module base, size

Diffstat:
Mcore/src/main/kotlin/me/rhunk/snapenhance/EventDispatcher.kt | 1-
Mcore/src/main/kotlin/me/rhunk/snapenhance/SnapEnhance.kt | 2+-
Mnative/jni/src/library.cpp | 49+++++++++++++++++++++++++++++++------------------
Mnative/jni/src/logger.h | 1+
Anative/jni/src/util.h | 74++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
Mnative/src/main/kotlin/me/rhunk/snapenhance/nativelib/NativeConfig.kt | 3++-
Mnative/src/main/kotlin/me/rhunk/snapenhance/nativelib/NativeLib.kt | 5+++--
7 files changed, 112 insertions(+), 23 deletions(-)

diff --git a/core/src/main/kotlin/me/rhunk/snapenhance/EventDispatcher.kt b/core/src/main/kotlin/me/rhunk/snapenhance/EventDispatcher.kt @@ -106,6 +106,5 @@ class EventDispatcher( if (event.canceled) param.setResult(null) } } - } } \ No newline at end of file diff --git a/core/src/main/kotlin/me/rhunk/snapenhance/SnapEnhance.kt b/core/src/main/kotlin/me/rhunk/snapenhance/SnapEnhance.kt @@ -98,7 +98,7 @@ class SnapEnhance { measureTime { with(appContext) { runCatching { - native.init() + native.initOnce(appContext.androidContext.classLoader) }.onFailure { Logger.xposedLog("Failed to init native", it) return diff --git a/native/jni/src/library.cpp b/native/jni/src/library.cpp @@ -1,17 +1,16 @@ #include <jni.h> #include <string> #include <dobby.h> -#include <sys/stat.h> #include <unistd.h> -#include <fstream> #include <vector> #include "logger.h" #include "config.h" +#include "util.h" static native_config_t *native_config; -static auto fstat_original = fstat; +static auto fstat_original = (int (*)(int, struct stat *)) nullptr; static int fstat_hook(int fd, struct stat *buf) { char name[256]; memset(name, 0, 256); @@ -35,14 +34,37 @@ static int fstat_hook(int fd, struct stat *buf) { } -#define GET_BOOL_FIELD(env, clazz, field) env->GetBooleanField(clazz, env->GetFieldID(clazz, field, "Z")) +extern "C" JNIEXPORT void JNICALL +init(JNIEnv *env, jobject clazz, jobject classloader) { + LOGD("initializing native"); + // config + native_config = new native_config_t; + + // load libclient.so + util::load_library(env, classloader, "client"); + auto client_module = util::get_module("libclient.so"); + if (client_module.base == 0) { + LOGE("libclient not found"); + return; + } + client_module.base -= 0x1000; + LOGD("libclient: offset: 0x%x size: 0x%x", client_module.base, client_module.size); + + // hooks + DobbyHook((void *) DobbySymbolResolver("libc.so", "fstat"), (void *) fstat_hook, + (void **) &fstat_original); + + LOGD("native initialized"); +} + extern "C" JNIEXPORT void JNICALL -loadConfig(JNIEnv *env, jobject clazz, jobject config_object) { - auto native_config_class = env->GetObjectClass(config_object); +loadConfig(JNIEnv *env, jobject clazz, jobject config_object) { + auto native_config_clazz = env->GetObjectClass(config_object); + #define GET_CONFIG_BOOL(name) env->GetBooleanField(config_object, env->GetFieldID(native_config_clazz, name, "Z")) - native_config->disable_bitmoji = GET_BOOL_FIELD(env, native_config_class, "disableBitmoji"); - native_config->disable_metrics = GET_BOOL_FIELD(env, native_config_class, "disableMetrics"); + native_config->disable_bitmoji = GET_CONFIG_BOOL("disableBitmoji"); + native_config->disable_metrics = GET_CONFIG_BOOL("disableMetrics"); LOGD("config loaded"); } @@ -50,18 +72,12 @@ loadConfig(JNIEnv *env, jobject clazz, jobject config_object) { //jni onload extern "C" JNIEXPORT jint JNICALL JNI_OnLoad(JavaVM *vm, void *reserved) { - LOGD("initializing native"); - // config - native_config = new native_config_t; - - // hooks - DobbyHook((void *) fstat_original,(void *) fstat_hook,(void **) &fstat_original); - // register native methods JNIEnv *env = nullptr; vm->GetEnv((void **) &env, JNI_VERSION_1_6); auto methods = std::vector<JNINativeMethod>(); + methods.push_back({"init", "(Ljava/lang/ClassLoader;)V", (void *) init}); methods.push_back({"loadConfig", "(Lme/rhunk/snapenhance/nativelib/NativeConfig;)V", (void *) loadConfig}); env->RegisterNatives( @@ -69,8 +85,5 @@ JNI_OnLoad(JavaVM *vm, void *reserved) { methods.data(), methods.size() ); - - LOGD("native initialized"); - return JNI_VERSION_1_6; } diff --git a/native/jni/src/logger.h b/native/jni/src/logger.h @@ -7,4 +7,5 @@ #define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG, LOG_TAG, __VA_ARGS__) #define LOGI(...) __android_log_print(ANDROID_LOG_INFO, LOG_TAG, __VA_ARGS__) #define LOGW(...) __android_log_print(ANDROID_LOG_WARN, LOG_TAG, __VA_ARGS__) +#define LOGE(...) __android_log_print(ANDROID_LOG_ERROR, LOG_TAG, __VA_ARGS__) diff --git a/native/jni/src/util.h b/native/jni/src/util.h @@ -0,0 +1,73 @@ +#pragma once +#include <unistd.h> +#include <dlfcn.h> +#include <elf.h> + +namespace util { + typedef struct { + uintptr_t base; + size_t size; + } ModuleInfo; + + static void hexDump(void* ptr, uint8_t line_length, uint32_t lines) { + auto* p = (unsigned char*)ptr; + for (uint8_t i = 0; i < lines; i++) { + std::string line; + for (uint8_t j = 0; j < line_length; j++) { + char buf[3]; + sprintf(buf, "%02x", p[i * line_length + j]); + line += buf; + line += " "; + } + LOGI("%s", line.c_str()); + } + } + + static ModuleInfo get_module(const char* libname) + { + char path[256]; + char buff[256]; + int len_libname = strlen(libname); + FILE* file; + uintptr_t addr = 0; + size_t size = 0; + + snprintf(path, sizeof path, "/proc/%d/smaps", getpid()); + file = fopen(path, "rt"); + if (file == NULL) + return {0, 0}; + + while (fgets(buff, sizeof buff, file) != NULL) { + int len = strlen(buff); + if (len > 0 && buff[len-1] == '\n') { + buff[--len] = '\0'; + } + if (len <= len_libname || memcmp(buff + len - len_libname, libname, len_libname)) { + continue; + } + size_t start, end, offset; + char flags[4]; + if (sscanf(buff, "%zx-%zx %c%c%c%c %zx", &start, &end, + &flags[0], &flags[1], &flags[2], &flags[3], &offset) != 7) { + continue; + } + + if (flags[0] != 'r' || flags[2] != 'x') { + continue; + } + addr = start - offset; + size = end - start; + break; + } + fclose(file); + return {addr, size}; + } + + void load_library(JNIEnv* env, jobject classLoader, const char* libName) { + auto runtimeClass = env->FindClass("java/lang/Runtime"); + auto getRuntimeMethod = env->GetStaticMethodID(runtimeClass, "getRuntime", "()Ljava/lang/Runtime;"); + auto runtime = env->CallStaticObjectMethod(runtimeClass, getRuntimeMethod); + auto loadLibraryMethod = env->GetMethodID(runtimeClass, "loadLibrary0", "(Ljava/lang/ClassLoader;Ljava/lang/String;)V"); + env->CallVoidMethod(runtime, loadLibraryMethod, classLoader, env->NewStringUTF(libName)); + } +}+ \ No newline at end of file diff --git a/native/src/main/kotlin/me/rhunk/snapenhance/nativelib/NativeConfig.kt b/native/src/main/kotlin/me/rhunk/snapenhance/nativelib/NativeConfig.kt @@ -3,4 +3,4 @@ package me.rhunk.snapenhance.nativelib data class NativeConfig( val disableBitmoji: Boolean = false, val disableMetrics: Boolean = false -) +)+ \ No newline at end of file diff --git a/native/src/main/kotlin/me/rhunk/snapenhance/nativelib/NativeLib.kt b/native/src/main/kotlin/me/rhunk/snapenhance/nativelib/NativeLib.kt @@ -1,10 +1,11 @@ package me.rhunk.snapenhance.nativelib class NativeLib { - fun init() { + fun initOnce(classloader: ClassLoader) { System.loadLibrary("nativelib") + init(classloader) } - + external fun init(classLoader: ClassLoader) external fun loadConfig(config: NativeConfig) } \ No newline at end of file