检查C /本机代码中的.apk-signature

我开发了一个Android应用程序,它还包含用C编写的本机部分(不依赖于应用程序)。

如果共享库不起作用,则应用程序本身是无用的。

我想让native-part(共享库)只做它的工作,如果它已经附带了一个未经修改的应用程序版本(.apk)。

对我来说最好的方法就是这样:

  1. 应用程序已安装
  2. 共享库检查应用程序/ .apk的签名/哈希
  3. 只有在签名与已知签名匹配时才能正常工作

通过这种方式,我想保护我的应用程序免受修改和盗版。

这样做有什么提示吗? 我刚刚在java中找到了检查自己签名的post,但如果可以解开并重新编译应用程序,这就是无聊的。

由于我被要求发布一些代码,我现在正在从C中检查我的Java应用程序的CRC代码,这里有一些片段。

由于性能原因,我无法发布一个完整的工作解决方案,因为它分布在多行,但我希望这是最完整和最有效的:

在MyApplication.java中:

public class MyApplication extends Application { private static Context context; public static Context getAppContext() { return MyApplication.context; } @Override public void onCreate() { super.onCreate(); MyApplication.context = getApplicationContext(); } } 

Android.mk:

 LOCAL_CFLAGS += -O3 -DDEBUG_MODE=0 -DCLASSES_CRC=2331492378 

在你的C代码里面:

 #define LOG_TAG "Your Log Tag" #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 LOGE(...) __android_log_print(ANDROID_LOG_ERROR, LOG_TAG, __VA_ARGS__) #if DEBUG_MODE #define LOGDH(...) LOGD(__VA_ARGS__) #define LOGIH(...) LOGI(__VA_ARGS__) #define LOGEH(...) LOGE(__VA_ARGS__) #else #define LOGDH(...) // #define LOGIH(...) // #define LOGEH(...) // #endif int isSecure = -1; jclass MyApplication; jclass Context; jclass ApplicationInfo; jclass ZipFile; jclass ZipEntry; jclass CheckedInputStream; jclass Adler32; jclass Checksum; jmethodID MyApplication_getAppContextMethodId; jmethodID Context_getApplicationInfoMethodId; jmethodID ZipFile_ConstructorMethodId; jmethodID CheckedInputStream_ConstructorMethodId; jmethodID Adler32_ConstructorMethodId; jmethodID ZipFile_getEntryMethodId; jmethodID ZipFile_getInputStreamMethodId; jmethodID CheckedInputStream_readMethodId; jmethodID CheckedInputStream_getChecksumMethodId; jmethodID Checksum_getValueMethodId; jfieldID ApplicationInfo_flagsFieldId; jfieldID ApplicationInfo_FLAG_DEBUGGABLEFieldId; jfieldID ApplicationInfo_sourceDirFieldId; static long getClassesCRC(JNIEnv *env) { jobject appContextInstance = (*env)->CallStaticObjectMethod(env, MyApplication, MyApplication_getAppContextMethodId); if (!appContextInstance) { LOGEH("Unable to get instance of AppContext"); return false; } jobject applicationInfoInstance = (*env)->CallObjectMethod(env, appContextInstance, Context_getApplicationInfoMethodId); if (!appContextInstance) { LOGEH("Unable to get instance of ApplicationInfo"); return false; } jobject zipFileInstance = (*env)->NewObject(env, ZipFile, ZipFile_ConstructorMethodId, (*env)->GetObjectField(env, applicationInfoInstance, ApplicationInfo_sourceDirFieldId)); if (!zipFileInstance) { LOGEH("Unable to get instance of ZipFile"); return -1; } jstring classesDexString = (*env)->NewStringUTF(env, "classes.dex"); jobject zipEntryInstance = (*env)->CallObjectMethod(env, zipFileInstance, ZipFile_getEntryMethodId, classesDexString); if (!zipFileInstance) { LOGEH("Unable to get instance of ZipEntry"); return -1; } (*env)->DeleteLocalRef(env, classesDexString); jobject adler32Instance = (*env)->NewObject(env, Adler32, Adler32_ConstructorMethodId); if (!adler32Instance) { LOGEH("Unable to get instance of Adler32"); return -1; } jobject inputStreamInstance = (*env)->CallObjectMethod(env, zipFileInstance, ZipFile_getInputStreamMethodId, zipEntryInstance); if (!inputStreamInstance) { LOGEH("Unable to get instance of InputStream"); return -1; } jobject checkedInputStreamInstance = (*env)->NewObject(env, CheckedInputStream, CheckedInputStream_ConstructorMethodId, inputStreamInstance, adler32Instance); if (!checkedInputStreamInstance) { LOGEH("Unable to get instance of CheckedInputStream"); return -1; } int bufferSize = 128; jbyteArray bufferBytes = (*env)->NewByteArray(env, bufferSize); while ((*env)->CallIntMethod(env, checkedInputStreamInstance, CheckedInputStream_readMethodId, bufferBytes) > 0) { } (*env)->DeleteLocalRef(env, bufferBytes); jobject checksumInstance = (*env)->CallObjectMethod(env, checkedInputStreamInstance, CheckedInputStream_getChecksumMethodId); if (!checksumInstance) { LOGEH("Unable to get instance of CheckSum"); return -1; } return (*env)->CallLongMethod(env, checksumInstance, Checksum_getValueMethodId); } static bool isDebuggable(JNIEnv *env) { jobject appContextInstance = (*env)->CallStaticObjectMethod(env, MyApplication, Application_getAppContextMethodId); if (!appContextInstance) { LOGEH("Unable to get instance of AppContext"); return false; } jobject applicationInfoInstance = (*env)->CallObjectMethod(env, appContextInstance, Context_getApplicationInfoMethodId); if (!appContextInstance) { LOGEH("Unable to get instance of ApplicationInfo"); return false; } int FLAG_DEBUGGABLE = (*env)->GetStaticIntField(env, ApplicationInfo, ApplicationInfo_FLAG_DEBUGGABLEFieldId); int flags = (*env)->GetIntField(env, applicationInfoInstance, ApplicationInfo_flagsFieldId); return (0 != (flags &= FLAG_DEBUGGABLE)); } static bool isSecureEnvironment(JNIEnv *env) { //isSecure = true; // TODO remove this if (isSecure == -1) { isSecure = true; if (isDebuggable(env)) { // someone used the app in debug-mode #if DEBUG_MODE != 1 // TODO report #endif LOGEH("App IS DEBUGGABLE!"); isSecure = false; } else { // check CRC long classesCRC = getClassesCRC(env); if (classesCRC != (long) CLASSES_CRC) { #if DEBUG_MODE != 1 // TODO report #endif LOGEH("CRC-CHECK FAILED: %lu", classesCRC); isSecure = false; } } } return isSecure; } static bool initJavaClasses(JNIEnv * env) { jclass local = (*env)->FindClass(env, "eu/my/MyApplication"); MyApplication = (*env)->NewGlobalRef(env, local); if (!MyApplication) { LOGEH("Unable to find the MyApplication class"); return false; } local = (*env)->FindClass(env, "android/content/Context"); Context = (*env)->NewGlobalRef(env, local); (*env)->DeleteLocalRef(env, local); if (!Context) { LOGEH("Unable to find the Context class"); return false; } local = (*env)->FindClass(env, "android/content/pm/ApplicationInfo"); ApplicationInfo = (*env)->NewGlobalRef(env, local); (*env)->DeleteLocalRef(env, local); if (!ApplicationInfo) { LOGEH("Unable to find the ApplicationInfo class"); return false; } local = (*env)->FindClass(env, "java/util/zip/ZipFile"); ZipFile = (*env)->NewGlobalRef(env, local); (*env)->DeleteLocalRef(env, local); if (!ZipFile) { LOGEH("Unable to find the ZipFile class"); return false; } local = (*env)->FindClass(env, "java/util/zip/ZipEntry"); ZipEntry = (*env)->NewGlobalRef(env, local); (*env)->DeleteLocalRef(env, local); if (!ZipEntry) { LOGEH("Unable to find the ZipEntry class"); return false; } local = (*env)->FindClass(env, "java/util/zip/CheckedInputStream"); CheckedInputStream = (*env)->NewGlobalRef(env, local); (*env)->DeleteLocalRef(env, local); if (!CheckedInputStream) { LOGEH("Unable to find the CheckedInputStream class"); return false; } local = (*env)->FindClass(env, "java/util/zip/Adler32"); Adler32 = (*env)->NewGlobalRef(env, local); (*env)->DeleteLocalRef(env, local); if (!Adler32) { LOGEH("Unable to find the Adler32 class"); return false; } local = (*env)->FindClass(env, "java/util/zip/Checksum"); Checksum = (*env)->NewGlobalRef(env, local); (*env)->DeleteLocalRef(env, local); if (!Checksum) { LOGEH("Unable to find the Checksum class"); return false; } return true; } static bool initJavaMethods(JNIEnv * env) { MyApplication_getAppContextMethodId = (*env)->GetStaticMethodID(env, MyApplication, "getAppContext", "()Landroid/content/Context;"); if (!MyApplication_getAppContextMethodId) { LOGEH("Unable to find the getAppContext method"); return false; } Context_getApplicationInfoMethodId = (*env)->GetMethodID(env, Context, "getApplicationInfo", "()Landroid/content/pm/ApplicationInfo;"); if (!Context_getApplicationInfoMethodId) { LOGEH("Unable to find the getApplicationInfo method"); return false; } ZipFile_ConstructorMethodId = (*env)->GetMethodID(env, ZipFile, "", "(Ljava/lang/String;)V"); if (!ZipFile_ConstructorMethodId) { LOGEH("Unable to find the constructor method"); return false; } CheckedInputStream_ConstructorMethodId = (*env)->GetMethodID(env, CheckedInputStream, "", "(Ljava/io/InputStream;Ljava/util/zip/Checksum;)V"); if (!CheckedInputStream_ConstructorMethodId) { LOGEH("Unable to find the constructor method"); return false; } Adler32_ConstructorMethodId = (*env)->GetMethodID(env, Adler32, "", "()V"); if (!Adler32_ConstructorMethodId) { LOGEH("Unable to find the constructor method"); return false; } ZipFile_getEntryMethodId = (*env)->GetMethodID(env, ZipFile, "getEntry", "(Ljava/lang/String;)Ljava/util/zip/ZipEntry;"); if (!ZipFile_getEntryMethodId) { LOGEH("Unable to find the getEntry method"); return false; } ZipFile_getInputStreamMethodId = (*env)->GetMethodID(env, ZipFile, "getInputStream", "(Ljava/util/zip/ZipEntry;)Ljava/io/InputStream;"); if (!ZipFile_getInputStreamMethodId) { LOGEH("Unable to find the getInputStream method"); return false; } CheckedInputStream_readMethodId = (*env)->GetMethodID(env, CheckedInputStream, "read", "([B)I"); if (!CheckedInputStream_readMethodId) { LOGEH("Unable to find the read method"); return false; } CheckedInputStream_getChecksumMethodId = (*env)->GetMethodID(env, CheckedInputStream, "getChecksum", "()Ljava/util/zip/Checksum;"); if (!CheckedInputStream_getChecksumMethodId) { LOGEH("Unable to find the getChecksum method"); return false; } Checksum_getValueMethodId = (*env)->GetMethodID(env, Checksum, "getValue", "()J"); if (!Checksum_getValueMethodId) { LOGEH("Unable to find the getValue method"); return false; } return true; } static bool initJavaFields(JNIEnv * env) { ApplicationInfo_flagsFieldId = (*env)->GetFieldID(env, ApplicationInfo, "flags", "I"); if (!ApplicationInfo_flagsFieldId) { LOGEH("Unable to find the flags field"); return false; } ApplicationInfo_FLAG_DEBUGGABLEFieldId = (*env)->GetStaticFieldID(env, ApplicationInfo, "FLAG_DEBUGGABLE", "I"); if (!ApplicationInfo_FLAG_DEBUGGABLEFieldId) { LOGEH("Unable to get static field FLAG_DEBUGGABLE"); return false; } ApplicationInfo_sourceDirFieldId = (*env)->GetFieldID(env, ApplicationInfo, "sourceDir", "Ljava/lang/String;"); if (!ApplicationInfo_sourceDirFieldId) { LOGEH("Unable to get static field sourceDir"); return false; } return true; } jint JNI_OnLoad(JavaVM* vm, void* reserved) { (void) reserved; // Suppress the warning. JNIEnv * env; if ((*vm)->GetEnv(vm, (void **) &env, JNI_VERSION_1_6) != JNI_OK) { return -1; } if (!initJavaClasses(env)) { return -1; } if (!initJavaMethods(env)) { return -1; } if (!initJavaFields(env)) { return -1; } return JNI_VERSION_1_6; } 

不要忘记将MyApplication的方法添加到Proguard设置中以防止它们被删除!

用法:

  1. 使用-DDEBUG_MODE = 1创建APK
  2. 在Android上启动APK
  3. 读取“CRC-check failed:xxx”中的值
  4. 取该值并将其插入CLASSES_CRC = xxxx
  5. 切换到-DDEBUG_MODE = 0
  6. 重建你的APK
  7. 检查您的应用是否运行良好
  8. 发布

这个方法稍微复杂一点,因为它需要2个构建才能获得有效的APK。 但是,由于在C内部检查CRC而不仅仅是采用了APK文件的虚拟签名,因此该检查几乎是防弹的。

在我的情况下,我想在签名无效的情况下甚至不进行许可机制的设置。

此外,由于此方法仅使用数字而不是字符,因此它已完全编译。

希望这有助于某人!