记录一次NDK开发中崩溃信号signal 11 (SIGSEGV), code 1 (SEGV_MAPERR) 的解决方法

一.前言

在NDK开发的过程中,如果遇到应用Crash的情况,JNI层的报错信息并不像java层那样可以直接看到错误信息和错误的位置,如果要想定位的到错误的位置,需要借助NDK的一些工具进行地址转换,本博文用addr2line(NDK的工具)进行调试

 

二.分析

  以下是我的报错信息

记录一次NDK开发中崩溃信号signal 11 (SIGSEGV), code 1 (SEGV_MAPERR) 的解决方法_第1张图片

 

读取其中的关键信息:

1.错误代号:signal 11 (SIGSEGV), code 1 (SEGV_MAPERR),一般都是空指针错误

2.错误发生所在的so文件:liblive-lib.so

3.错误发生的方法名:ZN12VideoEncoder11encodeFrameEPciiS0_Pi和Java_com_lib_1live_LiveNativeManager_encoderVideoEncode,前者为编译过的方法名

4.错误发生的内存地址:00026877和0002547f,不能直接看到源代码发生错误的位置

 

三.利用addr2line定位错误

 

1.首先找到addr2line的位置,我的addr2line位置为D:\Android\SDK\ndk-bundle\toolchains\arm-linux-androideabi-4.9\prebuilt\windows-x86_64\bin>arm-linux-androideabi-addr2line,然后在Android studio的Terminal中定位到这个位置

 

2.找到对应so文件的位置,上面发生错误的so文件liblive-lib.so的位置为E:\AndroidProjects\EPlayer\lib_live\build\intermediates\cmake\debug\obj\armeabi-v7a\liblive-lib.so

 

3. 然后在Terminal中定位D:\Android\SDK\ndk-bundle\toolchains\arm-linux-androideabi-4.9\prebuilt\windows-x86_64\bin>基础上输入:arm-linux-androideabi-addr2line -C -f -e E:\AndroidProjects\EPlayer\lib_live\build\intermediates\cmake\debug\obj\armeabi-v7a\liblive-lib.so 00026877 0002547f,最后的00026877 0002547f 分别为发生错误的内存地址,用空格隔开,图示如下:

 

4. 其中-C -f :表示打印错误行数所在的函数名称,-e:表示打印错误地址的对应路径及行数

 

5. 最后结果如下

在这里就明确显示出错误发生在源代码的哪一行了,通过代码分析就可以正确地解决问题了

 

四.后记

 通过上面的定位,我的这次错误发生在fwrite(outBytes, 1, size, out1)函数中,我第一反应就是手机的文件写入权限没允许,果然是,最后解决了问题

 

你可能感兴趣的:(Android)