共享库的版本控制


共享库的版本控制

关于 SONAME

编译动态库时,可以指定 SONAME 。程序运行时,加载器(Dynamic Loader)发现动态库有 DT_SONAME 字段,则会转而去链接 DT_SONAME 指定的动态库。

应用:库的版本

Linux 动态库的版本号一般格式为 lib + libname + .so + 主版本号 + 子版本号 + 发行版本号 。

情况一:如果版本号发生了改变之后,实际库仍然是兼容的,此时更新软链接,将主版本,即可实现库的更新。
情况二:如果版本号发生了改变之后(一般是主版本),库不再兼容,此时变更 SONAME ,即可实现库的多版本兼容(多个库同时存在,由程序根据 SONAME 选择加载)。

例子:

// add.c
// V1.0.0
int add(int a, int b)
{
    return a+b;
}
// main.c
#include<stdio.h>
int add(int a, int b);
int main()
{
    int a = 1;
    int b = 2;
    int c = add(a, b);
    printf ("c=%d/n", c);
    return 0;
}

对第一种情况:

版本更新前:

gcc add.c -shared -fPIC -Wl,-soname=libadd.so.1 -o libadd.so.1.0.0
ldconfig -nv .
ln -s libadd.so.1 libadd.so
gcc main.c -L. -ladd 
export LD_LIBRARY_PATH=.:$LD_LIBRARY_PATH
./a.out

观察动态库和 a.out 的符号:

readelf -d libadd.so.1.0.0
Dynamic section at offset 0x2e80 contains 18 entries:
  Tag        Type                         Name/Value
 0x000000000000000e (SONAME)             Library soname: [libadd.so.1]

readelf -d a.out
Dynamic section at offset 0x2db0 contains 28 entries:
  Tag        Type                         Name/Value
 0x0000000000000001 (NEEDED)             Shared library: [libadd.so.1]

也即,编译库的时候,生成的库的名字为最完整的名字(比如 libadd.so.1.0.0),库的 SONAME 字段为兼容版本的库名字(比如 libadd.so.1)。生成 a.out 时,链接的名字始终使用 -ladd ,也即需要创建一个 libadd.so 的链接指向兼容版本的库。最终生成 a.out 时,依赖的库的名字是根据 libadd.so 中的 SONAME 字段生成的(SONAME 字段存在时),也即 a.out 依赖的库名字为 libadd.so.1。

注意:libadd.so 只有编译生成 a.out 时需要使用(DEV)。如果只是部署,则只需要 libadd.so.1 和 libadd.so.1.0.0 就行了。

版本更新后(更新后仍然兼容,只更新子版本号):

rm libadd.1.*
gcc add.c -shared -fPIC -Wl,-soname=libadd.so.1 -o libadd.so.1.1.0
ldconfig -nv .

版本更新时,先删除当前已有的兼容版本,然后生成新的兼容版本,并利用 ldconfig 更新兼容版本链接。

ldconfig -nv .
.:
        libadd.so.1 -> libadd.so.1.1.0 (changed)

兼容版本更新过程中,需要删除已有版本。

对第二种情况:

// add.c
// V1.0.0
int add(int a, int b, int c)
{
    return a+b+c;
}
// main.c
#include<stdio.h>
int add(int a, int b, int c);
int main()
{
    int a = 1;
    int b = 2;
    int c = add(a, b, b);
    printf ("c=%d/n", c);
    return 0;
}

如果出现了库的不兼容版本更新,则两个库可以兼容存在。旧的程序链接旧版本的库,新的程序链接新版本的库,彼此互不干扰。

gcc add.c -shared -fPIC -Wl,-soname=libadd.so.2 -o libadd.so.2.0.0
ldconfig -nv .
ln -s libadd.so.2 libadd2.so
gcc main.c -L. -ladd2 -o a2.out
export LD_LIBRARY_PATH=.:$LD_LIBRARY_PATH
./a2.out

这里我们把用于编译的库的名字做了区分(libadd.so 和 libadd2.so),这时为了编译的时候进行版本区分。

这样就实现了多个库的兼容。

原创文章,作者:carmelaweatherly,如若转载,请注明出处:https://blog.ytso.com/278477.html

(0)
上一篇 2022年8月2日
下一篇 2022年8月2日

相关推荐

发表回复

登录后才能评论