mirror of
https://github.com/chiteroman/PlayIntegrityFix.git
synced 2025-04-29 01:22:07 +08:00
update readme
This commit is contained in:
parent
5a86be79f4
commit
4b64bbf864
@ -21,14 +21,13 @@ The purpose of the module is to avoid a hardware attestation.
|
|||||||
If you are failing basicIntegrity (SafetyNet) or MEETS_BASIC_INTEGRITY (Play Integrity) something is wrong in your setup. My recommended steps in order to find the problem:
|
If you are failing basicIntegrity (SafetyNet) or MEETS_BASIC_INTEGRITY (Play Integrity) something is wrong in your setup. My recommended steps in order to find the problem:
|
||||||
- Disable all modules except mine.
|
- Disable all modules except mine.
|
||||||
- Check your SELinux (must be enforced).
|
- Check your SELinux (must be enforced).
|
||||||
- If you are using Shamiko and have deny list disabled, you must manually add GMS main and unstable processes to hide list or DroidGuard will detect root.
|
|
||||||
|
|
||||||
Some modules which modify system can trigger DroidGuard detection, never hook GMS processes.
|
Some modules which modify system can trigger DroidGuard detection, never hook GMS processes.
|
||||||
|
|
||||||
## Certify Play Store and fix Google Wallet
|
## Certify Play Store and fix Google Wallet
|
||||||
Follow this steps:
|
Follow this steps:
|
||||||
- Clear Google Wallet data and cache.
|
- Clear Google Wallet cache.
|
||||||
- Clear Google Play Store data and cache.
|
- Clear Google Play Store cache.
|
||||||
- Clear GSF (com.google.android.gsf) data and cache.
|
- Clear GSF (com.google.android.gsf) data and cache.
|
||||||
- Flash my module in Magisk/KernelSU (if you already have my module, just ignore this step)
|
- Flash my module in Magisk/KernelSU (if you already have my module, just ignore this step)
|
||||||
|
|
||||||
|
@ -4,14 +4,12 @@ project(zygisk)
|
|||||||
|
|
||||||
find_package(cxx REQUIRED CONFIG)
|
find_package(cxx REQUIRED CONFIG)
|
||||||
|
|
||||||
|
link_libraries(cxx::cxx)
|
||||||
|
|
||||||
add_library(${CMAKE_PROJECT_NAME} SHARED main.cpp)
|
add_library(${CMAKE_PROJECT_NAME} SHARED main.cpp)
|
||||||
|
|
||||||
add_subdirectory(Dobby)
|
add_subdirectory(Dobby)
|
||||||
|
|
||||||
SET_OPTION(Plugin.Android.BionicLinkerUtil ON)
|
SET_OPTION(Plugin.Android.BionicLinkerUtil ON)
|
||||||
|
|
||||||
target_link_libraries(dobby cxx::cxx)
|
target_link_libraries(${CMAKE_PROJECT_NAME} log dobby_static)
|
||||||
|
|
||||||
target_link_libraries(dobby_static cxx::cxx)
|
|
||||||
|
|
||||||
target_link_libraries(${CMAKE_PROJECT_NAME} log dobby_static cxx::cxx)
|
|
@ -1,42 +1,43 @@
|
|||||||
#include <cstdlib>
|
#include <cstdlib>
|
||||||
|
#include <cstring>
|
||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
#include <android/log.h>
|
#include <android/log.h>
|
||||||
#include <sys/system_properties.h>
|
#include <sys/system_properties.h>
|
||||||
#include <string_view>
|
|
||||||
#include <map>
|
|
||||||
#include <vector>
|
|
||||||
|
|
||||||
#include "zygisk.hpp"
|
#include "zygisk.hpp"
|
||||||
#include "dobby.h"
|
#include "dobby.h"
|
||||||
|
|
||||||
#define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG, "PIF/Native", __VA_ARGS__)
|
#define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG, "PIF/Native", __VA_ARGS__)
|
||||||
|
|
||||||
typedef void (*T_Callback)(void *, const char *, const char *, uint32_t);
|
#define FIRST_API_LEVEL "25"
|
||||||
|
|
||||||
static std::map<void *, T_Callback> map;
|
static void (*o_callback)(void *, const char *, const char *, uint32_t);
|
||||||
|
|
||||||
static void modify_callback(void *cookie, const char *name, const char *value, uint32_t serial) {
|
static void modify_callback(void *cookie, const char *name, const char *value, uint32_t serial) {
|
||||||
|
|
||||||
if (cookie == nullptr || name == nullptr || value == nullptr || !map.contains(cookie)) return;
|
if (o_callback == nullptr || cookie == nullptr || name == nullptr || value == nullptr) return;
|
||||||
|
|
||||||
std::string_view prop(name);
|
if (strcmp(name, "ro.product.first_api_level") == 0) {
|
||||||
|
LOGD("[%s] Original: %s | Mod: %s", name, value, FIRST_API_LEVEL);
|
||||||
|
return o_callback(cookie, name, FIRST_API_LEVEL, serial);
|
||||||
|
}
|
||||||
|
|
||||||
if (prop.ends_with("api_level")) value = "25";
|
return o_callback(cookie, name, value, serial);
|
||||||
|
|
||||||
if (!prop.starts_with("cache")) LOGD("[%s] -> %s", name, value);
|
|
||||||
|
|
||||||
return map[cookie](cookie, name, value, serial);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void (*o_system_property_read_callback)(const prop_info *, T_Callback, void *);
|
static void (*o_system_property_read_callback)(const prop_info *,
|
||||||
|
void (*)(void *, const char *, const char *,
|
||||||
|
uint32_t), void *);
|
||||||
|
|
||||||
static void
|
static void
|
||||||
my_system_property_read_callback(const prop_info *pi, T_Callback callback, void *cookie) {
|
my_system_property_read_callback(const prop_info *pi,
|
||||||
|
void (*callback)(void *cookie, const char *name, const char *value,
|
||||||
|
uint32_t serial), void *cookie) {
|
||||||
|
|
||||||
if (pi == nullptr || callback == nullptr || cookie == nullptr) {
|
if (pi == nullptr || callback == nullptr || cookie == nullptr) {
|
||||||
return o_system_property_read_callback(pi, callback, cookie);
|
return o_system_property_read_callback(pi, callback, cookie);
|
||||||
}
|
}
|
||||||
map[cookie] = callback;
|
o_callback = callback;
|
||||||
return o_system_property_read_callback(pi, modify_callback, cookie);
|
return o_system_property_read_callback(pi, modify_callback, cookie);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -58,6 +59,12 @@ static void doHook() {
|
|||||||
LOGD("Got __system_property_read_callback handle and hooked it at %p", handle);
|
LOGD("Got __system_property_read_callback handle and hooked it at %p", handle);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static bool dontInject() {
|
||||||
|
char host[PROP_VALUE_MAX];
|
||||||
|
if (__system_property_get("ro.build.host", host) < 1) return false;
|
||||||
|
return strcmp(host, "xiaomi.eu") == 0 || strcmp(host, "EliteDevelopment") == 0;
|
||||||
|
}
|
||||||
|
|
||||||
class PlayIntegrityFix : public zygisk::ModuleBase {
|
class PlayIntegrityFix : public zygisk::ModuleBase {
|
||||||
public:
|
public:
|
||||||
void onLoad(zygisk::Api *api, JNIEnv *env) override {
|
void onLoad(zygisk::Api *api, JNIEnv *env) override {
|
||||||
@ -66,18 +73,20 @@ public:
|
|||||||
}
|
}
|
||||||
|
|
||||||
void preAppSpecialize(zygisk::AppSpecializeArgs *args) override {
|
void preAppSpecialize(zygisk::AppSpecializeArgs *args) override {
|
||||||
auto rawProcess = env->GetStringUTFChars(args->nice_name, nullptr);
|
bool isGms = false;
|
||||||
|
|
||||||
std::string_view process(rawProcess);
|
auto process = env->GetStringUTFChars(args->nice_name, nullptr);
|
||||||
|
|
||||||
bool isGms = process.starts_with("com.google.android.gms");
|
if (process != nullptr) {
|
||||||
bool isGmsUnstable = process.compare("com.google.android.gms.unstable") == 0;
|
isGms = strncmp(process, "com.google.android.gms", 22) == 0;
|
||||||
|
isGmsUnstable = strcmp(process, "com.google.android.gms.unstable") == 0;
|
||||||
|
}
|
||||||
|
|
||||||
env->ReleaseStringUTFChars(args->nice_name, rawProcess);
|
env->ReleaseStringUTFChars(args->nice_name, process);
|
||||||
|
|
||||||
if (isGms) api->setOption(zygisk::FORCE_DENYLIST_UNMOUNT);
|
if (isGms) api->setOption(zygisk::FORCE_DENYLIST_UNMOUNT);
|
||||||
|
|
||||||
if (isGmsUnstable) {
|
if (isGmsUnstable && !dontInject()) {
|
||||||
|
|
||||||
int fd = api->connectCompanion();
|
int fd = api->connectCompanion();
|
||||||
|
|
||||||
@ -86,13 +95,12 @@ public:
|
|||||||
|
|
||||||
if (size > 0) {
|
if (size > 0) {
|
||||||
|
|
||||||
char buffer[size];
|
moduleDex = static_cast<char *>(malloc(size));
|
||||||
read(fd, buffer, size);
|
|
||||||
|
read(fd, moduleDex, size);
|
||||||
|
|
||||||
close(fd);
|
close(fd);
|
||||||
|
|
||||||
moduleDex.insert(moduleDex.end(), buffer, buffer + size);
|
|
||||||
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -103,9 +111,12 @@ public:
|
|||||||
}
|
}
|
||||||
|
|
||||||
void postAppSpecialize(const zygisk::AppSpecializeArgs *args) override {
|
void postAppSpecialize(const zygisk::AppSpecializeArgs *args) override {
|
||||||
if (moduleDex.empty()) return;
|
if (!isGmsUnstable) return;
|
||||||
|
|
||||||
doHook();
|
doHook();
|
||||||
|
|
||||||
|
if (moduleDex == nullptr) return;
|
||||||
|
|
||||||
injectDex();
|
injectDex();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -116,7 +127,8 @@ public:
|
|||||||
private:
|
private:
|
||||||
zygisk::Api *api = nullptr;
|
zygisk::Api *api = nullptr;
|
||||||
JNIEnv *env = nullptr;
|
JNIEnv *env = nullptr;
|
||||||
std::vector<char> moduleDex;
|
bool isGmsUnstable = false;
|
||||||
|
char *moduleDex = nullptr;
|
||||||
|
|
||||||
void injectDex() {
|
void injectDex() {
|
||||||
LOGD("get system classloader");
|
LOGD("get system classloader");
|
||||||
@ -126,7 +138,7 @@ private:
|
|||||||
auto systemClassLoader = env->CallStaticObjectMethod(clClass, getSystemClassLoader);
|
auto systemClassLoader = env->CallStaticObjectMethod(clClass, getSystemClassLoader);
|
||||||
|
|
||||||
LOGD("create buffer");
|
LOGD("create buffer");
|
||||||
auto buf = env->NewDirectByteBuffer(moduleDex.data(), static_cast<jlong>(moduleDex.size()));
|
auto buf = env->NewDirectByteBuffer(moduleDex, sizeof(moduleDex));
|
||||||
LOGD("create class loader");
|
LOGD("create class loader");
|
||||||
auto dexClClass = env->FindClass("dalvik/system/InMemoryDexClassLoader");
|
auto dexClClass = env->FindClass("dalvik/system/InMemoryDexClassLoader");
|
||||||
auto dexClInit = env->GetMethodID(dexClClass, "<init>",
|
auto dexClInit = env->GetMethodID(dexClClass, "<init>",
|
||||||
@ -143,6 +155,9 @@ private:
|
|||||||
auto entryClass = (jclass) entryClassObj;
|
auto entryClass = (jclass) entryClassObj;
|
||||||
auto entryInit = env->GetStaticMethodID(entryClass, "init", "()V");
|
auto entryInit = env->GetStaticMethodID(entryClass, "init", "()V");
|
||||||
env->CallStaticVoidMethod(entryClass, entryInit);
|
env->CallStaticVoidMethod(entryClass, entryInit);
|
||||||
|
|
||||||
|
LOGD("clean");
|
||||||
|
free(moduleDex);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user