Linux内核导出符号宏定义EXPORT_SYMBOL的源码分析

源代码:

<include/linux/moudule.h>

…….

#ifndef MODULE_SYMBOL_PREFIX

#define MODULE_SYMBOL_PREFIX ""

#endif

…….

struct kernel_symbol       //内核符号结构

{

unsignedlong value;  //该符号在内存地址中的地址

constchar *name;     //该符号的名称

};

……

#define __EXPORT_SYMBOL(sym,sec)                                 \

externtypeof(sym) sym;                                                        \

__CRC_SYMBOL(sym,sec)                                            \

staticconst char __kstrtab_##sym[]                                 \

__attribute__((section(“__ksymtab_strings”),aligned(1)))   \

=MODULE_SYMBOL_PREFIX#sym;                      \

staticconst struct kernel_symbol __ksymtab_##sym         \

__used                                                                          \

__attribute__((section(“__ksymatab”sec),unused))                   \

={(unsignedlong)&sym,_kstrab_#sym}

#define    EXPORT_SYMBOL(sym)                   \

__EXPOTR_SYMBOL(sym,””)

#define    EXPORT_SYMBOL_GPL(sym)           \

__EXPOTR_SYMBOL(sym,”_gpl”)

#define    EXPORT_SYMBOL(sym)                   \

__EXPOTR_SYMBOL(sym,”_gpl_future”)

1、预备知识:

在分析前,先了解如下相关知识:

(1)#运算符,##运算符

通常在宏定义中使用#来创建字符串 #abc就表示字符串”abc”等。

##运算符称为预处理器的粘合剂,用来替换粘合两个不同的符号,

如:#definexName (n)  x##n

则xName(4)  则变为x4

(2)gcc的__attribute__属性:

__attribute__((section(“section_name”)))的作用是将指定的函数或变量放入到名为”section_name”的段中。

__attribute__属性添加可以再函数或变量定义的时候直接加入在定义语句中。

如:int myvar__attribute__((section("mydata"))) = 0;

表示定义了整形变量myvar=0;并且将该变量存放到名为”mydata”的section中

关于gcc_attribute详解可以参考:http://blog.sina.com.cn/s/blog_661314940100qujt.html

2、代码分析:

举例说明:若要导出内核符号(内核函数)myfc,

如调用           EXPORT_SYMBOL(myfc)

展开为           __EXPORT_SYMBOL(myfc,””)

展开为

static const char __kstrtab_myfc[]                                                                                           __attribute__((section(“__ksymtab_strings”),aligned(1)))

=MODULE_SYMBOL_PREFIX  myfc;

static const struct kernel_symbol __ksymtab_myfc

__used

__attribute__((section(“__ksymatab”),unsed))

={(unsigned long)&sym,_kstrab_myfc}

由前面可知__attribute__是gcc中的属性(__used也是gcc属性),用于向指定的函数或者变量添加相关的属性,为了不影响对变量定义的理解,先将__attribute__属性掩盖,则上面的定义变为:

static const char __kstrtab_myfc[]       =” myfc”;

static const struct kernel_symbol__ksymtab_myfc={(unsigned long)&myfc,_kstrab_myfc};

//定义了一个字符数组__kstrtab_myfc[]用于存放导出的符号名myfc

//定义了一个内核符号结构__ksymtab_myfc用于存放引出符号myfc在内存中的地址和名称。

添加了__attribute__属性后,则表示:

将字符数组__kstrtab_myfc[]放置到一个名为“__ksymtab_strings”的section中。

将内核符号结构__ksymtab_myfc放置到一个名为”__ksymatab”的section中。

若是调用了EXPORT_SYMBOL_GPL(myfc),则对应的内核符号结构被放置到名 为”__ksymatab_gpl”的section中。

3、总结:在内核符号导出中,调用了EXPORT_SYMBOL(sym),则会完成以下操作:

(1)   定义一个字符数组存放内核导出符号的名称,并放置到“__ksymtab_strings”的section中。

(2)   定义一个内核符号结构用于存放导出符号的内存地址和名称,并放置到”__ksymatab”中。

即通过EXPORT_SYMBOL(sym)告诉了内核以外的世界关于这个符号的两点信息:内核符号的名称和其内存地址。

Linux内核导出符号宏定义EXPORT_SYMBOL的源码分析

时间: 2024-08-04 15:25:10

Linux内核导出符号宏定义EXPORT_SYMBOL的源码分析的相关文章

Linux驱动修炼之道-SPI驱动框架源码分析(上)【转】

转自:http://blog.csdn.net/lanmanck/article/details/6895318 SPI驱动架构,以前用过,不过没这个详细,跟各位一起分享: 来自:http://blog.csdn.net/woshixingaaa/article/details/6574215 SPI协议是一种同步的串行数据连接标准,由摩托罗拉公司命名,可工作于全双工模式.相关通讯设备可工作于m/s模式.主设备发起数据帧,允许多个从设备的存在.每个从设备 有独立的片选信号,SPI一般来说是四线串

内核中的宏定义__init、__initdata和__exit、__exitdata

__init.__initdata和__exit.__exitdata的定义位于<kernel/include/linux/init.h> 1 /* These are for everybody (although not all archs will actually 2 discard it in modules) */ 3 #define __init __section(.init.text) __cold notrace 4 #define __initdata __section

Linux内核源码分析方法

  一.内核源码之我见 Linux内核代码的庞大令不少人“望而生畏”,也正因为如此,使得人们对Linux的了解仅处于泛泛的层次.如果想透析Linux,深入操作系统的本质,阅读内核源码是最有效的途径.我们都知道,想成为优秀的程序员,需要大量的实践和代码的编写.编程固然重要,但是往往只编程的人很容易把自己局限在自己的知识领域内.如果要扩展自己知识的广度,我们需要多接触其他人编写的代码,尤其是水平比我们更高的人编写的代码.通过这种途径,我们可以跳出自己知识圈的束缚,进入他人的知识圈,了解更多甚至我们一

Linux内核源码分析--内核启动之(4)Image内核启动(setup_arch函数)(Linux-3.0 ARMv7)【转】

原文地址:Linux内核源码分析--内核启动之(4)Image内核启动(setup_arch函数)(Linux-3.0 ARMv7) 作者:tekkamanninja 转自:http://blog.chinaunix.net/uid-25909619-id-4938393.html 在分析start_kernel函数的时候,其中有构架相关的初始化函数setup_arch. 此函数根据构架而异,对于ARM构架的详细分析如下: void __init setup_arch(char **cmdlin

《linux 内核完全剖析》sched.c sched.h 代码分析笔记

sched.c sched.h 代码分析笔记 首先上header file sched.h #ifndef _SCHED_H #define _SCHED_H #define HZ 100 #define NR_TASKS 64 #define TASK_SIZE 0x04000000 #define LIBRARY_SIZE 0x00400000 #if (TASK_SIZE & 0x3fffff) #error "TASK_SIZE must be multiple of 4M&qu

Linux内核源码分析--内核启动之(5)Image内核启动(rest_init函数)(Linux-3.0 ARMv7)【转】

原文地址:Linux内核源码分析--内核启动之(5)Image内核启动(rest_init函数)(Linux-3.0 ARMv7) 作者:tekkamanninja 转自:http://blog.chinaunix.net/uid-25909619-id-4938395.html 前面粗略分析start_kernel函数,此函数中基本上是对内存管理和各子系统的数据结构初始化.在内核初始化函数start_kernel执行到最后,就是调用rest_init函数,这个函数的主要使命就是创建并启动内核线

Linux内核源码分析--内核启动之(3)Image内核启动(C语言部分)(Linux-3.0 ARMv7) 【转】

原文地址:Linux内核源码分析--内核启动之(3)Image内核启动(C语言部分)(Linux-3.0 ARMv7) 作者:tekkamanninja 转自:http://blog.chinaunix.net/uid-25909619-id-4938390.html 在构架相关的汇编代码运行完之后,程序跳入了构架无关的内核C语言代码:init/main.c中的start_kernel函数,在这个函数中Linux内核开始真正进入初始化阶段, 下面我就顺这代码逐个函数的解释,但是这里并不会过于深入

源码分析:动态分析 Linux 内核函数调用关系

源码分析:动态分析 Linux 内核函数调用关系 时间 2015-04-22 23:56:07  泰晓科技 原文  http://www.tinylab.org/source-code-analysis-dynamic-analysis-of-linux-kernel-function-calls/ 主题 Linux源码分析 By Falcon ofTinyLab.org 2015/04/18 缘由 源码分析是程序员离不开的话题. 无论是研究开源项目,还是平时做各类移植.开发,都避免不了对源码的

Linux内核源码分析--内核启动之(6)Image内核启动(do_basic_setup函数)(Linux-3.0 ARMv7)【转】

原文地址:Linux内核源码分析--内核启动之(6)Image内核启动(do_basic_setup函数)(Linux-3.0 ARMv7) 作者:tekkamanninja 转自:http://blog.chinaunix.net/uid-25909619-id-4938396.html 在基本分析完内核启动流程的之后,还有一个比较重要的初始化函数没有分析,那就是do_basic_setup.在内核init线程中调用了do_basic_setup,这个函数也做了很多内核和驱动的初始化工作,详解