应用程序设计:在动态库中如何调用外部函数?

嵌入式ARM 2021-05-25 10:06


大家好,我是一个动态链接库

这个名字,相信你一定早就如雷贯耳了。

在计算机早期时代,由于内存资源紧张,我可是发挥了重大的作用!

不论是在 Windows 系统中,还是在 Unix 系列平台上,到处都能见到我的身影,因为我能为大家节省很多资源啊,资源就是人民币!


愉快的玩耍


比如:我的主人编写了这么一段简单的代码:

# 文件:lib.c

#include <stdio.h>

int func_in_lib(int k)
{
printf("func_in_lib is called \n");
return k + 1;
}

只要用如下命令来编译,我就诞生出来了 lib.so,也就是一个动态链接库:

$ gcc -m32 -fPIC --shared -o lib.so lib.c

这个时候,主人随便把我丢给谁,我都可以为他服务,只要他调用我肚子里的这个函数 func_in_lib 就可以了。

虽然目前你看到我提供的这个函数很简单,但是道理都是一样的,后面如果有机会,我就在这个函数里来计算机器人的运动轨迹,给你瞧一瞧!

例如:张三今天写了一段代码,需要调用我的这个函数。

张三这个人比较喜欢骚操作,明明他在编译可执行程序的时候,把我动态链接一下就可以了,就像下面这样:

$ gcc -m32 -o main main.c ./lib.so

但是张三偏偏不这么做,为了炫技,他选择使用 dlopen 动态加载的方式,来把我从硬盘上加载到进程中。

咱们来一起围观一下张三写的可执行程序代码:

# 文件:main.c

#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>
#include <dlfcn.h>

typedef int (*pfunc)(int);

int main(int argc, char *agv[])
{
int a = 1;
int b;

// 打开动态库
void *handle = dlopen("./lib.so", RTLD_NOW);
if (handle)
{
// 查找动态库中的函数
pfunc func = (pfunc) dlsym(handle, "func_in_lib");
if (func)
{
b = func(a);
printf("b = %d \n", b);
}
else
{
printf("dlsym failed! \n");
}
dlclose(handle);
}
else
{
printf("dlopen failed! \n");
}

return 0;
}

从代码中可以看到,张三预先知道我肚子里的这个函数名称是 func_in_lib,所以他使用了系统函数 dlsym(handle, "func_in_lib"); 来找到这个函数在内存中的加载地址,然后就可以直接调用这个函数了。

张三编译得到可执行文件 main 之后,执行结果完全正确,很开心!


悲从中来


可是有一天,我遇到一件烦人的事情,我的主人说:你这个服务函数的计算过程太单调了,给你找点乐子,你在执行的时候啊,到其他一个外部模块里调用一个函数。

话刚说完,就丢给我一个函数名:void func_in_main(void);

也就是说,我需要在我的服务函数中,去调用其他模块里的函数,就像下面这样:

#include <stdio.h>

// 外部函数声明
void func_in_main(void);

int func_in_lib(int k)
{
printf("func_in_lib is called \n");

// 调用外部函数
func_in_main();

return k + 1;
}

那么这个函数在哪里呢?天哪,我怎么知道这个函数是什么鬼?怎么才能找到它藏在内存的那个角落(地址)里?

不管怎么样,主人修改了代码之后,还是很顺利的把我编译了出来:

$ gcc -m32 -fPIC --shared -o lib.so lib.c

编译指令完全没有变化。

因为我仅仅是一个动态链接库,这个时候即使我不知道 func_in_main 函数的地址,也是可以编译成功的。

只不过我要把这个家伙标记一下:谁要是想使用我,就必须告诉我这个家伙的地址在哪里!,否则就别怪我耍赖。


无辜的张三


我的主人对张三说:兄弟,我的这个动态链接库升级了,功能更强大哦,想不想试一下?

张三心想:我是使用 dlopen 的方式来动态加载动态库文件的,不需要对可执行程序重新编译或者链接,直接运行就完事了!

于是他二话不说,直接就把我拿过去,丢在他的可执行程序目录下,然后执行 main 程序。

可是这一次,他看到的结果却是:

dlopen failed!

为什么会加载失败呢?上次明明是正常执行的!张三一脸懵逼!

其实,这压根就不能怪我!以为我刚才就说了:谁要是想使用我,就必须告诉我 func_in_main 这个函数的地址在哪里!

可是在张三的这个进程里,我到处都找不到这个函数的地址。既然你没法满足我,那我就没法满足你!


锦囊1: 导出符号表


张三这下也没辙了,只要找我的主人算账:我的应用程序代码一丝一毫都没有动,怎么换了你给的新动态链接库就不行了呢?

主人慢条斯理的回答:疏忽了,疏忽了,忘记跟你说一件事情了:这个动态库啊,它需要你多做一件事情:在你的程序中提供一个名为 func_in_main 的函数,这样就可以了。

张三一想:这个好办,加一个函数就是了。

因为这个可执行程序只有一个 main.c 文件,于是他在其中新加了一个函数

void func_in_main(void)
{
printf("func_in_main \n");
}

然后就开始编译、执行,一顿操作猛如虎:

# gcc -m32 -o main main.c -ldl
# ./main
dlopen failed!

咦?怎么还是失败?!已经按照要求加了 func_in_main 这个函数了啊?!

这个傻X张三,对,你确实是在 main.c 中加了这个函数,但是你仅仅是加在你的可执行程序中的,但是我却压根就看不到这个函数啊!

不信的话,你检查一下编译出来的可执行程序中,是否把 func_in_main 这个符号导出来了?如果不导出来,我怎么能看到?

# 查看导出的符号表
$ objdump -e main -T | grep func_in_main
# 这里输出为空

既然输出为空,就说明没有导出来!这个就不用我教你了吧?

茴香豆的“茴”字,一共有四种写法。。。

哦,不,导出符号,一共有两种方式:

方式1:导出所有的符号

$ gcc -m32 -rdynamic -o main main.c -ldl

当然,下面这个指令也可以:

gcc -m32 -Wl,--export-dynamic -o main main.c -ldl

方式2:导出指定的符号

先定义一个文件,把需要导出的符号全部罗列出来:

文件:exported.txt

{
extern "C"
{
func_in_main;
};
};

然后,在编译选项中指定这个导出文件:

gcc -m32 -Wl,-dynamic-list=./exported.txt -o main main.c -ldl

使用以上两种方式的任意一种即可,编译之后,再使用 objdump 指令看一下导出符号

$ objdump -e main -T | grep func_in_main
080485bb g DF .text 00000019 Base func_in_main

嗯,很好很好!张三赶紧按照这样的方式操作了一下,果真成功执行了函数!

$ ./main 
func_in_lib is called
func_in_main
b = 2

也就是说,在我的动态库文件中,正确的找到了外部其他模块中的函数地址,并且愉快的执行成功了!


锦囊2: 动态注册


虽然执行成功了,张三的心里隐隐约约的仍然有一丝不爽的感觉,每次编译都要导出符号,真麻烦,能不能优化一下?

于是他找到我的主人,表达了自己的不满。

主人一瞧,有个性!既然你不想提供,那我就满足你:

  1. 首先,在动态库中提供一个默认的函数实现(func_in_main_def);

  2. 然后,再提供一个专门的注册函数(register_func),如果外部模块想提供 func_in_main 这个函数,就调用注册函数注册进来;

此时,lib.c 最新的代码就变成这个样子了:

#include <stdio.h>

// 默认实现
void func_in_main_def(void)
{
printf("the main is lazy, do NOT register me! \n");
}

// 定义外部函数指针
void (*func_in_main)() = func_in_main_def;

void register_func(void (*pf)())
{
func_in_main = pf;
}

int func_in_lib(int k)
{
printf("func_in_lib is called \n");

if (func_in_main)
func_in_main();

return k + 1;
}

然后编译,全新的我再一次诞生了 lib.so

gcc -m32 -fPIC --shared -o lib.so lib.c

主人把我丢给张三的时候说:好了,满足你的需求,这一次你不用提供 func_in_main 这个函数了,当然也就不用再导出符号了。

不过,如果如果有一天,你改变了注意,又想提供这个函数了,那么你就要通过动态库中的 register_func 函数,把你的函数注册进来。

Have you got it?赶紧再去试一下!

这个时候,张三再次使用我的时候,就不需要导出他的 main.c 里的那个函数 func_in_main 了,实际上他可以把这个函数从代码中删掉!

编译、执行,张三再一次猛如虎的操作:

$ gcc -m32 -o main main.c -ldl
$ ./main
func_in_lib is called
the main is lazy, do NOT register me!
b = 2

嗯,结果看起来是正确的。

咦?怎么多了一行字:the main is lazy, do NOT register me!

难道是在质疑我的技术能力吗?好吧,既然如此,我也满足你,不就是注册一个函数嘛,简单:

// 文件: main.c

#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>
#include <dlfcn.h>

typedef int (*pfunc)(int);
typedef int (*pregister)(void (*)());

// 控制注册函数的宏定义
#define REG_FUNC

#ifdef REG_FUNC
void func_in_main(void)
{
printf("func_in_main \n");
}
#endif

int main(int argc, char *agv[])
{
int a = 1;
int b;

// 打开动态库
void *handle = dlopen("./lib.so", RTLD_NOW);
if (handle)
{
#ifdef REG_FUNC
// 查找动态库中的注册函数
pregister register_func = (pregister) dlsym(handle, "register_func");
if (register_func)
{

register_func(func_in_main);
}
#endif

// 查找动态库中的函数
pfunc func = (pfunc) dlsym(handle, "func_in_lib");
if (func)
{
b = func(a);
printf("b = %d \n", b);
}
else
{
printf("dlsym failed! \n");
}
dlclose(handle);
}
else
{
printf("dlopen failed! \n");
}

return 0;
}

然后编译、执行:

$ gcc -m32 -o main main.c -ldl
$ ./main
func_in_lib is called
func_in_main
b = 2

完美收官!

PS:很多平台级的代码,例如一些工控领域的运行时(Runtime)软件,大部分都是通过注册的方式,来把平台代码、用户代码进行连接、绑定的。


END

来源:IOT物联网小镇,作者:道哥

版权归原作者所有,如有侵权,请联系删除。

推荐阅读
用GD32替代STM32:温湿度采集传感器开发心得
手把手教会你:使用STM32F103驱动ST7567液晶屏
单片机经验:STC15W4系列6路专用PWM模块使用心得


→点关注,不迷路←
嵌入式ARM 关注这个时代最火的嵌入式ARM,你想知道的都在这里。
评论
  • 本文介绍瑞芯微RK3588主板/开发板Android12系统下,APK签名文件生成方法。触觉智能EVB3588开发板演示,搭载了瑞芯微RK3588芯片,该开发板是核心板加底板设计,音视频接口、通信接口等各类接口一应俱全,可帮助企业提高产品开发效率,缩短上市时间,降低成本和设计风险。工具准备下载Keytool-ImportKeyPair工具在源码:build/target/product/security/系统初始签名文件目录中,将以下三个文件拷贝出来:platform.pem;platform.
    Industio_触觉智能 2024-12-12 10:27 115浏览
  • 一、SAE J1939协议概述SAE J1939协议是由美国汽车工程师协会(SAE,Society of Automotive Engineers)定义的一种用于重型车辆和工业设备中的通信协议,主要应用于车辆和设备之间的实时数据交换。J1939基于CAN(Controller Area Network)总线技术,使用29bit的扩展标识符和扩展数据帧,CAN通信速率为250Kbps,用于车载电子控制单元(ECU)之间的通信和控制。小北同学在之前也对J1939协议做过扫盲科普【科普系列】SAE J
    北汇信息 2024-12-11 15:45 140浏览
  • 习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习笔记&记录学习习笔记&记学习学习笔记&记录学习学习笔记&记录学习习笔记&记录学习学习笔记&记录学习学习笔记记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记
    youyeye 2024-12-12 10:13 79浏览
  • 首先在gitee上打个广告:ad5d2f3b647444a88b6f7f9555fd681f.mp4 · 丙丁先生/香河英茂工作室中国 - Gitee.com丙丁先生 (mr-bingding) - Gitee.com2024年对我来说是充满挑战和机遇的一年。在这一年里,我不仅进行了多个开发板的测评,还尝试了多种不同的项目和技术。今天,我想分享一下这一年的故事,希望能给大家带来一些启发和乐趣。 年初的时候,我开始对各种开发板进行测评。从STM32WBA55CG到瑞萨、平头哥和平海的开发板,我都
    丙丁先生 2024-12-11 20:14 97浏览
  • 全球智能电视时代来临这年头若是消费者想随意地从各个通路中选购电视时,不难发现目前市场上的产品都已是具有智能联网功能的智能电视了,可以宣告智能电视的普及时代已到临!Google从2021年开始大力推广Google TV(即原Android TV的升级版),其他各大品牌商也都跟进推出搭载Google TV操作系统的机种,除了Google TV外,LG、Samsung、Panasonic等大厂牌也开发出自家的智能电视平台,可以看出各家业者都一致地看好这块大饼。智能电视的Wi-Fi连线怎么消失了?智能电
    百佳泰测试实验室 2024-12-12 17:33 119浏览
  • 习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习笔记&记录学习习笔记&记学习学习笔记&记录学习学习笔记&记录学习习笔记&记录学习学习笔记&记录学习学习笔记记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记
    youyeye 2024-12-11 17:58 108浏览
  • 铁氧体芯片是一种基于铁氧体磁性材料制成的芯片,在通信、传感器、储能等领域有着广泛的应用。铁氧体磁性材料能够通过外加磁场调控其导电性质和反射性质,因此在信号处理和传感器技术方面有着独特的优势。以下是对半导体划片机在铁氧体划切领域应用的详细阐述: 一、半导体划片机的工作原理与特点半导体划片机是一种使用刀片或通过激光等方式高精度切割被加工物的装置,是半导体后道封测中晶圆切割和WLP切割环节的关键设备。它结合了水气电、空气静压高速主轴、精密机械传动、传感器及自动化控制等先进技术,具有高精度、高
    博捷芯划片机 2024-12-12 09:16 111浏览
  • 应用环境与极具挑战性的测试需求在服务器制造领域里,系统整合测试(System Integration Test;SIT)是确保产品质量和性能的关键步骤。随着服务器系统的复杂性不断提升,包括:多种硬件组件、操作系统、虚拟化平台以及各种应用程序和服务的整合,服务器制造商面临着更有挑战性的测试需求。这些挑战主要体现在以下五个方面:1. 硬件和软件的高度整合:现代服务器通常包括多个处理器、内存模块、储存设备和网络接口。这些硬件组件必须与操作系统及应用软件无缝整合。SIT测试可以帮助制造商确保这些不同组件
    百佳泰测试实验室 2024-12-12 17:45 113浏览
  • 在智能化技术快速发展当下,图像数据的采集与处理逐渐成为自动驾驶、工业等领域的一项关键技术。高质量的图像数据采集与算法集成测试都是确保系统性能和可靠性的关键。随着技术的不断进步,对于图像数据的采集、处理和分析的需求日益增长,这不仅要求我们拥有高性能的相机硬件,还要求我们能够高效地集成和测试各种算法。我们探索了一种多源相机数据采集与算法集成测试方案,能够满足不同应用场景下对图像采集和算法测试的多样化需求,确保数据的准确性和算法的有效性。一、相机组成相机一般由镜头(Lens),图像传感器(Image
    康谋 2024-12-12 09:45 117浏览
  • 时源芯微——RE超标整机定位与解决详细流程一、 初步测量与问题确认使用专业的电磁辐射测量设备,对整机的辐射发射进行精确测量。确认是否存在RE超标问题,并记录超标频段和幅度。二、电缆检查与处理若存在信号电缆:步骤一:拔掉所有信号电缆,仅保留电源线,再次测量整机的辐射发射。若测量合格:判定问题出在信号电缆上,可能是电缆的共模电流导致。逐一连接信号电缆,每次连接后测量,定位具体哪根电缆或接口导致超标。对问题电缆进行处理,如加共模扼流圈、滤波器,或优化电缆布局和屏蔽。重新连接所有电缆,再次测量
    时源芯微 2024-12-11 17:11 141浏览
我要评论
0
点击右上角,分享到朋友圈 我知道啦
请使用浏览器分享功能 我知道啦