Shamrock: アンチシミュレータ検出

This commit is contained in:
WhiteChi 2023-11-26 22:29:11 +08:00
parent 4a4507dfcd
commit b7266e490f
9 changed files with 207 additions and 14 deletions

View File

@ -19,7 +19,7 @@
android:theme="@style/Theme.Shamrock"
android:zygotePreloadName="@string/app_name"
android:multiArch="true"
android:extractNativeLibs="false"
android:extractNativeLibs="true"
tools:targetApi="31">
<activity
android:name=".MainActivity"

View File

@ -6,6 +6,9 @@
# Sets the minimum CMake version required for this project.
cmake_minimum_required(VERSION 3.22.1)
set(CMAKE_CXX_STANDARD 23)
set(CMAKE_CXX_STANDARD_REQUIRED ON)
# Declares the project name. The project name can be accessed via ${ PROJECT_NAME},
# Since this is the top level CMakeLists.txt, the project name is also accessible
# with ${CMAKE_PROJECT_NAME} (both CMake variables are in-sync within the top level

View File

@ -1,19 +1,116 @@
#include <jni.h>
#include <string>
#include <utility>
#include <initializer_list>
#include <vector>
#include <sys/auxv.h>
#include <android/log.h>
#include <dlfcn.h>
#include <string_view>
#include "lsposed.h"
#include "jnihelper.h"
static HookFunType hook_function = nullptr;
static std::vector<std::string> qemu_detect_props = {
"init.svc.qemu-props", "qemu.hw.mainkeys", "qemu.sf.fake_camera", "ro.kernel.android.qemud",
"qemu.sf.lcd_density", "init.svc.qemud", "ro.kernel.qemu",
"libc.debug.malloc"
};
int (*backup_system_property_get)(const char *name, char *value);
int fake_system_property_get(const char *name, char *value) {
for (auto &prop: qemu_detect_props) {
if (strstr(name, prop.c_str())) {
LOGI("[Shamrock] bypass qemu detection");
value[0] = 0;
return 0;
}
}
if (strstr(name, "ro.debuggable")
|| strstr(name, "ro.kernel.qemu.gles")
|| strstr(name, "debug.atrace.tags.enableflags")) {
strcpy(value, "0");
return 1;
}
if (strstr(name, "ro.product.cpu.abilist")) {
int len = backup_system_property_get(name, value);
if (len > 0) {
if (strstr(value, "x86")) {
strcpy(value, "arm64-v8a,armeabi-v7a,armeabi");
return 29;
}
}
return len;
}
if (strstr(name, "ro.hardware")) {
int len = backup_system_property_get(name, value);
if (len > 0) {
if (strstr(value, "generic")
|| strstr(value, "unknown")
|| strstr(value, "emulator")
|| strstr(value, "vbox")
|| strstr(value, "genymotion")
|| strstr(value, "goldfish")) {
strcpy(value, "qcom");
return 4;
}
}
return len;
}
//LOGI("[Shamrock] fake_system_property_get(%s)", name);
return backup_system_property_get(name, value);
}
void on_library_loaded(const char *name, void *handle) {
if (std::string(name) == "libc.so") {
auto libraryName = std::string(name);
if (libraryName.ends_with("libc.so") || libraryName.ends_with("libfekit.so")) {
void *target = dlsym(handle, "__system_property_get");
if (target != nullptr) {
hook_function(target, (void *)fake_system_property_get, (void **) &backup_system_property_get);
} else {
LOGE("[Shamrock] failed to hook __system_property_get");
}
}
}
extern "C" [[gnu::visibility("default")]] [[gnu::used]]
NativeOnModuleLoaded native_init(const NativeAPIEntries *entries) {
jint JNI_OnLoad(JavaVM *jvm, void*) {
JNIHelper::initJavaVM(jvm);
int attach = 0;
JNIEnv *env = JNIHelper::getJNIEnv(&attach);
// do something
LOGI("[Shamrock] JNI_OnLoad NativeModule Init: %p", env);
if (attach == 1) {
JNIHelper::delJNIEnv();
}
//hook_function((void *)env->functions->FindClass, (void *)fake_FindClass, (void **)&backup_FindClass);
return JNI_VERSION_1_6;
}
extern "C" [[gnu::visibility("default")]] [[gnu::used]]
NativeOnModuleLoaded native_init(const NativeAPIEntries *entries) {
hook_function = entries->hook_func;
LOGI("[Shamrock] LSPosed NativeModule Init: %p", hook_function);
return on_library_loaded;
}
extern "C"
JNIEXPORT jboolean JNICALL
Java_moe_fuqiuluo_shamrock_xposed_XposedEntry_00024Companion_injected(JNIEnv *env, jobject thiz) {
return hook_function != nullptr;
}
extern "C"
JNIEXPORT jboolean JNICALL
Java_moe_fuqiuluo_shamrock_xposed_XposedEntry_00024Companion_hasEnv(JNIEnv *env, jobject thiz) {
return JNIHelper::global_jvm != nullptr;
}

View File

@ -0,0 +1,40 @@
#ifndef SHAMROCK_JNIHELPER_H
#define SHAMROCK_JNIHELPER_H
#include "android/log.h"
namespace JNIHelper {
static JavaVM *global_jvm = nullptr;
void initJavaVM(JavaVM *jvm) {
global_jvm = jvm;
}
JNIEnv *getJNIEnv(int *attach) {
if (global_jvm == NULL) return NULL;
*attach = 0;
JNIEnv *jni_env = NULL;
int status = global_jvm->GetEnv((void **)&jni_env, JNI_VERSION_1_6);
if (status == JNI_EDETACHED || jni_env == NULL) {
status = global_jvm->AttachCurrentThread(&jni_env, NULL);
if (status < 0) {
jni_env = NULL;
} else {
*attach = 1;
}
}
return jni_env;
}
jint delJNIEnv() {
if (global_jvm == nullptr) return 0;
return global_jvm->DetachCurrentThread();
}
}
#endif //SHAMROCK_JNIHELPER_H

View File

@ -3,6 +3,13 @@
#include "stdint.h"
#define TAG "LSPosed-Bridge"
#define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG,TAG ,__VA_ARGS__)
#define LOGI(...) __android_log_print(ANDROID_LOG_INFO,TAG ,__VA_ARGS__)
#define LOGW(...) __android_log_print(ANDROID_LOG_WARN,TAG ,__VA_ARGS__)
#define LOGE(...) __android_log_print(ANDROID_LOG_ERROR,TAG ,__VA_ARGS__)
#define LOGF(...) __android_log_print(ANDROID_LOG_FATAL,TAG ,__VA_ARGS__)
typedef int (*HookFunType)(void *func, void *replace, void **backup);
typedef int (*UnhookFunType)(void *func);

View File

@ -51,8 +51,29 @@ internal object LogCenter {
private val format = SimpleDateFormat("[HH:mm:ss] ")
fun log(string: String, level: Level = Level.INFO, toast: Boolean = false) =
log({ string }, level, toast)
fun log(string: String, level: Level = Level.INFO, toast: Boolean = false) {
if (!ShamrockConfig.isDebug() && level == Level.DEBUG) {
return
}
if (toast) {
MobileQQ.getContext().toast(string)
}
// 把日志广播到主进程
GlobalScope.launch(Dispatchers.Default) {
DataRequester.request("send_message", bodyBuilder = {
put("string", string)
put("level", level.id)
})
}
if (!LogFile.exists()) {
LogFile.createNewFile()
}
val format = "%s%s %s\n".format(format.format(Date()), level.name, string)
LogFile.appendText(format)
}
fun log(
string: () -> String,

View File

@ -5,6 +5,8 @@ import de.robv.android.xposed.IXposedHookLoadPackage
import de.robv.android.xposed.XposedBridge
import de.robv.android.xposed.callbacks.XC_LoadPackage
import de.robv.android.xposed.XposedBridge.log
import moe.fuqiuluo.shamrock.helper.Level
import moe.fuqiuluo.shamrock.helper.LogCenter
import moe.fuqiuluo.shamrock.utils.MMKVFetcher
import moe.fuqiuluo.shamrock.xposed.loader.ActionLoader
import moe.fuqiuluo.shamrock.xposed.loader.FuckAMS
@ -12,6 +14,7 @@ import moe.fuqiuluo.shamrock.xposed.loader.LuoClassloader
import moe.fuqiuluo.shamrock.tools.FuzzySearchClass
import moe.fuqiuluo.shamrock.tools.afterHook
import moe.fuqiuluo.shamrock.utils.PlatformUtils
import moe.fuqiuluo.shamrock.xposed.loader.NativeLoader
import mqq.app.MobileQQ
import java.lang.reflect.Field
import java.lang.reflect.Modifier
@ -26,6 +29,12 @@ internal class XposedEntry: IXposedHookLoadPackage {
companion object {
@JvmStatic
var sec_static_stage_inited = false
@JvmStatic
var sec_static_nativehook_inited = false
external fun injected(): Boolean
external fun hasEnv(): Boolean
}
private var firstStageInit = false
@ -108,7 +117,6 @@ internal class XposedEntry: IXposedHookLoadPackage {
if (sec_static_stage_inited) return
val classLoader = ctx.classLoader.also { requireNotNull(it) }
LuoClassloader.hostClassLoader = classLoader
if(injectClassloader(XposedEntry::class.java.classLoader)) {
@ -116,12 +124,7 @@ internal class XposedEntry: IXposedHookLoadPackage {
System.setProperty("qxbot_flag", "1")
} else return
log("Process Name = " + MobileQQ.getMobileQQ().qqProcessName.apply {
// if (!contains("msf", ignoreCase = true)) return // 非MSF进程 退出
//if (contains("peak")) {
// PlatformUtils.killProcess(ctx, this)
//}
})
log("Process Name = " + MobileQQ.getMobileQQ().qqProcessName)
PlatformUtils.isTim()

View File

@ -6,12 +6,15 @@ import android.content.pm.PackageManager
import android.content.pm.VersionedPackage
import android.os.Build
import de.robv.android.xposed.XC_MethodReplacement
import de.robv.android.xposed.XposedBridge
import de.robv.android.xposed.XposedHelpers
import moe.fuqiuluo.shamrock.helper.Level
import moe.fuqiuluo.shamrock.helper.LogCenter
import moe.fuqiuluo.shamrock.remote.service.config.ShamrockConfig
import moe.fuqiuluo.shamrock.tools.hookMethod
import moe.fuqiuluo.shamrock.xposed.XposedEntry
import moe.fuqiuluo.shamrock.xposed.loader.LuoClassloader
import moe.fuqiuluo.shamrock.xposed.loader.NativeLoader
/**
* 反检测
@ -19,6 +22,7 @@ import moe.fuqiuluo.shamrock.xposed.loader.LuoClassloader
class AntiDetection: IAction {
override fun invoke(ctx: Context) {
antiFindPackage(ctx)
antiNativeDetection()
if (ShamrockConfig.isAntiTrace())
antiTrace()
antiMemoryWalking()
@ -35,6 +39,23 @@ class AntiDetection: IAction {
return false
}
private fun antiNativeDetection() {
try {
//System.loadLibrary("clover")
NativeLoader.load("clover")
val env = XposedEntry.hasEnv()
val injected = XposedEntry.injected()
if (!env || !injected) {
LogCenter.log("[Shamrock] Shamrock反检测启动失败(env=$env, injected=$injected)", Level.ERROR)
} else {
XposedEntry.sec_static_nativehook_inited = true
LogCenter.log("[Shamrock] Shamrock反检测启动成功", Level.INFO)
}
} catch (e: Throwable) {
LogCenter.log("[Shamrock] Shamrock反检测启动失败请检查LSPosed版本使用大于100: ${e.message}", Level.ERROR)
}
}
private fun antiFindPackage(context: Context) {
val packageManager = context.packageManager
val applicationInfo = packageManager.getApplicationInfo("moe.fuqiuluo.shamrock", 0)

View File

@ -4,6 +4,7 @@ import android.annotation.SuppressLint
import de.robv.android.xposed.XposedBridge
import moe.fuqiuluo.shamrock.helper.Level
import moe.fuqiuluo.shamrock.helper.LogCenter
import moe.fuqiuluo.shamrock.xposed.XposedEntry
import mqq.app.MobileQQ
import java.io.File
@ -22,7 +23,7 @@ internal object NativeLoader {
@SuppressLint("UnsafeDynamicallyLoadedCode")
fun load(name: String) {
try {
if (name == "shamrock") {
if (name == "shamrock" || name == "clover") {
val context = MobileQQ.getContext()
val packageManager = context.packageManager
val applicationInfo = packageManager.getApplicationInfo("moe.fuqiuluo.shamrock.hided", 0)