AddressSanitizer 是一种内存错误检测工具,编译时添加 -fsanitize=address
选项可以在运行时检测出非法内存访问,当发生段错误时,AddressSanitizer 会输出详细的错误报告,包括出错位置的代码行号和调用栈,有助于快速定位问题。
demo.c段错误程序
#include <stdio.h>
#include <assert.h>
void buggy_function(int *ptr) {
// 确保指针不是 NULL
//assert(ptr != NULL);
*ptr = 10;
}
int main() {
int *ptr = NULL;
*ptr = 10;
return 0;
}
gcc编译
gcc -g -fsanitize=address demo.c
可以在运行程序时通过设置 ASAN_OPTIONS
环境变量,将 AddressSanitizer 的输出重定向到文件。具体命令如下:
export ASAN_OPTIONS="log_path=asan_log.txt"
./a.out
生成日志如下:
bwton@DESKTOP-UJNM808:~/project/tmp$ cat asan_log.txt.302
=================================================================
==302==ERROR: AddressSanitizer: SEGV on unknown address 0x000000000000 (pc 0x560783400895 bp 0x7ffdf421abd0 sp 0x7ffdf421abc0 T0)
==302==The signal is caused by a WRITE memory access.
==302==Hint: address points to the zero page.
#0 0x560783400894 in main /home/bwton/project/tmp/demo.c:12
#1 0x7ff6d0eb4c86 in __libc_start_main (/lib/x86_64-linux-gnu/libc.so.6+0x21c86)
#2 0x560783400719 in _start (/home/bwton/project/tmp/a.out+0x719)
AddressSanitizer can not provide additional info.
SUMMARY: AddressSanitizer: SEGV /home/bwton/project/tmp/demo.c:12 in main
==302==ABORTING
bwton@DESKTOP-UJNM808:~/project/tmp$
Eclipse编译
如果是嵌入式设备使用Eclipse编译,可如下设置:
1. 添加编译选项
-
在编译器的设置中,找到
Miscellaneous
(杂项)部分。 -
在
Other flags
(其他标志)框中添加-fsanitize=address
。 -
如果你使用的是 C++ 编译器,也可以在
GCC C++ Compiler
下重复相同的步骤。
2. 修改链接器设置
为了确保 AddressSanitizer 正常工作,还需要在链接器中添加相应的选项:
-
在同样的
Tool Settings
下,选择GCC C Linker
。 -
在
Miscellaneous
部分的Other flags
中添加-fsanitize=address
。 -
如果运行时报错"error while loading shared libraries: libasan.so.1:"可以在编译时添加静态链接-static-libasan