首页 > 系统相关 >关于Linux中so显式链接(dlopen)找不到函数符号地址的问题

关于Linux中so显式链接(dlopen)找不到函数符号地址的问题

时间:2024-02-23 18:56:55浏览次数:32  
标签:main 函数 libone Linux Two so 显式 libtwo

摘自:https://blog.csdn.net/qq_27281753/article/details/127202676

问题背景
在做项目的时候,遇到一个so调用问题,既别人提供了一些so库,其中一个so库包含了给我调用的函数,而这个库里面的函数又调用了其他库的函数,这些所有的库都是linux下编译出来的,而项目则是需要在windows下用Qt交叉编译的。这会导致一些问题,(1)如果想要在Qt工具编译项目的时候使用隐式链接方式,把so库链接进去,会出现找不到别人提供的so库中所依赖的linux系统自带的库。(2)如果使用显式链接,既使用dlopen加载so库,然后再使用dlsym加载函数的方式,那么就会出现找不到so库中函数所调用其他库函数的问题。
由于在交叉编译环境中,目前只能使用显示链接的方式调用别人提供的so库,下面就会用一些测试代码,来展示问题是如何出现的以及如何解决问题。
(一)隐式链接和显示链接的结果
首先我们先输出一个最基础的so库libone.so,具体代码如下:
one.h

extern "C" void One();

one.c

#include "one.h"
#include "stdio.h"
void One()
{
printf("One calling\n");
}

 


然后再写第二个so库libtwo.so,用它调用上面的libone.so中的函数。具体代码如下:
two.h

extern "C" void Two();

 

two.c

#include "one.h"
#include "two.h"
void Two()
{
One();
}

 


main.c

#include "dlfcn.h"
#include "stdio.h"
#include "two.h"
typedef void (*pTwo)();

//TEST宏控制是使用dlopen方式还是直接调用Two方式
#define TEST 0
int main()
{
#if TEST
printf("start main\n");
pTwo Two = NULL;
void *pOpen = NULL;
pOpen = dlopen("./libtwo.so", RTLD_LAZY);
if (pOpen == NULL)
{
printf("load libtwo.so fail\n");
return -1;
}

Two = (pTwo)dlsym(pOpen, "Two");
if (Two == NULL)
{
printf("load function Two fail\n");
return -1;
}
#endif
printf("start to call Two\n");
Two();
#if TEST
dlclose(pOpen);
pOpen = NULL;
#endif
return 0;
}

 

makefile文件

all: libone.so libtwo.so main
main: main.c
g++ main.c -ldl -L. -ltwo -lone -o main
libtwo.so: two.c
g++ two.c -fPIC -shared -o -libtwo.so
libone.so: one.c
g++ one.c -fPIC -shared -o -libone.so
clean:
rm ./main ./libtwo.so ./libone.so

 

上面的Makefile文件中,main程序是链接了libone.so和libtwo.so两个库,但是libtwo.so并没有把libone.so链接上。
运行结果:

ldd main

 


ldd libtwo.so

 


nm -D libtwo.so

 

可以看到One函数符号没有地址的

(1)没有打开TEST宏的结果:

 


(2)打开TEST宏的结果:

 


此时,问题已经出现了,main把两个需要用到的库在编译后已经链接起来了,所以程序中直接调用Two函数的话,可以看到One是被调用起来了。但是如果使用dlopen去加载libtwo.so库,然后再去加载Two函数的方式时,通过nm -D libtwo.so可以看到,One的函数地址是没有的,所以调用Two时,它再调用One就会报错。

libtwo.so和libone.so就相当于别人提供的一些库,libtwo.so中的Two函数就是我们需要使用的函数。那么在项目无法使用第一种方式而只能使用dlopen的方式时,怎么解决该问题呢?可以看到main是可以通过编译链接直接调用Two函数的。那么我们就可以再封装一层,把main看成第三个库,将它封装起来,因为它已经是在Linux环境下能够正常编译和运行的程序了。只需要把main封装成so库,再写一个对外的函数,而这个函数里面直接调用Two,那么就间接的(类似于第一种方式直接的)调用Two函数了。下面就是测试代码和运行结果的展示。

(二)解决显示链接问题的方式
封装出第三个库libthree.so
three.h

extern "C" void Three();

three.c

#include "three.h"
#include "two.h"
void Three()
{
Two();
}

 


然后再修改一下makefile文件:

all: libone.so libtwo.so libthree.so main
main: main.c
g++ main.c -ldl -L. -lthree -ltwo -lone -o main
//这个libthree.so就相当于上半部分中的main
libthree.so: three.c libtwo.so libone.so
g++ three.c -fPIC -shared -L. -ltwo -lone -o libthree.so
libtwo.so: two.c
g++ two.c -fPIC -shared -o -libtwo.so
libone.so: one.c
g++ one.c -fPIC -shared -o -libone.so
clean:
rm ./main ./libthree.so ./libtwo.so ./libone.so

main.c
main中就把Two改成Three就行:

#include "dlfcn.h"
#include "stdio.h"
#include "three.h"
typedef void (*pThree)();

//TEST宏控制是使用dlopen方式还是直接调用Two方式
#define TEST 0
int main()
{
#if TEST
printf("start main\n");
pThree Three = NULL;
void *pOpen = NULL;
pOpen = dlopen("./libthree.so", RTLD_LAZY);
if (pOpen == NULL)
{
printf("load libthree.so fail\n");
return -1;
}

Three = (pThree)dlsym(pOpen, "Three");
if (Three == NULL)
{
printf("load function Three fail\n");
return -1;
}
#endif
printf("start to call Three\n");
Three();
#if TEST
dlclose(pOpen);
pOpen = NULL;
#endif
return 0;
}

 

运行结果:

(1)没有打开宏的结果:

 

 


(2)打开宏的结果:

 

 

总结:从最后的结果可以看到,通过再封装一层的方式,解决了一开始展示的找不到函数符号的问题,使得libthree.so无论是直接的调用Three还是动态加载Three的方式,都能成功得调起了Two函数,直至最终One函数的打印被调用起来。

标签:main,函数,libone,Linux,Two,so,显式,libtwo
From: https://www.cnblogs.com/LiuYanYGZ/p/18030218

相关文章

  • linux(ubuntu22.04)+PicGo(gui版)+阿里云oss搭建图床教程
    linux(ubuntu22.04)+PicGo(gui版)+阿里云oss搭建图床教程资源库PicGo下载链接:山东镜像源github原版阿里云oss链接linux下PicGo(gui版)的安装从资源库链接里下载后缀为.AppImage的安装包,版本可以选择稳定版2.3.1也可以用更新的beta版。修改文件权限,打开文......
  • Cursor 介绍与基础生成用法
    Cursor介绍与基础生成用法实验介绍Cursor是一款与OpenAI合作并且基于GPT-4的新一代辅助编程神器,国内直接可以访问,它可以根据你的输入和需求自动生成代码片段,还可以帮助你重构、理解和优化代码,提高开发效率。在本节课程中,我们将介绍Cursor的基本使用方法。知识点Curso......
  • linux cpu 内存分析
    1.通过分析服务器资源,当发现资源消耗过多时,需要分析什么进程占用了,如下所示 2.分析第一台服务器通过登录服务器,使用top命令查看,出来信息如下所示: 进程182618的内存占用了52.6%,属于.net应用程序,通过已维护的文档,知道了哪些.net程序的部署了,最终找到是该web应用程序......
  • 业界唯一单芯片自适应射频平台:XCZU42DR-L2FSVE1156I、XCZU42DR-1FFVE1156I、XCZU65DR-
    ZynqUltraScale+RFSoC是业界唯一单芯片自适应射频平台。ZynqUltraScale+RFSoC是一种异构计算架构,包括完整的Arm处理子系统、FPGA架构,以及RF信号链中的完整模数可编程性,其不仅可为不同的应用提供一个完整的单片软件定义无线电平台,而且还有助于随着市场动态的发展,生产无线......
  • 多节点linux环境打造
    整体架构:1+31指:宿主机(作为客户端访问)3指:3个虚拟机(其中1个作为前端项目部署服务器,2个作为多实例的后端项目服务器集群)虚拟机上服务器配置step1:vmwareworkstation安装及破解+借助centos7镜像step2:确保宿主机可ping通外网且可ping通各虚拟机节点各虚拟机节点可以pi......
  • (笔记)Linux基础知识点总结
     一、从认识操作系统开始 1、操作系统简单分类Windows​目前最流行的个人桌面操作系统,不做多的介绍,大家都清楚。界面简单易操作,软件生态非常好。Unix​最早的多用户、多任务操作系统。后面崛起的Linux在很多方面都参考了Unix。目前这款操作系统已......
  • lazarus3.0 /fpc3.3.1编译某些控件会出现:Error: Forward declaration not solved xxx
    最近用lazarus3.0/fpc3.3.1时发现原来在lazarus2.2.6/fpc3.2.2能编译安装的控件出现类似下面的提示codebot.text.xml.pas(129,10)Error:Forwarddeclarationnotsolved"NewDocument:IDocument;"解决方法:本例子参照DocumentCreate:IDocument,在实现部分编写过程。{$i......
  • Linux搭建SFTP
    Linux搭建SFTP一、 创建sftp用户组(注:通常选择home路径下,按需自行调整)1、 创建sftp用户组根目录(详情看第4点)mkdir/home/sftpUsers/name创建单个用户name:addusername//新建name用户passwdname//给name用户设置密码—部分需要设置复杂密码包含数字,且长度不小于8位nam......
  • Linux学习-day1
    了解Linux运维。Linux运维在互联网行业中处于维护与优化作用,保障网站、服务器、程序等正常运转。2.画图网站:https://excalidraw.com/3.安装VMware虚拟机与centos系统。阿里云镜像下载地址:https://developer.aliyun.com/mirror/安装步骤:准备好镜像文件:DVD(大约4GB,包含常......
  • linux基础命令(七)
    find高级使用–find[目录][条件1][-a|-o][条件2]...–常用条件表示:-type类型(f、d、l)f:文本文件d:目录l:快捷方式-name"文档名称"-size+|-文件大小(k、M、G)-user用户名-mtime根据文件修改时间###################################......