Android端線上NativeCrash收集的兩種方法(上)

什么是NativeCrash

   Android端crash可分為Java crash和Native crash什荣,我們通常說的crash一般指的是Java層crash篮奄,Native crash主要指C/C++代碼(其在Android工程中以動態(tài)鏈接庫的形式存在)的崩潰鳄梅,一般難以抓取策治。下圖為Android系統(tǒng)框架圖拐云,NativeCrash主要就是在圖中紅框部分發(fā)生的崩潰田炭。
Android系統(tǒng)框架圖

Android中C/C++開發(fā)部分稱之為NDK查吊,Android開發(fā)中引入NDK一般是基于如下考慮:
數據安全:java層代碼易被反編譯谐区,而C/C++庫反匯難度較大。
性能考慮:將要求高性能的應用邏輯使用C開發(fā)逻卖,提高應用程序的執(zhí)行效率卢佣。
便于移植:用C/C++寫得庫可以方便在其他的嵌入式平臺上再次使用

google-breakpad工作原理

   對于NativeCrash,其復現(xiàn)難度大箭阶,且需要手機root權限(Native崩潰日志存儲在手機的/data/tombstones目錄下)虚茶。本文主要介紹兩種方法獲取Android端線上Native Crash的崩潰信息,分別是基于開源工具google-breakpad和基于c/c++信號異常處理仇参。
   首先介紹在開源工具google-breakpad的基礎上嘹叫,實時抓取線上Native crash的dmp日志,并對dmp日志進行上傳诈乒、解析罩扇、分類、聚合的過程怕磨。下圖為google-breakpad的工作原理圖喂饥。Breakpad有三個主要組件:
google-breakpad工作原理
  • Client 需要包含到應用程序中,它可以獲取當前線程的狀態(tài)和當前加載的可執(zhí)行文件和共享庫的ID寫轉儲文件肠鲫,控制程序發(fā)送Native崩潰時员帮,產生minidump文件。
  • Symbol dumper讀取由編譯器產生的調試信息导饲,并生成一個使用Breakpad格式的符號文件捞高。
  • minidump processor 讀取minidump文件,根據相應的符號文件渣锦,解析產生了一個可讀的C/C++堆棧硝岗。

關于google-breakpad的工作原理可詳見如下鏈接
http://blog.csdn.net/wpc320/article/details/8290501

線上native crash收集實現(xiàn)方案

NativeCrash收集實現(xiàn)方案

上圖給出NativeCrash收集的整體實現(xiàn)方案。APP中接入提供的SDK袋毙,包含一個通用SO和一個JAR,當APP中發(fā)生NDK崩潰時型檀,會在手機端生成一個dmp文件,待下次APP重啟后听盖,將此dmp文件上傳至服務端胀溺,在服務端進行解析裂七、分類、聚合月幌、可讀展示等過程碍讯⌒危總的來說扯躺,可分為三個模塊。

? 客戶端sdk
客戶端sdk主要包含一個通用so和jar蝎困,通用so主要是基于google-breakpad進行簡單封裝录语,使其通用化,jar主要提供上傳dmp文件的功能禾乘。通用so采用NDK編程澎埠,主要代碼片段如下:

JNI_OnLoad

jint JNI_OnLoad(JavaVM* vm, void* reserved) {

    JNIEnv* env = NULL;

    jint result = -1;

    if (vm->GetEnv((void**) &env, JNI_VERSION_1_4) != JNI_OK) {

        goto bail;

    }

    if (register_nativecrash(env) < 0) {

        goto bail;

    }

    fields.jvm = vm;

    result = JNI_VERSION_1_4;

    bail: return result;

}

register_nativecrash:注冊native方法

static int register_nativecrash(JNIEnv *env) {

    return registerNativeMethods(env, CLASSPATH, gMethods,sizeof(gMethods) / sizeof(gMethods[0]));

}

static int registerNativeMethods(JNIEnv* env, const char* className,JNINativeMethod* gMethods, int numMethods) {

    jclass clazz;

    clazz = env->FindClass(className);

    if (clazz == NULL) {

        return JNI_FALSE;

    }

    if (env->RegisterNatives(clazz, gMethods, numMethods) < 0) {

        return JNI_FALSE;

    }

    fields.clazz = (jclass) env->NewGlobalRef(clazz);

    if (fields.clazz == NULL) {

        return JNI_FALSE;

    }

    return JNI_TRUE;

}

設置dmp文件存放目錄

#include "client/linux/handler/exception_handler.h"
#include "client/linux/handler/minidump_descriptor.h"

static void Java_com_baidu_nativecrash_NativeCrash_setNativeCrashDir(JNIEnv* env, jobject thiz, jstring dir) {

    if (dir == NULL) {

        __android_log_print(ANDROID_LOG_ERROR, LOG_TAG,"native crash dir is null");

        jniThrowException(env, "java/lang/IllegalArgumentException", NULL);

        return;
    }

    const char *filepathStr = env->GetStringUTFChars(dir, NULL);
    
    if (filepathStr == NULL) {  // Out of memory

        __android_log_print(ANDROID_LOG_ERROR, LOG_TAG,"native crash filepathStr is null");
       
       jniThrowException(env, "java/lang/IllegalArgumentException", NULL);
        return;

    }

    if (filepathStr && (strlen(filepathStr) > 0)) {

        __android_log_print(ANDROID_LOG_DEBUG, LOG_TAG, "native crash %s",filepathStr);

        static google_breakpad::MinidumpDescriptor descriptor(filepathStr);

        static google_breakpad::ExceptionHandler eh(descriptor, NULL,

        DumpCallback, NULL, true, -1);

    }
}

DumpCallback回調方法中可以向Jave層回傳一些其他額外信息。

android.mk

MY_ROOT_PATH := $(call my-dir)

include $(MY_ROOT_PATH)/../google-breakpad/android/jni/Android.mk

LOCAL_PATH := $(MY_ROOT_PATH)

include $(CLEAR_VARS)

LOCAL_MODULE    := nativecrash

LOCAL_SRC_FILES := nativecrash_jni.cpp

LOCAL_C_INCLUDES := \

$(LOCAL_PATH)/../google-breakpad/src/common/android/include \

$(LOCAL_PATH)/../google-breakpad/src

LOCAL_CPPFLAGS :=  -Os -fvisibility=hidden

LOCAL_CFLAGS :=  -Os -fvisibility=hidden

LOCAL_CFLAGS += -Wno-psabi

LOCAL_STATIC_LIBRARIES += breakpad_client

include $(BUILD_SHARED_LIBRARY)

Application.mk

APP_ABI := armeabi-v7a armeabi mips x86

APP_STL := stlport_static

APP_OPTIM := release

APP_CPPFLAGS += -Wno-error=format-security

? native日志上傳
客戶端dmp文件上傳的服務端采用百度云服務始藕,主要用來接收客戶端上傳的dmp文件并保存蒲稳。注意,這里dmp文件上傳時機為APP重啟后伍派,在wifi的情況下上傳(dmp文件較大江耀,非wifi上傳不合理)。接收客戶端上傳的dmp文件并存儲后寫庫诉植,數據庫中需要兩個表祥国,分別是用來保存dmp文件的存儲記錄表以及dmp文件解析后的日志分類表。

? 日志解析分類

  1. 解析過程:每隔3min讀取日志存儲記錄表最近30條未解析記錄晾腔,進行解析舌稀,dmp解析需要用到breakpad的minidump processor提供的解析方法,故需事先編譯breakpad源碼灼擂,生成解析所需tool壁查,解析完成后,修改解析狀態(tài)剔应。解析腳本需要輪詢執(zhí)行潮罪,通過linux下crontab進行設置。解析過程參考如下鏈接领斥。
    http://blog.csdn.net/brook0344/article/details/20126351

  2. 分類過程:每隔3min讀取存儲記錄表最近20條已解析記錄嫉到,進行分類。分類算法主要是將解析后的崩潰堆棧信息的棧頂文本作為關鍵字月洛,進行分類何恶,可大大簡化文本分類過程。我們可以認為在同一處崩潰的堆棧信息是一致的嚼黔,也即可以認為棧頂文本相同的崩潰细层,是同一類崩潰惜辑。分類完成后,寫入日志分類表疫赎。

  3. 日志展示:讀取日志分類表盛撑,按分類的崩潰次數逆序排序,展示結果捧搞。

    以上是基于google-breakpad的NativeCrash日志收集方法的全過程抵卫,google-breakpad是夸平臺開源工具,體量較大胎撇,在其基礎上生成的通用SO和dmp日志也都較大介粘,對于sdk大小有嚴格要求的APP,可能不是很方便晚树。下一節(jié)介紹體量較小的基于c/c++異常信號處理的NativeCrash日志收集方法姻采。
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市爵憎,隨后出現(xiàn)的幾起案子慨亲,更是在濱河造成了極大的恐慌,老刑警劉巖宝鼓,帶你破解...
    沈念sama閱讀 216,544評論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件刑棵,死亡現(xiàn)場離奇詭異,居然都是意外死亡席函,警方通過查閱死者的電腦和手機铐望,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,430評論 3 392
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來茂附,“玉大人正蛙,你說我怎么就攤上這事∮” “怎么了乒验?”我有些...
    開封第一講書人閱讀 162,764評論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長蒂阱。 經常有香客問我锻全,道長,這世上最難降的妖魔是什么录煤? 我笑而不...
    開封第一講書人閱讀 58,193評論 1 292
  • 正文 為了忘掉前任鳄厌,我火速辦了婚禮,結果婚禮上妈踊,老公的妹妹穿的比我還像新娘了嚎。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 67,216評論 6 388
  • 文/花漫 我一把揭開白布歪泳。 她就那樣靜靜地躺著萝勤,像睡著了一般。 火紅的嫁衣襯著肌膚如雪呐伞。 梳的紋絲不亂的頭發(fā)上敌卓,一...
    開封第一講書人閱讀 51,182評論 1 299
  • 那天,我揣著相機與錄音伶氢,去河邊找鬼趟径。 笑死,一個胖子當著我的面吹牛鞍历,可吹牛的內容都是我干的舵抹。 我是一名探鬼主播肪虎,決...
    沈念sama閱讀 40,063評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼劣砍,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了扇救?” 一聲冷哼從身側響起刑枝,我...
    開封第一講書人閱讀 38,917評論 0 274
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎迅腔,沒想到半個月后装畅,有當地人在樹林里發(fā)現(xiàn)了一具尸體,經...
    沈念sama閱讀 45,329評論 1 310
  • 正文 獨居荒郊野嶺守林人離奇死亡沧烈,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 37,543評論 2 332
  • 正文 我和宋清朗相戀三年掠兄,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片锌雀。...
    茶點故事閱讀 39,722評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡蚂夕,死狀恐怖,靈堂內的尸體忽然破棺而出腋逆,到底是詐尸還是另有隱情婿牍,我是刑警寧澤,帶...
    沈念sama閱讀 35,425評論 5 343
  • 正文 年R本政府宣布惩歉,位于F島的核電站等脂,受9級特大地震影響,放射性物質發(fā)生泄漏撑蚌。R本人自食惡果不足惜上遥,卻給世界環(huán)境...
    茶點故事閱讀 41,019評論 3 326
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望争涌。 院中可真熱鬧粉楚,春花似錦、人聲如沸第煮。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,671評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至撵摆,卻和暖如春底靠,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背特铝。 一陣腳步聲響...
    開封第一講書人閱讀 32,825評論 1 269
  • 我被黑心中介騙來泰國打工暑中, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人鲫剿。 一個月前我還...
    沈念sama閱讀 47,729評論 2 368
  • 正文 我出身青樓鳄逾,卻偏偏與公主長得像,于是被迫代替她去往敵國和親灵莲。 傳聞我的和親對象是個殘疾皇子雕凹,可洞房花燭夜當晚...
    茶點故事閱讀 44,614評論 2 353

推薦閱讀更多精彩內容