Android安全系列之:如何在native層保存關(guān)鍵信息

相信大家在日常開(kāi)發(fā)中都有安全層面的需求券坞,最典型的莫過(guò)于加密琅锻。而apk是脆弱的帮掉,反編譯拿到你的源碼輕而易舉,這時(shí)候我們就需要更保險(xiǎn)的手段來(lái)保存密鑰之類(lèi)的關(guān)鍵信息耿戚。本文就細(xì)致地講解簡(jiǎn)單卻實(shí)用的native手段湿故,文中涉及部分jni的知識(shí),但都有注釋?zhuān)瑴\顯易懂膜蛔,歡迎留言溝通坛猪。文末有示例代碼地址。

目前ndk開(kāi)發(fā)有三種編譯手段:

  1. ndk-build皂股。這是從eclipse時(shí)代就存在的一種編譯方式墅茉,ndk-build是ndk開(kāi)發(fā)包中的一個(gè)可執(zhí)行文件,在這里不贅述呜呐,因?yàn)槟壳癆ndroid Studio已經(jīng)普及就斤,新帶來(lái)的編譯方式十分便捷。
  2. gradle-experimental蘑辑。這是一款A(yù)ndroid Gradle插件洋机,跟我們常用的classpath 'com.android.tools.build:gradle:2.3.0'是同一個(gè)概念的東西,截至寫(xiě)作時(shí)洋魂,已經(jīng)發(fā)展到了0.10.0版本绷旗,以后可能取代現(xiàn)有的gradle插件喜鼓。
  3. CMake。CMake是個(gè)開(kāi)源的跨平臺(tái)的自動(dòng)化構(gòu)建系統(tǒng)衔肢,也是目前Studio默認(rèn)集成的構(gòu)建系統(tǒng)庄岖。CMakeLists.txt的配置這里不詳細(xì)講解了,在創(chuàng)建include c++的新項(xiàng)目時(shí)膀懈,Studio會(huì)幫你做好默認(rèn)配置顿锰。

簡(jiǎn)單的使用jni

首先我們要聲明一個(gè)本地方法,比如是一個(gè)獲取密鑰串的方法启搂,如下:

package com.chenenyu.security;

public class Security {
    static { // 加載libsecurity.so硼控,只要在方法調(diào)用前加載,放哪都行胳赌。
        System.loadLibrary("security");
    }
    public static native String getSecret();
}

這時(shí)候編譯器可能會(huì)警告牢撼,因?yàn)檎也坏綄?duì)應(yīng)jni函數(shù)。我們按照Studio的提示創(chuàng)建一個(gè)function即可疑苫,或者自己手動(dòng)創(chuàng)建源文件和頭文件熏版,這里我們采用靜態(tài)注冊(cè)方式(關(guān)于靜態(tài)注冊(cè)和動(dòng)態(tài)注冊(cè)的區(qū)別,可以google一下)捍掺,對(duì)應(yīng)的頭文件和源文件中的函數(shù)如下:

### .h
#ifdef __cplusplus
extern "C" {
#endif

JNIEXPORT jstring JNICALL
Java_com_chenenyu_security_Security_getSecret(JNIEnv *env, jclass type);

#ifdef __cplusplus
}
#endif
### .cpp
jstring Java_com_chenenyu_security_Security_getSecret(JNIEnv *env, jclass type) {
    return env->NewStringUTF("Security str from native.");
}

這時(shí)我們?cè)陧?xiàng)目中調(diào)用Security.getSecret()就會(huì)得到這個(gè)字符串撼短,這樣看起來(lái)是不是比直接寫(xiě)在Java代碼里安全多了?

然而......并沒(méi)有Mξ稹G帷!

直接使用jni的不足

jni是通過(guò)反射的方式來(lái)相互調(diào)用不瓶,也就是說(shuō)刻坊,我們的native方法是不能混淆的蜒茄,那么就可以反編譯拿到.so庫(kù)和同名的native方法际长,然后通過(guò)二次打包debug出這個(gè)密鑰串馍迄。所以我們需要一種預(yù)防debug的手段,這里我們采取驗(yàn)證apk簽名的方式來(lái)達(dá)到目的麦备,當(dāng)發(fā)現(xiàn)apk簽名和我們自己的簽名不一致的時(shí)候孽椰,調(diào)用so庫(kù)直接崩潰即可。

如何對(duì)so進(jìn)行保護(hù)

Java代碼獲取簽名

首先我們來(lái)看看如何通過(guò)Java代碼獲取簽名信息凛篙,

PackageManager pm = context.getPackageManager();
PackageInfo pi = pm.getPackageInfo(context.getPackageName(), PackageManager.GET_SIGNATURES);
Signature[] signatures = pi.signatures;
Signature signature0 = signatures[0];
signature0.toCharsString();

這里可以發(fā)現(xiàn)獲取簽名需要一個(gè)Context對(duì)象黍匾。

獲取Context對(duì)象

這里我們仿照java代碼獲取簽名的方式,首先我們是否需要傳遞一個(gè)Context對(duì)象到native中呢鞋诗?答案是否定的。因?yàn)?壞人"可以通過(guò)重寫(xiě)Context和PackageManager的方式來(lái)偽造簽名迈嘹。那么不傳Context怎么獲取簽名呢削彬,這里我們可以通過(guò)反射獲取一個(gè)Context:

// 下面幾行代碼展示如何任意獲取Context對(duì)象全庸,在jni中也可以使用這種方式
Class<?> activityThreadClz = Class.forName("android.app.ActivityThread");
Method currentApplication =  activityThreadClz.getMethod("currentApplication");
Application application = (Application) currentApplication.invoke(null);

具體代碼可以參考ActivityThread.java

所以在native中我們也可以通過(guò)這種方式來(lái)獲取Context對(duì)象融痛,相關(guān)代碼如下:

static jobject getApplication(JNIEnv *env) {
    jobject application = NULL;
    jclass activity_thread_clz = env->FindClass("android/app/ActivityThread");
    if (activity_thread_clz != NULL) {
        jmethodID currentApplication = env->GetStaticMethodID(
                activity_thread_clz, "currentApplication", "()Landroid/app/Application;");
        if (currentApplication != NULL) {
            application = env->CallStaticObjectMethod(activity_thread_clz, currentApplication);
        } else {
            LOGE("Cannot find method: currentApplication() in ActivityThread.");
        }
        env->DeleteLocalRef(activity_thread_clz);
    } else {
        LOGE("Cannot find class: android.app.ActivityThread");
    }

    return application;
}

Native代碼獲取簽名

有了Context對(duì)象壶笼,我們就可以通過(guò)native調(diào)用java的方式來(lái)獲取簽名了:

// Application object
jobject application = getApplication(env);
if (application == NULL) {
    return JNI_ERR;
}
// Context(ContextWrapper) class
jclass context_clz = env->GetObjectClass(application);
// getPackageManager()方法
jmethodID getPackageManager = env->GetMethodID(context_clz, "getPackageManager", "()Landroid/content/pm/PackageManager;");
// 獲取PackageManager實(shí)例
jobject package_manager = env->CallObjectMethod(application, getPackageManager);
// PackageManager class
jclass package_manager_clz = env->GetObjectClass(package_manager);
// getPackageInfo()方法
jmethodID getPackageInfo = env->GetMethodID(package_manager_clz, "getPackageInfo", "(Ljava/lang/String;I)Landroid/content/pm/PackageInfo;");
// getPackageName()方法
jmethodID getPackageName = env->GetMethodID(context_clz, "getPackageName", "()Ljava/lang/String;");
// 調(diào)用getPackageName()
jstring package_name = (jstring) (env->CallObjectMethod(application, getPackageName));
// PackageInfo實(shí)例
jobject package_info = env->CallObjectMethod(package_manager, getPackageInfo, package_name, 64);
// PackageInfo class
jclass package_info_clz = env->GetObjectClass(package_info);
// signatures字段
jfieldID signatures_field = env->GetFieldID(package_info_clz, "signatures", "[Landroid/content/pm/Signature;");
jobject signatures = env->GetObjectField(package_info, signatures_field);
jobjectArray signatures_array = (jobjectArray) signatures;
jobject signature0 = env->GetObjectArrayElement(signatures_array, 0);
// Signature class
jclass signature_clz = env->GetObjectClass(signature0);
// toCharsString()方法
jmethodID toCharsString = env->GetMethodID(signature_clz, "toCharsString", "()Ljava/lang/String;");
// 調(diào)用toCharsString()
jstring signature_str = (jstring) (env->CallObjectMethod(signature0, toCharsString));
// 最終的簽名串
const char *sign = env->GetStringUTFChars(signature_str, NULL);

可以看到這個(gè)過(guò)程是很繁瑣的,但是都是class雁刷、object覆劈、method、field等的來(lái)回調(diào)用沛励,沒(méi)什么難點(diǎn)责语。

使用完之后記得要釋放內(nèi)存哦

// release memory
env->DeleteLocalRef(application);
env->DeleteLocalRef(context_clz);
env->DeleteLocalRef(package_manager);
env->DeleteLocalRef(package_manager_clz);
env->DeleteLocalRef(package_name);
env->DeleteLocalRef(package_info);
env->DeleteLocalRef(package_info_clz);
env->DeleteLocalRef(signatures);
env->DeleteLocalRef(signature0);
env->DeleteLocalRef(signature_clz);
...

獲取到簽名之后,要和我們內(nèi)置的簽名串進(jìn)行對(duì)比:

int result = strcmp(sign, "內(nèi)置的簽名串目派,可以通過(guò)上文的Java代碼提前獲取");
env->ReleaseStringUTFChars(signature_str, sign);
env->DeleteLocalRef(signature_str);
if (result == 0) { // 簽名一致
    return JNI_OK;
}
return JNI_ERR;

何時(shí)校驗(yàn)so庫(kù)

前面我們講了怎樣通過(guò)簽名校驗(yàn)so調(diào)用的合法性坤候,但是應(yīng)該在何時(shí)校驗(yàn)?zāi)兀棵看握{(diào)用共享庫(kù)中的方法都校驗(yàn)嗎企蹭?這顯然是不合理的白筹,對(duì)性能也是一種無(wú)端消耗。這里我們要用到JNI_OnLoad()函數(shù)谅摄,該函數(shù)會(huì)在so庫(kù)加載的時(shí)候自動(dòng)調(diào)用徒河,在加載時(shí)我們先驗(yàn)證一下apk的簽名,不一致就直接崩潰送漠,讓“壞人”無(wú)可奈何~

jint JNI_OnLoad(JavaVM *vm, void *reserved) {
    JNIEnv *env = NULL;
    if (vm->GetEnv((void **) &env, JNI_VERSION_1_4) != JNI_OK) {
        return JNI_ERR;
    }
    if (verifySign(env) == JNI_OK) {
        return JNI_VERSION_1_4;
    }
    LOGE("簽名不一致!");
    return JNI_ERR;
}

結(jié)語(yǔ)

至此顽照,一個(gè)簡(jiǎn)單而有效地native安全庫(kù)就完成了。請(qǐng)注意螺男,沒(méi)有絕對(duì)的安全棒厘,我們能做的,就是盡量提高破解難度下隧。光保證客戶(hù)端的安全是沒(méi)有用的奢人,我們還要保證傳輸過(guò)程的安全,比如杜絕明文傳輸淆院,對(duì)關(guān)鍵信息進(jìn)行(非)對(duì)稱(chēng)加密何乎,不要用Base64或者M(jìn)D5這種自欺欺人的方式!還有使用https代替http土辩,這才是保險(xiǎn)的安全手段支救。

最后,貼上文中示例代碼地址 https://github.com/chenenyu/AndroidSecurity 拷淘,歡迎一起交流更安全的保護(hù)手段各墨。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市启涯,隨后出現(xiàn)的幾起案子贬堵,更是在濱河造成了極大的恐慌恃轩,老刑警劉巖,帶你破解...
    沈念sama閱讀 212,383評(píng)論 6 493
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件黎做,死亡現(xiàn)場(chǎng)離奇詭異叉跛,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)蒸殿,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,522評(píng)論 3 385
  • 文/潘曉璐 我一進(jìn)店門(mén)筷厘,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人宏所,你說(shuō)我怎么就攤上這事酥艳。” “怎么了楣铁?”我有些...
    開(kāi)封第一講書(shū)人閱讀 157,852評(píng)論 0 348
  • 文/不壞的土叔 我叫張陵玖雁,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我盖腕,道長(zhǎng)赫冬,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 56,621評(píng)論 1 284
  • 正文 為了忘掉前任溃列,我火速辦了婚禮劲厌,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘听隐。我一直安慰自己补鼻,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,741評(píng)論 6 386
  • 文/花漫 我一把揭開(kāi)白布雅任。 她就那樣靜靜地躺著风范,像睡著了一般。 火紅的嫁衣襯著肌膚如雪沪么。 梳的紋絲不亂的頭發(fā)上硼婿,一...
    開(kāi)封第一講書(shū)人閱讀 49,929評(píng)論 1 290
  • 那天,我揣著相機(jī)與錄音禽车,去河邊找鬼寇漫。 笑死,一個(gè)胖子當(dāng)著我的面吹牛殉摔,可吹牛的內(nèi)容都是我干的州胳。 我是一名探鬼主播,決...
    沈念sama閱讀 39,076評(píng)論 3 410
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼逸月,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼栓撞!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起碗硬,我...
    開(kāi)封第一講書(shū)人閱讀 37,803評(píng)論 0 268
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤瓤湘,失蹤者是張志新(化名)和其女友劉穎捌归,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體岭粤,經(jīng)...
    沈念sama閱讀 44,265評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,582評(píng)論 2 327
  • 正文 我和宋清朗相戀三年特笋,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了剃浇。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,716評(píng)論 1 341
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡猎物,死狀恐怖虎囚,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情蔫磨,我是刑警寧澤淘讥,帶...
    沈念sama閱讀 34,395評(píng)論 4 333
  • 正文 年R本政府宣布,位于F島的核電站堤如,受9級(jí)特大地震影響蒲列,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜搀罢,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 40,039評(píng)論 3 316
  • 文/蒙蒙 一蝗岖、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧榔至,春花似錦抵赢、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,798評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至枫弟,卻和暖如春邢享,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背媒区。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 32,027評(píng)論 1 266
  • 我被黑心中介騙來(lái)泰國(guó)打工驼仪, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人袜漩。 一個(gè)月前我還...
    沈念sama閱讀 46,488評(píng)論 2 361
  • 正文 我出身青樓绪爸,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親宙攻。 傳聞我的和親對(duì)象是個(gè)殘疾皇子奠货,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,612評(píng)論 2 350

推薦閱讀更多精彩內(nèi)容