您的位置:首页 > 运维架构 > Linux

linux中动态加载动态库的方法

2017-07-10 18:38 323 查看
1:创建test.h, test.c文件

[cpp] view
plain copy

 print?

//test.h  

#ifndef TEST_H_  

#define TEST_H_  

  

#include <stdio.h>  

  

void PrintHello();  

int Add(int a, int b);  

  

#endif  

[cpp] view
plain copy

 print?

//test.cpp  

#include "test.h"  

  

//输出文本hello, world  

void Hello()  

{  

    printf("hello, world\n");  

}  

  

//返回两个参数的和  

int Add(int a, int b)  

{  

    return a + b;  

}  

2:将其编译成动态库

[cpp] view
plain copy

 print?

gcc test.c -shared -fPIC -o libtest.so  

3:创建主文件main.c

[cpp] view
plain copy

 print?

//main.c  

#include <stdio.h>  

#include <stdlib.h>  

#include <dlfcn.h>  

#include <signal.h>  

#include <errno.h>  

  

//输出错误信息并退出        

void error_quit(const char *str)        

{        

    fprintf(stderr, "%s\n", str);     

    exit(1);        

}    

  

int main(int argc, char *argv [])  

{  

    void *plib;                     //指向so文件的指针  

    typedef void (*FUN_HELLO)();  

    typedef int (*FUN_ADD)(int, int);  

    FUN_HELLO funHello = NULL;      //函数指针  

    FUN_ADD funAdd = NULL;  

  

    //打开so文件  

    //为了方便演示,我将库文件和可执行文件放在同一个目录下  

    plib = dlopen("./libtest.so", RTLD_NOW | RTLD_GLOBAL);  

    if( NULL == plib )  

        error_quit("Can't open the libtest.so");  

  

    //加载函数void Hello()  

    funHello = dlsym(plib, "Hello");  

    if( NULL == funHello )  

        error_quit("Can't load function 'Hello'");  

      

    //加载函数int Add(int a, int b)  

    funAdd = dlsym(plib, "Add");  

    if( NULL == funAdd )  

        error_quit("Can't load function 'Add'");  

  

    //调用成功加载的函数  

    funHello();  

    printf("5 + 8 = %d\n", funAdd(5, 8));  

  

    //关闭so文件  

    dlclose(plib);  

  

    return 0;  

}  

4:编译,运行

[cpp] view
plain copy

 print?

gcc main.c -o main -ldl  

./main  

完成了,呵呵

/////////////////////////////////////////////////////////////

  dlopen()是一个强大的库函数。该函数将打开一个新库,并把它装入内存。该函数主要用来加载库中的符号,这些符号在编译的时候是不知道的。比如 Apache Web 服务器利用这个函数在运行过程中加载模块,这为它提供了额外的能力。一个配置文件控制了加载模块的过程。这种机制使得在系统中添加或者删除一个模块时,都不需要重新编译  可以在自己的程序中使用
dlopen()。dlopen() 在 dlfcn.h 中定义,并在 dl 库中实现。它需要两个参数:一个文件名和一个标志。文件名可以是我们学习过的库中的 soname。标志指明是否立刻计算库的依赖性。如果设置为 RTLD_NOW 的话,则立刻计算;如果设置的是 RTLD_LAZY,则在需要的时候才计算。另外,可以指定 RTLD_GLOBAL,它使得那些在以后才加载的库可以获得其中的符号。

  当库被装入后,可以把 dlopen() 返回的句柄作为给 dlsym() 的第一个参数,以获得符号在库中的地址。使用这个地址,就可以获得库中特定函数的指针,并且调用装载库中的相应函数。

dlsym()的函数原型是void* dlsym(void* handle,const char* symbol)该函数在<dlfcn.h>文件中。handle是由dlopen打开动态链接库后返回的指针,symbol就是要求获取的函数或全局变量的名称,函数返回值是void*,指向函数的地址,供调用使。

dlclose(void *handle))用于关闭指定句柄的动态链接库,只有当此动态链接库的使用计数为0时,才会真正被系统卸载。

//////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////

dlopen()

  dlopen函数打开一个函数库然后为后面的使用做准备。C语言原形是:void * dlopen(const char *filename, int flag);

  如果文件名filename是以“/”开头,也就是使用绝对路径,那么dlopne就直接使用它,而不去查找某些环境变量或者系统设置的

函数库所在的目录了。否则dlopen()

  就会按照下面的次序查找函数库文件:

  1. 环境变量LD_LIBRARY指明的路径。 2. /etc/ld.so.cache中的函数库列表。 3. /lib目录,然后/usr/lib。不过一些很老的

a.out的loader则是采用相反的次序,也就是先查/usr/lib,然后是/lib。dlopen()函数中,参数flag的值必须是RTLD_LAZY或者RTLD_NOW,RTLD_LAZY的意思是resolve undefined symbols as code from the dynamic library is executed,而RTLD_NOW的含义是resolve all undefined symbols before dlopen() returns and fail if this cannot
be done'。如果有好几个函数库,它们之间有一些依赖关系的话,例如X依赖Y,那么你就要先加载那些被依赖的函数。例如先加载Y,然后加载X。dlopen()函数的返回值是一个句柄,然后后面的函数就通过使用这个句柄来做进一步的操作。如果打开失败dlopen()就返回一个NULL。如果一个函数库被多次打开,它会返回同样的句柄。如果一个函数库里面有一个输出的函数名字为_init,那么_init就会在dlopen()这个函数返回前被执行。我们可以利用这个函数在我的函数库里面做一些初始化的工作。我们后面会继续讨论这个问题的。

dlerror() :通过调用dlerror()函数,我们可以获得最后一次调用dlopen(),dlsym(),或者dlclose()的错误信息。

/*************************************************************************************************************************************************************************************/

Linux提供了一套API来动态装载库。下面列出了这些API:

- dlopen,打开一个库,并为使用该库做些准备。
- dlsym,在打开的库中查找符号的值。
- dlclose,关闭库。
- dlerror,返回一个描述最后一次调用dlopen、dlsym,或dlclose的错误信息的字符串。

C语言用户需要包含头文件dlfcn.h才能使用上述API。glibc还增加了两个POSIX标准中没有的API:
- dladdr,从函数指针解析符号名称和所在的文件。
- dlvsym,与dlsym类似,只是多了一个版本字符串参数。

在Linux上,使用动态链接的应用程序需要和库libdl.so一起链接,也就是使用选项-ldl。但是,编译时不需要和动态装载的库一起链接。程序3-1是一个在Linux上使用dl*例程的简单示例。

延迟重定位(Lazy Relocation)
延迟重定位/装载是一个允许符号只在需要时才重定位的特性。这常在各UNIX系统上解析函数调用时用到。当一个和共享库一起链接的应用程序几乎不会用到该共享库中的函数时,该特性被证明是非常有用的。这种情况下,只有库中的函数被应用程序调用时,共享库才会被装载,否则不会装载,因此会节约一些系统资源。但是如果把环境变量LD_BIND_NOW设置成一个非空值,所有的重定位操作都会在程序启动时进行。也可以在链接器命令行通过使用-z
now链接器选项使延迟绑定对某个特定的共享库失效。需要注意的是,除非重新链接该共享库,否则对该共享库的这种设置会一直有效。

初始化(initializing)和终止化(finalizing)函数
有时候,以前的代码可能用到了两个特殊的函数:_init和_fini。_init和_fini函数用在装载和卸载某个模块(注释14)时分别控制该模块的构造器和析构器(或构造函数和析构函数)。他们的C语言原型如下:
void _init(void);
void _fini(void);
当一个库通过dlopen()动态打开或以共享库的形式打开时,如果_init在该库中存在且被输出出来,则_init函数会被调用。如果一个库通过dlclose()动态关闭或因为没有应用程序引用其符号而被卸载时,_fini函数会在库卸载前被调用。当使用你自己的_init和_fini函数时,需要注意不要与系统启动文件一起链接。可以使用GCC选项 -nostartfiles
做到这一点。
但是,使用上面的函数或GCC的-nostartfiles选项并不是很好的习惯,因为这可能会产生一些意外的结果。相反,库应该使用__attribute__((constructor))和__attribute__((destructor))函数属性来输出它的构造函数和析构函数。如下所示:
void __attribute__((constructor)) x_init(void)
void __attribute__((destructor)) x_fini(void)
构造函数会在dlopen()返回前或库被装载时调用。析构函数会在这样几种情况下被调用:dlclose()返回前,或main()返回后,或装载库过程中exit()被调用时。

我们通过一个例子来讲解dlopen系列函数的使用和操作:

主程序:

#include <stdlib.h>

#include <dlfcn.h>

#include <stdio.h>

//申明结构体

typedef struct __test {

    int i;

    void (* echo_fun)(struct
__test *p);

}Test;

//供动态库使用的注册函数

void __register(Test *p) {

    p->i = 1;

    p->echo_fun(p);

}

int main(void) {

    void *handle = NULL;

    char *myso = "./mylib.so";

    if((handle = dlopen(myso, RTLD_NOW)) == NULL) {

        printf("dlopen - %sn", dlerror());

        exit(-1);

    }

    return 0;

}

动态库:

#include <stdio.h>

#include <stdlib.h>

//申明结构体类型

typedef struct __test {

    int i;

    void (*echo_fun)(struct
__test *p);

}Test;

//申明注册函数原型

void __register(Test *p);

static void __printf(Test *p) {

    printf("i = %dn", p->i);

}

//动态库申请一个全局变量空间

//这种 ".成员"的赋值方式为c99标准

static Test config = {

    .i = 0,

    .echo_fun = __printf,

};

//加载动态库的自动初始化函数

void _init(void) {

    printf("initn");

    //调用主程序的注册函数

    __register(&config);

}

主程序编译: gcc test.c -ldl -rdynamic

动态库编译: gcc -shared -fPIC -nostartfiles -o mylib.so mylib.c

主程序通过dlopen()加载一个.so的动态库文件, 然后动态库会自动运行 _init() 初始化函数, 初始化函数打印一个提示信息, 然后调用主程序的注册函数给结构体重新赋值, 然后调用结构体的函数指针, 打印该结构体的值. 这样就充分的达到了主程序和动态库的函数相互调用和指针的相互传递.

gcc参数 -rdynamic 用来通知链接器将所有符号添加到动态符号表中(目的是能够通过使用 dlopen 来实现向后跟踪).

gcc参数 -fPIC 作用: 当使用.so等类的库时,当遇到多个可执行文件共用这一个库时, 在内存中,这个库就不会被复制多份,让每个可执行文件一对一的使用,而是让多个可执行文件指向一个库文件,达到共用. 宗旨:节省了内存空间,提高了空间利用率.

内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: