Linux内核的 EXPORT_SYMBOL 和 EXPORT_SYMBOL_GPL 的作用

这篇具有很好参考价值的文章主要介绍了Linux内核的 EXPORT_SYMBOL 和 EXPORT_SYMBOL_GPL 的作用。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。


一、EXPORT_SYMBOL 的作用

  1. 在一个模块中使用 EXPORT_SYMBOL(name)。name 表示函数或者变量等符号,它是对全部内核代码公开的,因此在您的内核模块中可以直接调用 name,即使用 EXPORT_SYMBOL 可以将一个函数以符号的方式导出给其他模块使用
  2. 这里要和 System.map 做一下对比:
    System.map 中的是连接时的函数地址。连接完成以后,在2.6内核运行过程中,是不知道哪个符号在哪个地址的。
  3. EXPORT_SYMBOL 的符号,是把这些符号和对应的地址保存起来,在内核运行的过程中,可以找到这些符号对应的地址。而模块在加载过程中,其本质就是能动态连接到内核,
  4. 如果在模块中引用了内核或其它模块的符号,就要 EXPORT_SYMBOL 这些符号,这样才能找到对应的地址连接。

二、EXPORT_SYMBOL 和 EXPORT_SYMBOL_GPL 的区别

EXPORT_SYMBOL(name);
EXPORT_SYMBOL_GPL(name);
这两个宏均用于将给定的符号导出到模块外, _GPL版本的宏定义只能使符号对GPL许可的模块可用。 符号必须在模块文件的全局部分导出,不能在函数中导出,
这是因为上述这两个宏将被扩展成一个特殊用途的声明,而该变量必须是全局的。这个变量存储于模块的一个特殊的可执行部分(一个"ELF段" ),在装载时,内核通过这个段来寻找模块导出的变量
(具体可以看 <linux/module.h> 获知更详细的信息)。

三、EXPORT_SYMBOL 和 EXPORT_SYMBOL_GPL 使用方法

3.1 EXPORT_SYMBOL 使用方法

  1. 在模块函数定义之后使用 EXPORT_SYMBOL(函数名);

  2. 在调用该函数的模块中使用 extern 对它声明,才可以开始调用函数;

  3. 首先加载(insmod mod1.ko)定义该函数的模块,再加载(insmod mod2.ko)调用该函数的模块;
    举例简要说明:

    在模块(mod1)中使用 EXPORT_SYMBOL(func1);
    在模块(mod2)中声明 extern int func1();
    如此就可以在(mod2)中调用 func1 了。

    例如,在一个驱动中 drivers/net/ethernet/stmmac/tnkhw.c 定义了函数 tnkhw_bonding_setcurr_active_slave,然后在另外的.c文件中可以直接extern来使用:

    void tnkhw_bonding_setcurr_active_slave(int curr_slave)
    {
    	uint32_t data;
    	unsigned long flags;
    
    	spin_lock_irqsave(&tnkhw_reg_lock, flags);
    	data = readl(tnkhw_ioaddr + TNK_REG_TOE_BONDING_CTRL);
    	if (curr_slave) {
    		TNKBD_DBG("%s curr_slave = eth1\n", __func__);
    		writel((0x00000004 | data),
    				tnkhw_ioaddr + TNK_REG_TOE_BONDING_CTRL);
    	} else {
    		TNKBD_DBG("%s curr_slave = eth0\n", __func__);
    		writel((0xfffffffb & data),
    				tnkhw_ioaddr + TNK_REG_TOE_BONDING_CTRL);
    	}
    	spin_unlock_irqrestore(&tnkhw_reg_lock, flags);
    }
    EXPORT_SYMBOL(tnkhw_bonding_setcurr_active_slave);
    

    使用 extern 来声明:

    extern void tnkhw_bonding_setcurr_active_slave(int curr_slave);
    
    void bond_select_active_slave(struct bonding *bond)
    {
    	struct slave *best_slave;
    	int rv;
    
    	best_slave = bond_find_best_slave(bond);
    	if (best_slave != bond->curr_active_slave) {
    		bond_change_active_slave(bond, best_slave);
    #ifdef TNK_BONDING
    		if (hitoe && bond->curr_active_slave) {
    			int slave_dev_id = 0;
    			/* set bond current active slave */
    			slave_dev_id =
    				select_slave_dev(bond->curr_active_slave->dev);
    			TNKB_DBG(" %s bond->curr_active_dev->name = %s\n",
    				__func__, bond->curr_active_slave->dev->name);
    			TNKB_DBG("slave_dev_id = %d\n", slave_dev_id);
    			tnkhw_bonding_setcurr_active_slave(slave_dev_id);
    		}
    #endif
    		rv = bond_set_carrier(bond);
    		if (!rv)
    			return;
    
    		if (netif_carrier_ok(bond->dev)) {
    			pr_info("%s: first active interface up!\n",
    				bond->dev->name);
    		} else {
    			pr_info("%s: now running without any active interface !\n",
    				bond->dev->name);
    		}
    	}
    }
    

3.2 EXPORT_SYMBOL_GPL 使用方法

  • EXPORT_SYMBOL_GPL 使用方法基本与 EXPORT_SYMBOL 相同,但是也有差异。
  1. 在模块函数定义之后使用 EXPORT_SYMBOL_GPL(函数名);

  2. 在调用该函数的模块中使用 extern 对它声明,然后必须使用 MODULE_LICENSE(“GPL”) 或者 MODULE_LICENSE(“Dual BSD/GPL”),才可以开始调用函数(这是因为 EXPORT_SYMBOL_GPL 主要是给有GPL认证的模块使用的);

  3. 首先加载(insmod mod1.ko)定义该函数的模块,再加载(insmod mod2.ko)调用该函数的模块;
    举例简要说明:

    在模块(mod1)中使用 EXPORT_SYMBOL_GPL (func1);
    在模块(mod2)中先声明 extern int func1(),然后使用宏 MODULE_LICENSE(“GPL”) ;
    如此就可以在(mod2)中调用 func1 了。

    例如,在一个驱动中 drivers/usb/core/urb.c 定义了函数 usb_scuttle_anchored_urbs,然后在另外的.c文件中可以直接extern来使用:

    void usb_scuttle_anchored_urbs(struct usb_anchor *anchor)
    {
    	struct urb *victim;
    	unsigned long flags;
    
    	spin_lock_irqsave(&anchor->lock, flags);
    	while (!list_empty(&anchor->urb_list)) {
    		victim = list_entry(anchor->urb_list.prev, struct urb,
    				    anchor_list);
    		__usb_unanchor_urb(victim, anchor);
    	}
    	spin_unlock_irqrestore(&anchor->lock, flags);
    }
    
    EXPORT_SYMBOL_GPL(usb_scuttle_anchored_urbs);
    

    使用 extern 来声明:

    extern void usb_scuttle_anchored_urbs(struct usb_anchor *anchor);
    
    MODULE_LICENSE("GPL");
    
    static void carl9170_usb_cancel_urbs(struct ar9170 *ar)
    {
    	int err;
    
    	carl9170_set_state(ar, CARL9170_UNKNOWN_STATE);
    
    	err = carl9170_usb_flush(ar);
    	if (err)
    		dev_err(&ar->udev->dev, "stuck tx urbs!\n");
    
    	usb_poison_anchored_urbs(&ar->tx_anch);
    	carl9170_usb_handle_tx_err(ar);
    	usb_poison_anchored_urbs(&ar->rx_anch);
    
    	tasklet_kill(&ar->usb_tasklet);
    
    	usb_scuttle_anchored_urbs(&ar->rx_work);
    	usb_scuttle_anchored_urbs(&ar->rx_pool);
    	usb_scuttle_anchored_urbs(&ar->tx_cmd);
    }
    

四、如何查看 EXPORT_SYMBOL 和 EXPORT_SYMBOL_GPL 内核"导出"的符号表

  • EXPORT_SYMBOL(符号名)、 EXPORT_SYMBOL_GPL(符号名) 主要作之一: 内核"导出"的符号表,这个表在 insmod 时候会用到。

  • cat /proc/kallsyms 会打印出内核当前的符号表,例如:

    bf00d8c0 T tnkhw_bonding_setcurr_active_slave   [stmmac]
    bf00b460 t tnk_tcp_set_thin_linear_timeouts     [stmmac]
    bf014764 t tnkhw_check_vlan     [stmmac]
    bf0134c0 t tnkhw_tx_err_dump_info       [stmmac]
    bf00a43c t tnk_proc_init        [stmmac]
    bf01cfd8 b syscfg_base_ioaddr   [stmmac]
    bf0110b4 t tnk_set_thin_linear_timeouts [stmmac]
    bf017094 r ndesc_ops    [stmmac]
    bf01595c t tnkhw_set_txadvwnd_scale     [stmmac]
    bf0095d0 t tnk_ct_link_state_update     [stmmac]
    bf015724 t tnkhw_send_ack       [stmmac]
    bf00f790 t tnkhw_lro_free_skb   [stmmac]
    bf004f20 t dwmac_dma_stop_rx    [stmmac]
    bf00fad0 t tnkhw_rx_lro_recycle_skb     [stmmac]
    bf00d868 T tnkhw_bonding_setmode        [stmmac]
    bf015b68 t tnkhw_lookup_access  [stmmac]
    bf01c53c d stmmac_irq_num       [stmmac]
    bf00a95c t tnk_tcp_disable_rcv  [stmmac]
    bf0041e8 t stmmac_proc  [stmmac]
    bf0124a4 t curr_bonding_dev     [stmmac]
    bf01659c t cleanup_module       [stmmac]
    
  • 上述列表信息具体含义解释:

    第一列,是该符号在内核地址空间中的地址;
    第二列,是符号属性,小写表示局部符号,大写表示全局符号(具体含义参考man nm);
    第三列,表示符号字符串(即函数名或变量等);
    第四列,表示加载的驱动名称;

  • man nm 符号含义:

    “A” 符号的值是绝对值,不会因进一步链接而更改。
    “B”
    “b” 符号位于BSS数据部分。此部分通常包含初始化为零或未初始化的数据,但具体行为取决于系统。
    “C” 符号很常见。公共符号是未初始化的数据。链接时,可能会出现多个同名的通用符号。如果符号被定义在任意的地方,公共符号被视为未定义的引用。
    “D”
    “d” 符号位于初始化数据段中。
    “G”
    “g” 符号位于小对象的初始化数据段中。某些对象文件格式允许更有效地访问小数据对象,例如全局int变量,而不是大型全局数组。
    “i” 对于PE格式文件,这表示符号位于特定于DLL实现的节中。对于ELF格式文件,这表示符号是间接功能。这是ELF符号类型标准集的GNU扩展。它表示一个符号,如果被重新定位引用,该符号的值不会与其地址,但必须在运行时调用。然后,运行时执行将返回要在重新定位中使用的值。
    “I” 符号是对另一符号的间接引用。
    “N” 符号是调试符号。
    “p” 符号位于堆栈展开部分。
    “R”
    “r” 符号位于只读数据段中。
    “S”
    “s” 符号位于小对象的未初始化或零初始化数据段中。
    “T”
    “t” 符号位于文本(代码)部分。
    “U” 符号未定义。
    “u” 符号是唯一的全局符号。这是ELF符号绑定标准集的GNU扩展。对于这样的符号,动态链接器将确保在在整个过程中,只有一个符号使用此名称和类型。
    “V”
    “v” 符号是一个弱对象。当弱定义符号与正常定义符号链接时,使用正常定义符号时不会出错。当弱未定义符号被链接并且符号未被定义,则弱符号的值变为零,没有错误。在某些系统上,大写表示默认值具有已指定。
    “W”
    “w” 符号是一个弱符号,未被明确标记为弱对象符号。当弱定义符号与正常定义符号链接时使用定义的符号时没有错误。当弱未定义符号被链接且未定义该符号时,该符号的值将在特定系统中确定没有错误的方式。在某些系统上,大写表示已指定默认值。
    “-” 符号是a.out对象文件中的刺符号。在这种情况下,接下来打印的值是插入其他字段、插入描述字段和插入类型。Stabs符号用于保存调试信息。
    “?” 符号类型未知,或特定于对象文件格式。文章来源地址https://www.toymoban.com/news/detail-766519.html


到了这里,关于Linux内核的 EXPORT_SYMBOL 和 EXPORT_SYMBOL_GPL 的作用的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 【Shell 命令集合 系统设置 】⭐⭐⭐Linux 置环境变量 export命令 使用指南

    Shell 命令专栏:Linux Shell 命令全解析 export命令是Linux中的一个内建命令,用于设置环境变量。环境变量是在操作系统中用于存储各种配置信息的一种机制。通过设置环境变量,可以影响系统的行为和程序的运行。 使用export命令,可以将一个变量从当前shell传递到子shell或其他

    2024年02月05日
    浏览(67)
  • 调用GPL 开源库的法律问题

    调用 GPL(General Public License)开源库通常是合法的,但是需要遵守 GPL 协议的规定。使用 GPL 开源库可能会涉及到一些问题,以下是一些你可能需要考虑的方面: 遵守 GPL 协议 : 使用 GPL 开源库时,你需要遵守 GPL 协议的规定。这包括在你的项目中使用 GPL 开源库时,必须将你

    2024年03月11日
    浏览(46)
  • 开源的协议(GPL和MIT的区别)

    开源不仅能够帮助整个生态共同进步,也能够帮助个人开发者提升技术和名气,但是,开源的意思并不是没有规则,全部无条件的免费提供给别人用,必须要遵循一定的规则。 这个规则就是开源协议(Open Source License)。常用的只有5、6种,网络上的很多文章只是笼统的介绍,

    2024年02月06日
    浏览(40)
  • export 是一个在 Unix 和类 Unix 系统(比如 Linux 和 macOS)中常用的 shell 命令,主要用于设置或导出环境变量。

    export 是一个在 Unix 和类 Unix 系统(比如 Linux 和 macOS)中常用的 shell 命令,主要用于设置或导出环境变量。环境变量是在操作系统中用于存储系统设置和命令行程序配置的全局值。下面提供了一些 export 命令的基本用法和示例。 基本用法 设置环境变量 : 这里, VARIABLE_NAME 是

    2024年01月19日
    浏览(43)
  • Windows内核和Linux内核比较(附Linux内核各版本历史纪年表)

    我是荔园微风,作为一名在IT界整整25年的老兵,最近受邀给年轻人讲了一场Windows内核和Linux内核相关的讲座。大家听得非常认真。 计算机操作系统的功能角色:作为用户和计算机硬件资源之间的交互,管理调度硬件资源,为应用软件提供运行环境。操作系统属于基础软件,

    2024年02月10日
    浏览(55)
  • Linux内核8. Linux内核的经典调试方式

    内核总是那么捉摸不透, 内核也会犯错, 但是调试却不能像用户空间程序那样, 为此内核开发者为我们提供了一系列的工具和系统来支持内核的调试. 内核的调试, 其本质是内核空间与用户空间的数据交换, 内核开发者们提供了多样的形式来完成这一功能. 内核中有三个常用的伪

    2024年02月07日
    浏览(51)
  • 开源世界许可证Copyleft GPL LGPL MIT BSD Apache

    Copyleft 利用版权法来提供分发作品的副本和修改版本的权利,并要求在作品的修改版本中保留同样的权利。换句话说,copyleft是让创造性的作品可以自由修改,并要求作品的所有修改和扩展版本也是自由的一般方法。 GPL是首个通用的copyleft许可证。 GNU通用公共许可证 ( GNU

    2024年01月17日
    浏览(57)
  • Linux内核学习(六)—— 中断(基于Linux 2.6内核)

    目录 一、中断 二、中断处理程序 三、注册中断处理程序 四、卸载中断处理程序 五、编写中断处理程序 六、中断上下文 七、中断下半部(bottom half) 软中断 Tasklet  工作队列 中断使得硬件得以发出通知给处理器。中断随时都可以产生,如键盘敲击就会触发中断,通知操作系

    2024年02月12日
    浏览(43)
  • Linux内核学习(包含Linux 2.6内核编译安装流程)

    Linux内核官方网站为:http://www.kernel.org 或者使用git将源码clone下来(我这里使用的版本为2.6): clone下来的源码目录结构如下:  其中比较重要的目录的官方描述如下: 对应的中文描述如下: 由于centos7.6初始gcc版本为4.8.5,而编译内核需要的最低版本为5.1.0,所以需要安装高

    2024年02月07日
    浏览(63)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包