免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 14756 | 回复: 4
打印 上一主题 下一主题

转贴 - 如何在 Linux 下调试动态链接库 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2006-11-17 10:56 |只看该作者 |倒序浏览
大家都知道在 Linux 可以用 gdb 来调试应用程序,当然前提是用 gcc 编译程序时要加上 -g 参数。
我这篇文章里将讨论一下用 gdb 来调试动态链接库的问题。
首先,假设我们准备这样的一个动态链接库:
库名称是: ggg
动态链接库文件名是: libggg.so
头文件是: get.h
提供这样两个函数调用接口:
    int get ();
    int set (int a);


要生成这样一个动态链接库,我们首先编写这样一个头文件:

  1. /************关于本文档********************************************
  2. *filename: get.h
  3. *purpose:  一个动态链接库头文件示例
  4. *tided by: zhoulifa(zhoulifa@163.com) 周立发 (http://zhoulifa.9999mb.com)
  5. Linux 爱好者 Linux 知识传播者 SOHO 族 开发者 最擅长 C 语言
  6. *date time: 2006-11-15 21:11:54
  7. *Note: 任何人可以任意复制代码并运用这些文档,当然包括你的商业用途
  8. * 但请遵循 GPL
  9. *Hope:希望越来越多的人贡献自己的力量,为科学技术发展出力
  10. * 科技站在巨人的肩膀上进步更快!感谢有开源前辈的贡献!
  11. *感谢 vcclass@hotmail.com 提供原始代码,我在他的基础上整理了此文
  12. *********************************************************************/
  13. int get ();
  14. int set (int a);
复制代码


然后准备这样一个生成动态链接库的源文件:

  1. /************关于本文档********************************************
  2. *filename:  get.cpp
  3. *purpose: 一个动态链接库源文件示例
  4. *tided by: zhoulifa(zhoulifa@163.com) 周立发 (http://zhoulifa.9999mb.com)
  5. Linux 爱好者 Linux 知识传播者 SOHO 族 开发者 最擅长 C 语言
  6. *date time:2006-11-15 21:11:54
  7. *Note: 任何人可以任意复制代码并运用这些文档,当然包括你的商业用途
  8. * 但请遵循 GPL
  9. *Hope:希望越来越多的人贡献自己的力量,为科学技术发展出力
  10. * 科技站在巨人的肩膀上进步更快!感谢有开源前辈的贡献!
  11. *感谢 vcclass@hotmail.com 提供原始代码,我在他的基础上整理了此文
  12. *********************************************************************/
  13. #include <stdio.h>

  14. #include "get.h"



  15. static int x=0;

  16. int get ()

  17. {

  18.         printf ("get x=%d\n", x);

  19.         return x;

  20. }

  21. int set (int a)

  22. {

  23.         printf ("set a=%d\n", a);

  24.         x = a;

  25.         return x;

  26. }

复制代码


然后我们用 GNU 的 C/C++ 编译器来生成动态链接库,编译命令如下:
g++ get.cpp -shared -g -DDEBUG -o libggg.so

这样我们就准备好了动态链接库了,下面我们编写一个应用程序来调用此动态链接库,源代码如下:

  1. /************关于本文档********************************************
  2. *filename: pk.cpp
  3. *purpose: 一个调用动态链接库的示例
  4. *tided by: zhoulifa(zhoulifa@163.com) 周立发 (http://zhoulifa.9999mb.com)
  5. Linux 爱好者 Linux 知识传播者 SOHO 族 开发者 最擅长 C 语言
  6. *date time:2006-11-15 21:11:54
  7. *Note: 任何人可以任意复制代码并运用这些文档,当然包括你的商业用途
  8. * 但请遵循 GPL
  9. *Hope:希望越来越多的人贡献自己的力量,为科学技术发展出力
  10. * 科技站在巨人的肩膀上进步更快!感谢有开源前辈的贡献!
  11. *感谢 vcclass@hotmail.com 提供原始代码,我在他的基础上整理了此文
  12. *********************************************************************/
  13. #include <stdio.h>

  14. #include "get.h"

  15. int main (int argc, char** argv)

  16. {

  17.         int a = 100;

  18.         int b = get ();

  19.         int c = set (a);

  20.         int d = get ();



  21.         printf ("a=%d,b=%d,c=%d,d=%d\n",a,b,c,d);

  22.         return 0;

  23. }

复制代码

编译此程序用下列命令,如果已经把上面生成的 libggg.so 放到了库文件搜索路径指定的文件目录,比如 /lib 或 /usr/lib 之类的,就用下面这条命令:
g++ pk.cpp -o app -Wall -g -lggg

否则就用下面这条命令:
g++ pk.cpp -o app -Wall -g -lggg -L`pwd`


下面我们就开始调试上面命令生成的 app 程序吧。如果已经把上面生成的 libggg.so 放到了库文件搜索路径指定的文件目录,比如 /lib 或 /usr/lib 之类的,调试就顺利完成,如下:
zhoulifa@linux#gdb ./app
GNU gdb 6.4-debian
Copyright 2005 Free Software Foundation, Inc.
GDB is free software, covered by the GNU General Public License, and you are
welcome to change it and/or distribute copies of it under certain conditions.
Type "show copying" to see the conditions.
There is absolutely no warranty for GDB.  Type "show warranty" for details.
This GDB was configured as "i486-linux-gnu"...Using host libthread_db library "/lib/tls/i686/cmov/libthread_db.so.1".

(gdb) b main    /* 这是在程序的 main 处设置断点 */
Breakpoint 1 at 0x804853c: file pk.cpp, line 7.
(gdb) b set      /* 这是在程序的 set 处设置断点 */
Function "set" not defined.
Make breakpoint pending on future shared library load? (y or [n]) y /* 这里必须选择 y 调试程序才会跟踪到动态链接库内部去 */
Breakpoint 2 (set) pending.
(gdb) run /* 开始运行我们的程序,直到遇见断点时暂停 */
Starting program: /data/example/c/app
Breakpoint 3 at 0xb7f665f8: file get.cpp, line 11.
Pending breakpoint "set" resolved

Breakpoint 1, main (argc=1, argv=0xbf990504) at pk.cpp:7
7               int a = 100;
(gdb) n     /* 继续执行程序的下一行代码 */
8               int b = get ();
(gdb) n      /* 程序执行到了我们断点所在的动态链接库了 */
get x=0
9               int c = set (a);
(gdb) n

Breakpoint 3, set (a=100) at get.cpp:11
11              printf ("set a=%d\n", a);
(gdb) list   /* 查看当前代码行周围的代码,证明我们已经跟踪到动态链接库的源代码里面了 */
6               printf ("get x=%d\n", x);
7               return x;
8       }
9       int set (int a)
10      {
11              printf ("set a=%d\n", a);
12              x = a;
13              return x;
14      }
(gdb) n
set a=100
12              x = a;
(gdb) n
13              return x;
(gdb) n
14      }
(gdb) n
main (argc=1, argv=0xbf990504) at pk.cpp:10
10              int d = get ();
(gdb) n
get x=100
11              printf ("a=%d,b=%d,c=%d,d=%d\n",a,b,c,d);
(gdb) n
a=100,b=0,c=100,d=100
12              return 0;
(gdb) c
Continuing.

Program exited normally.
(gdb) quit  /* 程序顺利执行结束 */
zhoulifa@linux#


如果我们没有把动态链接库放到指定目录,比如/lib里面,调试就会失败,过程如下:
zhoulifa@linux# gdb ./app
GNU gdb 6.4-debian
Copyright 2005 Free Software Foundation, Inc.
GDB is free software, covered by the GNU General Public License, and you are
welcome to change it and/or distribute copies of it under certain conditions.
Type "show copying" to see the conditions.
There is absolutely no warranty for GDB.  Type "show warranty" for details.
This GDB was configured as "i486-linux-gnu"...Using host libthread_db library "/lib/tls/i686/cmov/libthread_db.so.1".

(gdb) b main
Breakpoint 1 at 0x804853c: file pk.cpp, line 7.
(gdb) b set
Function "set" not defined.
Make breakpoint pending on future shared library load? (y or [n]) y
Breakpoint 2 (set) pending.
(gdb) run  /* 虽然调试操作都一样,但程序执行失败 */
Starting program: /data/example/c/app
/data/example/c/app: error while loading shared libraries: libggg.so: cannot open shared object file: No such file or directory

Program exited with code 0177.
(gdb) quit
zhoulifa@linux#


本次实验的环境是:
CPU:AMD Athlon(tm) 64 Processor 3000+
内存:512M
OS:Ubuntu GNU/Linux 6.06 dapper LTS
gcc:gcc 版本 4.0.3 (Ubuntu 4.0.3-1ubuntu5)

此贴转自 周立发的 Linux 论坛 http://zhoulifa.9999mb.com/bbs

论坛徽章:
0
2 [报告]
发表于 2006-11-17 15:48 |只看该作者
原帖由 mycuname 于 2006-11-17 10:56 发表
大家都知道在 Linux 可以用 gdb 来调试应用程序,当然前提是用 gcc 编译程序时要加上 -g 参数。
我这篇文章里将讨论一下用 gdb 来调试动态链接库的问题。
首先,假设我们准备这样的一个动态链接库:


要生成 ...

>>
>>
>>好贴啊!建议版主收进精华。支持一下!
>>
>>

论坛徽章:
0
3 [报告]
发表于 2006-11-17 18:06 |只看该作者
Linux下不叫动态连接库,叫共享对象了,share object
《深入理解计算机系统》
第7章

[ 本帖最后由 1jjk 于 2006-11-17 18:30 编辑 ]

论坛徽章:
0
4 [报告]
发表于 2006-11-17 18:22 |只看该作者
好贴,学习~

论坛徽章:
0
5 [报告]
发表于 2007-04-03 21:25 |只看该作者
/cygdrive/c/DOCUME~1/ADMIN~1/Temp/ccM5QDLH.o:/cpram/pk.cpp:24: undefined reference to 'get()'
/cpram/pk.cpp:26: undefined reference to 'set(int)'
/cpram/pk.cpp:28: undefined reference to 'get()'
collect2: ld returned 1 exit status


我按着上面说的做,但是报错,我是用Cygwin在Windows下写的,好像是没有动态链接库。我有把文件拷到/lib下。
不知道有没有哪高手指一下。
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP