全部博文(685)
分类: 嵌入式
2014-05-02 12:40:34
在C源码中,使用了一个函数:module_init(init_hw),该函数的参数是另外一个函数的指针,这个函数将告诉内核,模块初始化时该调用哪个函数。当使用命令insmod插入模块时,内核将自动调用指定的初始话函数,因此才会在日志信息中打印出“Hello, I'm Octagram.”,初始化函数的声明应该是:
[cpp]view plaincopy
“__init”表示这个函数是初始化函数,将被放到内存内核空间的初始化段。这个符号在Linux内核源码中有定义成:
//“内核目录/include/linux/init.h”中定义
#define __init __section(.init.text) __cold notrace
相关宏的定义:
//“内核目录/include/linux/compiler.h”中定义
#define __section(S) __attribute__ ((__section__(#S)))
这是gcc编译器特有的语法,表示函数具有“.init.text”段的属性,其中“.text”表示代码段。之后由连接器ld会按照内核lds脚本提供的信息,将具有相同段属性的代码放在一起,形成一个连续的内存区域。
“__attribute__”是gcc的一个关键字,用于告诉gcc,代码或数据具有怎么样的属性,“__section__(#S)”表示代码或数据在哪一个段,连接器ld连接程序时就会将这些代码或数据放到特定的位置,“#S”是一种不常用的C语言语法,表示将S代表的内容转换为字符串,例如#.init.text等价于".init.text"。
//“内核目录/include/linux/compiler-gcc4.h”中定义
#define __cold __attribute__((__cold__))
“__cold__”同样是gcc的一个关键字,有关内容在“内核目录/include/linux/compiler-gcc4.h”有一段描述:
Mark functions as cold. gcc will assume any path leading to a call to them will be unlikely. This means a lot of manual unlikely()s are unnecessary now for any paths leading to the usual suspects like BUG(), printk(), panic() etc. [but let's keep them for now for older compilers.
Early snapshots of gcc 4.3 don't support this and we can't detect this in the preprocessor, but we can live with this because they're unreleased. Maketime probing would be overkill here. gcc also has a __attribute__((__hot__)) to move hot functions into a special section, but I don't see any sense in this right now in the kernel context.
//“内核目录/include/linux/compiler.h”中定义
#define notrace __attribute__((no_instrument_function))
“no_instrument_function”在gcc的手册中有解释:
A function may be given the attribute no_instrument_function, in which case this instrumentation will not be done. This can be used, for example, for the profiling functions listed above, high-priority interrupt routines, and any functions from which the profiling functions cannot safely be called (perhaps signal handlers, if the profiling routines generate output or allocate memory).
3.5 清理函数
类似初始化函数,模块从内核中删除时需要将调用清理函数完成清理工作,由module_exit()函数指定模块的清理函数。当使用命令rmmod清除模块时,内核会自动调用模块的清理函数,因此才能在日志中看到“Goodbye!”,清理函数的声明应该是:
static void __exit cleanup_function(void)
{
/*Cleanup code here */
}
相关关宏的定义:
“__exit”和“__init”类似,在内核中被定义成:
//“内核目录/include/linux/init.h”中定义
#define __exit __section(.exit.text) __exitused __cold
表示函数具有段属性“.exit”,该段同样是位于内核空间的一段内存中,在这段内存中集中了所有用于卸载模块的函数,如果模块直接建立在内核里,或者内核配置成不允许模块卸载,这个函数可以省略,如果模块没有定义一个清理函数,内核不会允许它被卸载。一个标识__exit的函数只在模块卸载或者系统停止时调用,任何别的情况,调用都是错误的。
//“内核目录/include/linux/init.h”中定义
#ifdef MODULE
#define __exitused
#else
#define __exitused __used
#endif
关于“__used”的说明在“内核目录/include/linux/compiler.h”中有一段:
Allow us to avoid 'defined but not used' warnings on functions and data, as well as force them to be emitted to the assembly file. As of gcc 3.4, static functions that are not marked with attribute((used)) may be elided from the assembly file. As of gcc 3.4, static data not so marked will not be elided, but this may change in a future gcc version. NOTE: Because distributions shipped with a backported unit-at-a-time compiler in gcc 3.3, we must define __used to be __attribute__((used)) for gcc >=3.3 instead of 3.4. In prior versions of gcc, such functions and data would be emitted, but would be warned about except with attribute((unused)). Mark functions that are referenced only in inline assembly as __used so the code is emitted even though it appears to be unreferenced.
这个宏是防止gcc提出“定义而未使用”的警告的。
4 Makefile注释
(1) $(MAKE) -C $(KERNELDIR) M=$(PWD) modules
“$(MAKE)”表示再一次执行make指令,但是这次执行将通过选项“-C”切换到目录“$(KERNELDIR)”,也即是内核源代码目录下去执行,这样读取的就会是内核的Makefile,执行的目标是“modules”,但是如果就这样开始执行了,生成的目标文件就会跑到内核目录下,因此还需要只用“M=$(PWD)”选项切换到当前目录再输出。
(2) CROSS_COMPILE ?=
指定交叉编译的工具链前缀,如果这项不为空,make会调用对应的工具链来编译,产生非PC端的程序。
(3) obj-m := hw.o
在内核的Makefile中,规定:如果要编译一个模块,需要将模块对应的obj文件放在obj-m中,如果是在编译内核时要将模块直接编译进内核中,需要将模块对应的obj文件放在obj-y中。如果一个模块名为 module.ko,来自2个源文件(姑且称之为, file1.c 和 file2.c)正确的书写应当是:
obj-m := module.o
module-objs := file1.o file2.o
5 如何产生ARM端的模块程序
5.1 修改源码
上面的程序由于使用了ubuntu的内核目录,所以产生的是PC的程序,但是我们需要产生ARM端的程序,这个只需要修改Makefile就可以了。新的Makefile如下:
Makefile文件
[cpp]view plaincopy
(1) KERNELDIR ?=
重新指定内核目录的路径,因为这回要使用ARM的内核源码而非PC的,上面的这个路径是为OMAP3530移植的内核目录。
(2) CROSS_COMPILE ?=
指定交叉编译工具链的路径和前缀,因为ARM和PC是完全不同的体系,因此同一个程序产生的二进制机器码也是不同的,所以需要对应的编译工具。PC端的为空是因为PC的工具链存放在系统路径下,能被自动搜索到,而且PC的工具链没有前缀。
5.2 开始编译
在模块源代码目录下执行下面命令:
$ make
(1) 第一次编译出现以下错误:
ERROR: Kernel configuration is invalid.
include/generated/autoconf.h or include/config/auto.conf are missing.
Run 'make oldconfig && make prepare' on kernel src to fix it.
根据提示在内核目录下(而非模块源代码目录)执行以下命令可解决:
$ make octagram-ocsom3530_defconfig
$ make prepare
这里没有执行oldconfig是因为那不是我的开发板的配置文件,这个配置文件需要使用make menuconfig命令,然后按照开发板的情况进行配置,会自动生成隐藏文件.config,最后将配置文件复制到arch/arm/configs目录下,改名做备份,这个属于内核移植时的工作。
(2) 第二次编译出现以下错误:
MODPOST 1 modules
/bin/sh: scripts/mod/modpost: not found
这是因为没有编译scripts造成的。在内核目录下(而非模块源代码目录)执行下面命令:
$ make scripts
(3) 还有一个警告信息
WARNING: Symbol version dump /home/octagram/workspace/ARM/OMAP3530/linux-2.6.36.3/Module.symvers
is missing; modules will have no dependencies and modversions.
这是因为没有编译内核造成的。在内核目录下(而非模块源代码目录)执行下面命令:
$ make