PlayIntegrityFix/app/src/main/cpp/main.cpp

261 lines
8.1 KiB
C++
Raw Normal View History

2023-11-20 07:42:21 +08:00
#include <android/log.h>
#include <sys/system_properties.h>
#include <unistd.h>
2023-11-27 01:32:01 +08:00
#include <stdlib.h>
2023-11-20 07:42:21 +08:00
#include "zygisk.hpp"
2023-11-26 17:51:22 +08:00
#include "shadowhook.h"
2023-11-26 00:31:37 +08:00
#include "json.hpp"
2023-11-20 07:42:21 +08:00
2023-11-26 17:51:22 +08:00
#define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG, "PIF/Native", __VA_ARGS__)
2023-11-26 00:31:37 +08:00
#define DEX_FILE_PATH "/data/adb/modules/playintegrityfix/classes.dex"
2023-11-27 00:02:51 +08:00
#define JSON_FILE_PATH "/data/adb/modules/playintegrityfix/pif.json"
2023-11-20 07:42:21 +08:00
2023-11-26 00:31:37 +08:00
static std::string FIRST_API_LEVEL, SECURITY_PATCH;
2023-11-24 03:14:58 +08:00
2023-11-20 07:42:21 +08:00
typedef void (*T_Callback)(void *, const char *, const char *, uint32_t);
2023-11-24 18:52:57 +08:00
static std::map<void *, T_Callback> callbacks;
2023-11-20 07:42:21 +08:00
static void modify_callback(void *cookie, const char *name, const char *value, uint32_t serial) {
2023-11-24 18:52:57 +08:00
if (cookie == nullptr || name == nullptr || value == nullptr ||
!callbacks.contains(cookie))
return;
2023-11-20 07:42:21 +08:00
2023-11-24 18:52:57 +08:00
std::string_view prop(name);
2023-11-20 07:42:21 +08:00
2023-11-23 01:08:47 +08:00
if (prop.ends_with("api_level")) {
2023-11-26 00:31:37 +08:00
if (FIRST_API_LEVEL == "NULL") {
value = nullptr;
} else {
value = FIRST_API_LEVEL.c_str();
}
2023-11-25 09:13:22 +08:00
LOGD("[%s] -> %s", name, value);
2023-11-23 01:08:47 +08:00
} else if (prop.ends_with("security_patch")) {
2023-11-26 00:31:37 +08:00
if (SECURITY_PATCH == "NULL") {
value = nullptr;
} else {
value = SECURITY_PATCH.c_str();
}
2023-11-25 09:13:22 +08:00
LOGD("[%s] -> %s", name, value);
2023-11-23 01:08:47 +08:00
}
2023-11-20 07:42:21 +08:00
2023-11-24 18:52:57 +08:00
return callbacks[cookie](cookie, name, value, serial);
2023-11-20 07:42:21 +08:00
}
static void (*o_system_property_read_callback)(const prop_info *, T_Callback, void *);
static void
my_system_property_read_callback(const prop_info *pi, T_Callback callback, void *cookie) {
if (pi == nullptr || callback == nullptr || cookie == nullptr) {
return o_system_property_read_callback(pi, callback, cookie);
}
2023-11-24 18:52:57 +08:00
callbacks[cookie] = callback;
2023-11-20 07:42:21 +08:00
return o_system_property_read_callback(pi, modify_callback, cookie);
}
2023-11-23 07:07:43 +08:00
static void doHook() {
2023-11-26 17:51:22 +08:00
shadowhook_init(SHADOWHOOK_MODE_UNIQUE, false);
void *handle = shadowhook_hook_sym_name(
"libc.so",
"__system_property_read_callback",
reinterpret_cast<void *>(my_system_property_read_callback),
reinterpret_cast<void **>(&o_system_property_read_callback)
);
2023-11-23 07:07:43 +08:00
if (handle == nullptr) {
LOGD("Couldn't find '__system_property_read_callback' handle. Report to @chiteroman");
return;
2023-11-22 07:32:52 +08:00
}
2023-11-23 07:07:43 +08:00
LOGD("Found '__system_property_read_callback' handle at %p", handle);
2023-11-26 17:51:22 +08:00
}
2023-11-20 07:42:21 +08:00
class PlayIntegrityFix : public zygisk::ModuleBase {
public:
void onLoad(zygisk::Api *api, JNIEnv *env) override {
this->api = api;
this->env = env;
}
void preAppSpecialize(zygisk::AppSpecializeArgs *args) override {
2023-11-25 03:34:31 +08:00
bool isGms = false;
bool isGmsUnstable = false;
2023-11-20 07:42:21 +08:00
auto rawProcess = env->GetStringUTFChars(args->nice_name, nullptr);
2023-11-25 03:34:31 +08:00
if (rawProcess) {
std::string_view process(rawProcess);
isGms = process.starts_with("com.google.android.gms");
isGmsUnstable = process.compare("com.google.android.gms.unstable") == 0;
}
2023-11-20 07:42:21 +08:00
env->ReleaseStringUTFChars(args->nice_name, rawProcess);
2023-11-24 18:52:57 +08:00
if (!isGms) {
api->setOption(zygisk::DLCLOSE_MODULE_LIBRARY);
return;
}
2023-11-24 03:14:58 +08:00
2023-11-24 18:52:57 +08:00
api->setOption(zygisk::FORCE_DENYLIST_UNMOUNT);
2023-11-24 03:14:58 +08:00
2023-11-24 18:52:57 +08:00
if (!isGmsUnstable) {
api->setOption(zygisk::DLCLOSE_MODULE_LIBRARY);
return;
}
2023-11-20 07:42:21 +08:00
2023-11-27 00:02:51 +08:00
long size = 0;
2023-11-24 18:52:57 +08:00
int fd = api->connectCompanion();
2023-11-20 07:42:21 +08:00
2023-11-27 00:02:51 +08:00
read(fd, &size, sizeof(long));
if (size < 1) {
close(fd);
LOGD("Couldn't read from file descriptor 'classes.dex' file!");
api->setOption(zygisk::DLCLOSE_MODULE_LIBRARY);
return;
}
dexVector.resize(size);
read(fd, dexVector.data(), size);
size = 0;
read(fd, &size, sizeof(long));
if (size < 1) {
close(fd);
LOGD("Couldn't read from file descriptor 'pif.json' file!");
api->setOption(zygisk::DLCLOSE_MODULE_LIBRARY);
return;
}
propVector.resize(size);
read(fd, propVector.data(), size);
2023-11-24 03:14:58 +08:00
2023-11-25 03:34:31 +08:00
close(fd);
2023-11-25 09:13:22 +08:00
2023-11-26 17:51:22 +08:00
LOGD("Read from file descriptor file 'classes.dex' -> %d bytes",
static_cast<int>(dexVector.size()));
LOGD("Read from file descriptor file 'pif.json' -> %d bytes",
static_cast<int>(propVector.size()));
2023-11-20 07:42:21 +08:00
}
void postAppSpecialize(const zygisk::AppSpecializeArgs *args) override {
2023-11-26 17:51:22 +08:00
if (dexVector.empty() || propVector.empty()) return;
2023-11-22 01:25:33 +08:00
2023-11-26 00:31:37 +08:00
readJson();
2023-11-25 09:13:22 +08:00
inject();
2023-11-22 07:32:52 +08:00
2023-11-24 18:52:57 +08:00
doHook();
2023-11-26 17:51:22 +08:00
dexVector.clear();
propVector.clear();
2023-11-20 07:42:21 +08:00
}
void preServerSpecialize(zygisk::ServerSpecializeArgs *args) override {
api->setOption(zygisk::DLCLOSE_MODULE_LIBRARY);
}
private:
zygisk::Api *api = nullptr;
JNIEnv *env = nullptr;
2023-11-26 17:51:22 +08:00
std::vector<char> dexVector, propVector;
2023-11-26 00:31:37 +08:00
void readJson() {
2023-11-26 17:51:22 +08:00
std::string data(propVector.cbegin(), propVector.cend());
nlohmann::json json = nlohmann::json::parse(data, nullptr, false, true);
2023-11-26 00:31:37 +08:00
auto getStringFromJson = [&json](const std::string &key) {
return json.contains(key) && !json[key].is_null() ? json[key].get<std::string>()
: "NULL";
};
SECURITY_PATCH = getStringFromJson("SECURITY_PATCH");
FIRST_API_LEVEL = getStringFromJson("FIRST_API_LEVEL");
}
2023-11-20 07:42:21 +08:00
2023-11-23 07:07:43 +08:00
void inject() {
2023-11-20 07:42:21 +08:00
LOGD("get system classloader");
auto clClass = env->FindClass("java/lang/ClassLoader");
auto getSystemClassLoader = env->GetStaticMethodID(clClass, "getSystemClassLoader",
"()Ljava/lang/ClassLoader;");
auto systemClassLoader = env->CallStaticObjectMethod(clClass, getSystemClassLoader);
LOGD("create class loader");
2023-11-25 09:13:22 +08:00
auto dexClClass = env->FindClass("dalvik/system/InMemoryDexClassLoader");
auto dexClInit = env->GetMethodID(dexClClass, "<init>",
"(Ljava/nio/ByteBuffer;Ljava/lang/ClassLoader;)V");
2023-11-26 17:51:22 +08:00
auto buffer = env->NewDirectByteBuffer(dexVector.data(),
static_cast<jlong>(dexVector.size()));
2023-11-25 09:13:22 +08:00
auto dexCl = env->NewObject(dexClClass, dexClInit, buffer, systemClassLoader);
2023-11-20 07:42:21 +08:00
LOGD("load class");
auto loadClass = env->GetMethodID(clClass, "loadClass",
"(Ljava/lang/String;)Ljava/lang/Class;");
auto entryClassName = env->NewStringUTF("es.chiteroman.playintegrityfix.EntryPoint");
auto entryClassObj = env->CallObjectMethod(dexCl, loadClass, entryClassName);
auto entryClass = (jclass) entryClassObj;
2023-11-23 07:07:43 +08:00
2023-11-26 00:31:37 +08:00
LOGD("read json");
auto readProps = env->GetStaticMethodID(entryClass, "readJson",
"(Ljava/lang/String;)V");
2023-11-26 17:51:22 +08:00
std::string data(propVector.cbegin(), propVector.cend());
auto javaStr = env->NewStringUTF(data.c_str());
2023-11-26 00:31:37 +08:00
env->CallStaticVoidMethod(entryClass, readProps, javaStr);
2023-11-23 07:07:43 +08:00
LOGD("call init");
2023-11-20 07:42:21 +08:00
auto entryInit = env->GetStaticMethodID(entryClass, "init", "()V");
env->CallStaticVoidMethod(entryClass, entryInit);
}
};
2023-11-24 03:14:58 +08:00
static void companion(int fd) {
2023-11-27 00:02:51 +08:00
FILE *dex = fopen(DEX_FILE_PATH, "rb");
2023-11-27 01:32:01 +08:00
long dexSize = 0;
char *dexBuffer = nullptr;
long jsonSize = 0;
char *jsonBuffer = nullptr;
2023-11-23 07:07:43 +08:00
2023-11-27 01:32:01 +08:00
if (dex) {
fseek(dex, 0, SEEK_END);
dexSize = ftell(dex);
fseek(dex, 0, SEEK_SET);
2023-11-23 07:07:43 +08:00
2023-11-27 01:32:01 +08:00
dexBuffer = (char*)calloc(1, dexSize);
fread(dexBuffer, 1, dexSize, dex);
2023-11-26 17:51:22 +08:00
2023-11-27 01:32:01 +08:00
fclose(dex);
}
2023-11-26 17:51:22 +08:00
2023-11-27 00:02:51 +08:00
FILE *json = fopen(JSON_FILE_PATH, "r");
2023-11-27 01:32:01 +08:00
if (json) {
fseek(json, 0, SEEK_END);
jsonSize = ftell(json);
fseek(json, 0, SEEK_SET);
2023-11-27 00:02:51 +08:00
2023-11-27 01:32:01 +08:00
jsonBuffer = (char*)calloc(1, jsonSize);
fread(jsonBuffer, 1, jsonSize, json);
2023-11-27 00:02:51 +08:00
2023-11-27 01:32:01 +08:00
fclose(json);
}
2023-11-27 00:02:51 +08:00
write(fd, &dexSize, sizeof(long));
write(fd, dexBuffer, dexSize);
write(fd, &jsonSize, sizeof(long));
write(fd, jsonBuffer, jsonSize);
2023-11-27 01:32:01 +08:00
free(dexBuffer);
free(jsonBuffer);
2023-11-20 07:42:21 +08:00
}
2023-11-27 00:02:51 +08:00
2023-11-20 07:42:21 +08:00
REGISTER_ZYGISK_MODULE(PlayIntegrityFix)
2023-11-27 01:32:01 +08:00
REGISTER_ZYGISK_COMPANION(companion)