Linux下查看函數(shù)被那些函數(shù)調(diào)用過(guò)?
一、問(wèn)題
有個(gè)打印log的函數(shù),想知道該函數(shù)執(zhí)行的時(shí)候,之前執(zhí)行了哪些函數(shù)?

二、分析
在應(yīng)用程序打印函數(shù)棧需要通過(guò)函數(shù)backtrace(),該函數(shù)對(duì)應(yīng)頭文件如下:
- #include <execinfo.h>
1、三個(gè)與打印調(diào)用棧相關(guān)的函數(shù)
打印函數(shù)棧需要使用到以下3個(gè)函數(shù)
- int backtrace(void** buffer, int size);
函數(shù)功能:用于獲取當(dāng)前線程的調(diào)用堆棧。參數(shù):buffer:它是一個(gè)指針數(shù)組,函數(shù)獲取的當(dāng)前線程的調(diào)用堆棧將會(huì)被存放在buffer中。在buffer中的指針實(shí)際是從堆棧中獲取的返回地址,每一個(gè)堆棧 框架有一個(gè)返回地址。size:用來(lái)指定buffer中可以保存多少個(gè)void*元素。返回值:實(shí)際獲取的指針個(gè)數(shù),最大不超過(guò)size大小。
char** backtrace_symbols (void *const *buffer, int size);
函數(shù)功能:將從backtrace函數(shù)獲取的信息轉(zhuǎn)化為一個(gè)字符串?dāng)?shù)組。參數(shù):buffer:從backtrace函數(shù)獲取的數(shù)組指針。size:是該數(shù)組中的元素個(gè)數(shù)(backtrace函數(shù)的返回值)。返回值:是一個(gè)指向字符串?dāng)?shù)組的指針,它的大小同buffer相同。每個(gè)字符串包含了一個(gè)相對(duì)于buffer中對(duì)應(yīng)元素的 可打印信息。它包括函數(shù)名,函數(shù)的偏移地址,和實(shí)際的返回地址。
注:
- 1、只有使用ELF二進(jìn)制格式的程序才能獲取函數(shù)名稱(chēng)和偏移地址。在其他系統(tǒng),只有16進(jìn)制的返回地址能被獲取。另外,需要傳遞相應(yīng)的標(biāo)志給鏈接器,以能支持函數(shù)名功能即編譯選項(xiàng)-rdynamic。
- 2、backtrace_symbols生成的字符串都是malloc出來(lái)的,最后需要free該塊內(nèi)存。
- void backtrace_symbols_fd (void *const *buffer, int size, int fd)
功能:backtrace_symbols_fd與backtrace_symbols函數(shù)具有相同的功能,不同的是它不會(huì)給調(diào)用者返回字符串?dāng)?shù)組,而是將結(jié)果寫(xiě)入文件描述符為fd的文件中,每個(gè)函數(shù)對(duì)應(yīng)一行.它不需要調(diào)用malloc函數(shù),因此適用于有可能調(diào)用該函數(shù)會(huì)失敗的情況。參數(shù):fd:通常填寫(xiě)STDOUT_FILENO
2. 鏈接庫(kù)
在編譯的時(shí)候需要加上**-rdynamic**選項(xiàng)。
- -rdynamic
- Pass the flag -export-dynamic to the ELF linker, on targets that support it. This instructs the linker to add all symbols, not only used ones, to the dynamic symbol table. This option is needed for some uses of "dlopen" or to allow obtaining backtraces from within a program.
該選項(xiàng)讓鏈接器將所有符號(hào)添加到動(dòng)態(tài)符號(hào)表中,這樣才能將函數(shù)地址翻譯成函數(shù)名,否則打印的結(jié)果是不會(huì)打印函數(shù)名的。
另外,這個(gè)選項(xiàng)不會(huì)處理static函數(shù),所以,static函數(shù)的符號(hào)無(wú)法得到。
3. 舉例
- #include <execinfo.h>
- #include <stdio.h>
- #include <stdlib.h>
- #include <unistd.h>
- void fun1();
- void fun2();
- void fun3();
- void print_stacktrace();
- void print_stacktrace()
- {
- int size = 16;
- void * array[100];
- int stack_num = backtrace(array, size);
- char ** stacktrace = backtrace_symbols(array, stack_num);
- backtrace_symbols_fd(array,size,STDOUT_FILENO);
- #if 0
- char ** stacktrace = backtrace_symbols(array, stack_num);
- for (int i = 0; i < stack_num; ++i)
- {
- printf("%s\n", stacktrace[i]);
- }
- free(stacktrace);
- #endif
- }
- void fun1()
- {
- printf("stackstrace begin:\n");
- print_stacktrace();
- }
- void fun2()
- {
- fun1();
- }
- void fun3()
- {
- fun2();
- }
- int main()
- {
- fun3();
- }
編譯運(yùn)行g(shù)cc編譯時(shí)加上-rdynamic參數(shù),通知鏈接器支持函數(shù)名功能(不加-rdynamic參數(shù)則無(wú)函數(shù)名打印):
- gcc 123.c -o run -rdynamic -g
執(zhí)行結(jié)果:

4. 補(bǔ)充 address2line
同一個(gè)函數(shù)可以在代碼中多個(gè)地方調(diào)用,如果我們只是知道函數(shù),要想知道在哪里調(diào)用了該函數(shù),可以通過(guò)address2line命令來(lái)完成,我們用第2步中編譯出來(lái)的test2來(lái)做實(shí)驗(yàn)(address2line的-f選項(xiàng)可以打出函數(shù)名, -C選項(xiàng)也可以demangle):
address2line
三、內(nèi)核代碼中如何打印函數(shù)棧?
在Linux內(nèi)核中提供了一個(gè)可以打印出內(nèi)核調(diào)用堆棧的函數(shù) dump_stack()。
該函數(shù)在我們調(diào)試內(nèi)核的過(guò)程中可以打印出函數(shù)調(diào)用關(guān)系,該函數(shù)可以幫助我們進(jìn)行內(nèi)核調(diào)試,以及讓我們了解內(nèi)核的調(diào)用關(guān)系。
1. 頭文件
該函數(shù)頭文件為:
- #include <asm/ptrace.h>
使用方式:
直接在想要查看的函數(shù)中添加
- dump_stack();
2. 舉例
測(cè)試代碼如下:hello.c
- 1 #include <linux/init.h>
- 2 #include <linux/module.h>
- 3 #include <asm/ptrace.h>
- 4
- 6 MODULE_LICENSE("GPL");
- 7 MODULE_AUTHOR("PD");
- 8 void aaa(int a);
- 9 void bbb(int b);
- 10 void ccc(int c);
- 11
- 14 void ccc(int c)
- 15 {
- 16 printk(KERN_SOH"cccc \n");
- 17 dump_stack();
- 18 printk("c is %d\n",c);
- 19 }
- 20 void bbb(int b)
- 21 {
- 22 int c = b + 10;
- 23 printk(KERN_SOH"bbbb \n");
- 24 ccc(c);
- 25 }
- 26 void aaa(int a)
- 27 {
- 28 int b = a + 10;
- 29 printk(KERN_SOH"aaaa \n");
- 30 bbb(b);
- 31 }
- 32
- 34 static int hello_init(void)
- 35 {
- 36 int a = 10;
- 37
- 38 aaa(a);
- 39 printk(KERN_SOH"hello_init \n");
- 40
- 41 return 0;
- 42 }
- 43 static void hello_exit(void)
- 44 {
- 45 printk("hello_exit \n");
- 46 return;
- 47 }
- 48
- 49 module_init(hello_init); //insmod
- 50 module_exit(hello_exit);//rmmod
Makefile
- ifneq ($(KERNELRELEASE),)
- obj-m:=hello.o
- else
- KDIR :=/lib/modules/$(shell uname -r)/build
- PWD :=$(shell pwd)
- all:
- make -C $(KDIR) M=$(PWD) modules
- clean:
- rm -f *.ko *.o *.mod.o *.symvers *.cmd *.mod.c *.order
- endif
編譯安裝模塊
- dmesg -c
- make
- insmod hello.ko
【注意】 都在root權(quán)限下操作
結(jié)果
可以看到在函數(shù)ccc中使用dump_stack()打印出了ccc的函數(shù)調(diào)用棧。
在內(nèi)核開(kāi)發(fā)中,我們可以使用dump_stack()來(lái)打印相關(guān)信息,同時(shí)在內(nèi)核源碼學(xué)習(xí)中也可以用來(lái)了解函數(shù)調(diào)用關(guān)系。