使用BreakPad捕获native异常

简介

breakpad是由谷歌提供的跨平台的异常捕获分析工具,目前支持Mac、Linux、Windows平台,主要包括三大部分:

  • client,以library的形式内置在你的应用中,当崩溃发生时写 minidump文件
  • symbol dumper, 读取由编译器生成的调试信息(debugging information),并生成 symbol file
  • processor, 读取 minidump文件 和 symbol file ,生成可读的c/c++ Stack trace.

编译

下载breakpad源码

git colne https://codeload.github.com/google/breakpad/zip/master

编译

./configure
make
make install

编译完成后会生成minidump_stackwalk执行程序,是用来转换dmp文件的

Android中使用breakpad

  1. 把breakpad目录下src下源码到Androidstudio工程的cpp中
  2. 配置cmake
cmake_minimum_required(VERSION 3.4.1)

set(BREAKPAD_ROOT ${CMAKE_CURRENT_SOURCE_DIR})

include_directories(${BREAKPAD_ROOT}/src ${BREAKPAD_ROOT}/src/common/android/include)


file(GLOB BREAKPAD_SOURCES_COMMON
        ${BREAKPAD_ROOT}/src/client/linux/crash_generation/crash_generation_client.cc
        ${BREAKPAD_ROOT}/src/client/linux/dump_writer_common/thread_info.cc
        ${BREAKPAD_ROOT}/src/client/linux/dump_writer_common/ucontext_reader.cc
        ${BREAKPAD_ROOT}/src/client/linux/handler/exception_handler.cc
        ${BREAKPAD_ROOT}/src/client/linux/handler/minidump_descriptor.cc
        ${BREAKPAD_ROOT}/src/client/linux/log/log.cc
        ${BREAKPAD_ROOT}/src/client/linux/microdump_writer/microdump_writer.cc
        ${BREAKPAD_ROOT}/src/client/linux/minidump_writer/linux_dumper.cc
        ${BREAKPAD_ROOT}/src/client/linux/minidump_writer/linux_ptrace_dumper.cc
        ${BREAKPAD_ROOT}/src/client/linux/minidump_writer/minidump_writer.cc
        ${BREAKPAD_ROOT}/src/client/minidump_file_writer.cc
        ${BREAKPAD_ROOT}/src/common/convert_UTF.c
        ${BREAKPAD_ROOT}/src/common/md5.cc
        ${BREAKPAD_ROOT}/src/common/string_conversion.cc
        ${BREAKPAD_ROOT}/src/common/linux/elfutils.cc
        ${BREAKPAD_ROOT}/src/common/linux/file_id.cc
        ${BREAKPAD_ROOT}/src/common/linux/guid_creator.cc
        ${BREAKPAD_ROOT}/src/common/linux/linux_libc_support.cc
        ${BREAKPAD_ROOT}/src/common/linux/memory_mapped_file.cc
        ${BREAKPAD_ROOT}/src/common/linux/safe_readlink.cc

        )

file(GLOB BREAKPAD_ASM_SOURCE ${BREAKPAD_ROOT}/src/common/android/breakpad_getcontext.S
        )

set_source_files_properties(${BREAKPAD_ASM_SOURCE} PROPERTIES LANGUAGE C)

add_library(breakpad STATIC ${BREAKPAD_SOURCES_COMMON} ${BREAKPAD_ASM_SOURCE})

target_link_libraries(breakpad log)

  1. 需要写个初始化方法
JNIEXPORT void JNICALL
Java_com_sample_breakpad_BreakpadInit_initBreakpadNative(JNIEnv *env, jclass type, jstring path_) {
    const char *path = env->GetStringUTFChars(path_, 0);

    google_breakpad::MinidumpDescriptor descriptor(path);
    static google_breakpad::ExceptionHandler eh(descriptor, NULL, DumpCallback, NULL, true, -1);

    env->ReleaseStringUTFChars(path_, path);
}
//crash时回调函数
bool DumpCallback(const google_breakpad::MinidumpDescriptor &descriptor,
                  void *context,
                  bool succeeded) {
    ALOGD("Dump path: %s\n", descriptor.path());
    return succeeded;
}
  1. 写测试程序
//人为写个空指针
void Crash() {
    int *test=NULL;
    *test=1;
}

当native crash时会生成1f1e4c88-6a3c-4827-ee3626b2-4f74ccee.dmp文件,直接打开

&F/system/lib64/libstagefright_omx.soLEpB@/system/framework/ims-common.jarLEpBD/system/framework/qcom.fmradio.jarLEpB&��┙tw���}�_@/system/lib64/libpowermanager.soLEpBF/system/framework/UxPerformance.jarLEpBD/system/framework/QPerformance.jarLEpBX/apex/com.android.runtime/javalib/okhttp.jarLEpB�
�����W��Eҽ�F/system/lib64/libmedia_jni_utils.soLEpBH/system/framework/com.nxp.nfc.nq.jarLEpB�4�Un�������i�=�8/system/lib64/libmediandk.soLEpBF/system/framework/telephony-ext.jarLEpBN/system/framework/android.test.base.jarLEpBB/system/framework/voip-common.jarv/dev/__properties__/u:object_r:exported_fingerprint_prop:s0h/dev/__properties__/u:object_r:persist_debug_prop:s0LEpB��B�6��R�3"k��;`/apex/com.android.runtime/lib64/libartpalette.soLEpB</system/framework/tcmiface.jarl/dev/__properties__/u:object_r:exported_system_prop:s0h/dev/__properties__/u:object_r:exported_vold_prop:s0l/dev/__properties__/u:object_r:exported_config_prop:s0Z/dev/__properties__/u:object_r:dalvik_prop:s0\/dev/__properties__/u:object_r:default_prop:s0l/dev/__properties__/u:object_r:exported_dalvik_prop:s0t/dev/ashmem/c8ecc87e-3327-49f7-b51c-655f1c5e393c (deleted)\/dev/__properties__/u:object_r:log_tag_prop:s0V/dev/__properties__/u:object_r:logd_prop:s0`/dev/__properties__/u:object_r:heapprofd_prop:s0p/dev/__properties__/u:object_r:exported2_default_prop:s0X/dev/__properties__/u:object_r:debug_prop:s0J/dev/__properties__/properties_serialB/dev/__properties__/property_infon/dev/__properties__/u:object_r:exported_default_prop:s0Z/dev/__properties__/u:object_r:system_prop:s0p/dev/__properties__/u:object_r:exported3_default_prop:s0n/dev/__properties__/u:object_r:exported_default_prop:s0X/dev/__properties__/u:object_r:debug_prop:s0J/dev/__properties__/properties_serialB/dev/__properties__/property_infoLEpBP��.(����q4�oa!'�linux-gate.soLEpB gF䊠��]�j���"�L/apex/com.android.runtime/bin/linker64�����`��е
[r��d�  \r��h�}�r��{0��r���jgr���dgr�(���[r@H���[r@h����[rP�N���[rP����[rP���`�[rP�G��[r@��0w[r@(��`g[r@H!�1[r@hd�0![r@�����VrP������q@�=����q@��� �����e��*���,�^��Q�Linux 4.14.117-perf-gd771a6b #1 SMP PREEMPT Mon Dec 9 20:10:54 CST 2019 aarch64Processor  : AArch64 Processor rev 14 (aarch64)
processor   : 0
BogoMIPS    : 38.40
Features    : fp asimd evtstrm aes pmull sha1 sha2 crc32 atomics fphp asimdhp cpuid asimdrdm lrcpc dcpop asimddp
CPU implementer : 0x51
CPU architecture: 8
CPU variant : 0xd
CPU part    : 0x805
CPU revision    : 14

上面的内容无法分析,需要使用上述编译的minidump_stackwalk转换

minidump_stackwalk 目录/1f1e4c88-6a3c-4827-ee3626b2-4f74ccee.dmp >crashlog.txt

打开crashlog.txt

Crash reason:  SIGSEGV /SEGV_MAPERR
Crash address: 0x0
Process uptime: not available

Thread 0 (crashed)
 0  libcrash-lib.so + 0x650
     x0 = 0x0000007267acfb80    x1 = 0x0000007ffef9ef14
     x2 = 0x0000000000000000    x3 = 0x0000007267a69c00
     x4 = 0x0000007ffefa0080    x5 = 0x000000725b4a8965

libcrash-lib.so崩溃的动态库,0x650崩溃地址,需要使用addr2line转换,这里需要注意的是根据你的设备使用动态库的类型选择对应的addr2line,我这里使用的是arm64,所以选择/ndk目录/toolchains/aarch64-linux-android-4.9/prebuilt/darwin-x86_64/bin/aarch64-linux-android-addr2line

 ndk目录/toolchains/aarch64-linux-android-4.9/prebuilt/darwin-x86_64/bin/aarch64-linux-android-addr2line -f -C -e libcrash-lib.so 0x650

打印如下

//出错的函数
Crash()
//出错的源码行数
/Users/lqc/Desktop/demo/Chapter01-master/sample/src/main/cpp/crash.cpp:10
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,922评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,591评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,546评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,467评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,553评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,580评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,588评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,334评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,780评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,092评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,270评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,925评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,573评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,194评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,437评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,154评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,127评论 2 352

推荐阅读更多精彩内容