Smart210学习记录-------linux内核模块

Linux 驱动工程师需要牢固地掌握 Linux 内核的编译方法以为嵌入式系统构建可运行的
Linux 操作系统映像。在编译 LDD6410 的内核时,需要配置内核,可以使用下面命令中的
一个:
#make config(基于文本的最为传统的配置界面,不推荐使用)
#make menuconfig(基于文本菜单的配置界面)
#make xconfig(要求 QT 被安装)
#make gconfig(要求 GTK+被安装)
在配置Linux 2.6内核所使用的make config、make menuconfig、make xconfig和make gconfig
这 4 种方式中,最值得推荐的是 make menuconfig,它不依赖于 QT 或 GTK+,且非常直观,对
LDD6410 的 Linux 2.6.28 内核运行 make menuconfig

内核配置包含的项目相当多,arch/arm/configs/ldd6410lcd_defconfig 文件包含了 LDD6410 的
默认配置,因此,只需要运行 make ldd6410lcd_defconfig 就可以为 LDD6410 开发板配置内核。
编译内核和模块的方法是:
make zImage
make modules
执行完上述命令后,在源代码的根目录下会得到未压缩的内核映像 vmlinux 和内核符号表文
件 System.map,在 arch/arm/boot/目录会得到压缩的内核映像 zImage,在内核各对应目录得到选中
的内核模块。

Linux 2.6 内核的配置系统由以下 3 个部分组成。
! Makefile:分布在 Linux 内核源代码中的 Makefile,定义 Linux 内核的编译规则。
! 配置文件(Kconfig):给用户提供配置选择的功能。
! 配置工具:包括配置命令解释器(对配置脚本中使用的配置命令进行解释)和配置用户
界面(提供基于字符界面和图形界面)。这些配置工具都是使用脚本语言,如 Tcl/TK、
Perl 等编写。

Kconfig Makfile
在 Linux 内核中增加程序需要完成以下 3 项工作。
! 将编写的源代码拷入 Linux 内核源代码的相应目录。
! 在目录的 Kconfig 文件中增加关于新源代码对应项目的编译配置选项。
! 在目录的 Makefile 文件中增加对新源代码的编译条目。

一个 Linux 内核模块主要由如下几个部分组成。
(1)模块加载函数(一般需要) 。
当通过 insmod 或 modprobe 命令加载内核模块时,模块的加载函数会自动被内核执行,完成
本模块的相关初始化工作。
(2)模块卸载函数(一般需要) 。
当通过 rmmod 命令卸载某模块时,模块的卸载函数会自动被内核执行,完成与模块卸载函数
相反的功能。
(3)模块许可证声明(必须) 。
许可证(LICENSE)声明描述内核模块的许可权限,如果不声明 LICENSE,模块被加载时,
将收到内核被污染 (kernel tainted)的警告。
在Linux 2.6内核中,可接受的LICENSE包括“GPL” 、 “GPL v2” 、 “GPL and additional rights” 、
“Dual BSD/GPL” 、 “Dual MPL/GPL”和“Proprietary” 。
大多数情况下,内核模块应遵循 GPL 兼容许可权。Linux 2.6 内核模块最常见的是以
MODULE_LICENSE( "Dual BSD/GPL" )语句声明模块采用 BSD/GPL 双 LICENSE。
(4)模块参数(可选) 。
模块参数是模块被加载的时候可以被传递给它的值,它本身对应模块内部的全局变量。
(5)模块导出符号(可选) 。
内核模块可以导出符号(symbol,对应于函数或变量),这样其他模块可以使用本模块中的变
量或函数。
(6)模块作者等信息声明(可选) 。

**********************************************************************************************
模块加载函数

Linux 内核模块加载函数一般以_ _init 标识声明,典型的模块加载函数的形式如代码清单
static int __init initialization_function(void)
{
...
}
module_init(initialization_function);

模块加载函数必须以“module_init(函数名)”的形式被指定。它返回整型值,若初始化成功,
应返回 0。而在初始化失败时,应该返回错误编码。在 Linux 内核里,错误编码是一个负值,在
<linux/errno.h>中定义,包含-ENODEV、-ENOMEM 之类的符号值。总是返回相应的错误编码是
种非常好的习惯,因为只有这样,用户程序才可以利用 perror 等方法把它们转换成有意义的错误
信息字符串。
在 Linux 2.6 内核中,可以使用 request_module(const char *fmt, …)函数加载内核模块,驱动开
发人员可以通过调用
request_module(module_name);

request_module("char-major-%d-%d", MAJOR(dev), MINOR(dev));
这种灵活的方式加载其他内核模块。
在 Linux 中,所有标识为_ _init 的函数在连接的时候都放在.init.text 这个区段内,此外,所有
的_ _init 函数在区段.initcall.init 中还保存了一份函数指针,在初始化时内核会通过这些函数指针
调用这些_ _init 函数,并在初始化完成后,释放 init 区段(包括.init.text、.initcall.init 等)。

**********************************************************************************************

**********************************************************************************************
模块卸载函数
Linux 内核模块加载函数一般以_ _exit 标识声明,典型的模块卸载函数的形式如代码清单

static void __exit cleanup_function(void)
{
...
}
module_exit(cleanup_function);

模块卸载函数在模块卸载的时候执行,不返回任何值,必须以“module_exit(函数名)”的形
式来指定。
通常来说,模块卸载函数要完成与模块加载函数相反的功能,如下所示。
! 若模块加载函数注册了 XXX,则模块卸载函数应该注销 XXX。
! 若模块加载函数动态申请了内存,则模块卸载函数应释放该内存。
! 若模块加载函数申请了硬件资源(中断、DMA 通道、I/O 端口和 I/O 内存等)的占用,
则模块卸载函数应释放这些硬件资源。
! 若模块加载函数开启了硬件,则卸载函数中一般要关闭之。
和_ _init一样,_ _exit也可以使对应函数在运行完成后自动回收内存。实际上,_ _init和_ _exit
都是宏,其定义分别为:
#define _ _init _ _attribute_ _ ((_ _section_ _ (".init.text")))

#ifdef MODULE
#define _ _exit _ _attribute_ _ ((_ _section_ _(".exit.text")))
#else
#define _ _exit _ _attribute_used_ _attribute_ _ ((_ _section_ _(".exit.text")))
#endif
数据也可以被定义为_ _initdata 和_ _exitdata,这两个宏分别为:
#define _ _initdata _ _attribute_ _ ((_ _section_ _ (".init.data")))

#define _ _exitdata _ _attribute_ _ ((_ _section_ _(".exit.data")))

**********************************************************************************************

模块参数
我们可以用“module_param(参数名,参数类型,参数读/写权限)”为模块定义一个参数,例如下
列代码定义了 1 个整型参数和 1 个字符指针参数:

static char* pr = "hello world";
static int num = 5;
module_param(pr, charp, S_IRUGO);
module_param(num, int, S_IRUGO);

在装载内核模块时,用户可以向模块传递参数,形式为“insmode(或 modprobe)模块名 参
数名=参数值” ,如果不传递,参数将使用模块内定义的缺省值。
参数类型可以是byte、short、ushort、int、uint、long、ulong、charp(字符指针)、bool或invbool
(布尔的反),在模块被编译时会将 module_param 中声明的类型与变量定义的类型进行比较,判断
是否一致。
模块被加载后,在/sys/module/目录下将出现以此模块名命名的目录。当“参数读/写权限”为 0
时,表示此参数不存在 sysfs 文件系统下对应的文件节点,如果此模块存在“参数读/写权限”不为 0
的命令行参数,在此模块的目录下还将出现parameters目录,包含一系列以参数名命名的文件节点,
这些文件的权限值就是传入module_param()的“参数读/写权限” ,而文件的内容为参数的值。

除此之外,模块也可以拥有参数数组,形式为“module_param_array(数组名,数组类型,数组长,参数
读/写权限)” 。从2.6.0~2.6.10版本,需将数组长变量名赋给“数组长” ,从2.6.10 版本开始,需将数组
长变量的指针赋给“数组长” ,当不需要保存实际输入的数组元素个数时,可以设置“数组长”为NULL。
运行 insmod 或 modprobe 命令时,应使用逗号分隔输入的数组元素。

**********************************************************************************************
导出符号

Linux 2.6的“/proc/kallsyms”文件对应着内核符号表,它记录了符号以及符号所在的内存地址。
模块可以使用如下宏导出符号到内核符号表:
EXPORT_SYMBOL(符号名);
EXPORT_SYMBOL_GPL(符号名);
导出的符号将可以被其他模块使用,使用前声明一下即可

**********************************************************************************************

模块声明与描述
在 Linux 内核模块中,我们可以用 MODULE_AUTHOR、MODULE_DESCRIPTION、
MODULE_VERSION、MODULE_DEVICE_TABLE、MODULE_ALIAS 分别声明模块的作者、描
述、版本、设备表和别名,例如:
MODULE_AUTHOR(author);
MODULE_DESCRIPTION(description);
MODULE_VERSION(version_string);
MODULE_DEVICE_TABLE(table_info);
MODULE_ALIAS(alternate_name);
http://weibo.com/21cnbao

**********************************************************************************************
模块的使用计数

try_module_get ()与 module_put()的引入与使用与Linux 2.6内核下的设备模型密切相关。Linux
2.6 内核为不同类型的设备定义了 struct module *owner 域,用来指向管理此设备的模块。当开始
使用某个设备时,内核使用 try_module_get(dev->owner)去增加管理此设备的 owner 模块的使用计
数;当不再使用此设备时,内核使用 module_put(dev->owner)减少对管理此设备的 owner 模块的
使用计数。这样,当设备在使用时,管理此设备的模块将不能被卸载。只有当设备不再被使用时,
模块才允许被卸载。
在 Linux 2.6 内核下,对于设备驱动工程师而言,很少需要亲自调用 try_module_get()与
module_put(),因为此时开发人员所写的驱动通常为支持某具体设备的owner模块,对此设备owner
模块的计数管理由内核里更底层的代码如总线驱动或是此类设备共用的核心模块来实现,从而简
化了设备驱动开发。

详情请看Linucx设备驱动开发

时间: 2024-10-18 06:29:37

Smart210学习记录-------linux内核模块的相关文章

Smart210学习记录-----Linux i2c驱动

一:Linux i2c子系统简介: 1.Linux 的 I2C 体系结构分为 3 个组成部分: (1) I2C 核心. I2C 核心提供了 I2C 总线驱动和设备驱动的注册.注销方法,I2C 通信方法(即“algorithm”)上层的.与具体适配器无关的代码以及探测设备.检测设备地址的上层代码等. (2) I2C 总线驱动. I2C 总线驱动是对 I2C 硬件体系结构中适配器端的实现,适配器可由 CPU 控制,甚至可以直接集成在 CPU 内部. I2C 总线驱动主要包含了 I2C 适配器数据结构

Smart210学习记录------linux串口驱动

转自:http://blog.chinaunix.net/xmlrpc.php?r=blog/article&uid=27025492&id=327609 一.核心数据结构 串口驱动有3个核心数据结构,它们都定义在<#include linux/serial_core.h> 1.uart_driver uart_driver包含了串口设备名.串口驱动名.主次设备号.串口控制台(可选)等信息,还封装了tty_driver(底层串口驱动无需关心tty_driver). struct

Smart210学习记录-----linux定时器

1.内核定时器: Linux 内核所提供的用于操作定时器的数据结构和函数如下: (1) timer_list 在 Linux 内核中,timer_list 结构体的一个实例对应一个定时器 1 struct timer_list { 2    struct list_head entry; /* 定时器列表 */ 3    unsigned long expires; /*定时器到期时间*/ 4    void (*function)(unsigned long); /* 定时器处理函数 */ 5

Smart210学习记录-------linux驱动中断

Linux中断 1.申请和释放中断 申请中断 int request_irq(unsigned int irq, irq_handler_t handler,  unsigned long irqflags, const char *devname, void *dev_id) irq 是要申请的硬件中断号. handler 是向系统登记的中断处理函数(顶半部),是一个回调函数,中断发生时,系统调用 这个函数, dev_id 参数将被传递给它. irqflags是中断处理的属性,可以指定中断的触

Smart210学习记录------块设备

转自:http://bbs.chinaunix.net/thread-2017377-1-1.html 本章的目的用尽可能最简单的方法写出一个能用的块设备驱动.所谓的能用,是指我们可以对这个驱动生成的块设备进行mkfs,mount和读写文件.为了尽可能简单,这个驱动的规模不是1000行,也不是500行,而是100行以内. 这里插一句,我们不打算在这里介绍如何写模块,理由是介绍的文章已经满天飞舞了.如果你能看得懂.并且成功地编译.运行了这段代码,我们认为你已经达到了本教程的入学资格,当然,如果你不

我的学习记录--Linux (CentOS) 程序安装包管理,yum

此博客,仅仅只是记录本人学习Linux的学习笔记,和学习经验,本人此时也只是一个初学Linux的菜鸟,所以有写得不对的地方还望包涵.谢谢! 简介: Yum(全称为 Yellow dog Updater, Modified)是一个在Fedora和RedHat以及CentOS中的Shell前端软件包管理器.基于RPM包管理,能够从指定的服务器自动下载RPM包并且安装,可以自动处理依赖性关系,并且一次安装所有依赖的软件包,无须繁琐地一次次下载.安装. yum配置文件: 主配置文件:/etc/yum.c

Smart210学习记录-----SD/MMC/SDIO驱动

转自:http://jingpin.jikexueyuan.com/article/23369.html 一.SD/MMC/SDIO概念区分 SD(SecureDigital)与 MMC(MultimediaCard) SD 是一种 flash memory card 的标准,也就是一般常见的 SD 记忆卡,而 MMC 则是较早的一种记忆卡标准,目前已经被 SD 标准所取代.在维基百科上有相当详细的 SD/MMC 规格说明:[http://zh.wikipedia.org/wiki/Secure

我的学习记录--Linux (CentOS) 程序安装包管理,rpm

概述: 因为一个应用程序是由很多文件所组成,所以安装卸载非常复杂,为了让用户更加方便的管理应用程序.所以引入了程序打包管理.主要包括:安装,卸载,升级,查询,检验. 常见的包管理器: .deb: Debian研发 .rpm: Red Hat研发(Linux上的包管理器公共标准,故我学习的为rpm) 常见的CPU平台:(所以选择应用程序安装的时候,一定选择和自己CPU所匹配的版本) x86:i386, i486, i586, i686 x86_64: x86_64, amd64 分包机制: 核心包

Smart210学习记录------nor flash驱动

nor flash驱动与nand flash驱动的差别不大,只是设置不同的结构体而已,, nor flash驱动代码: #include <linux/module.h> #include <linux/types.h> #include <linux/kernel.h> #include <linux/init.h> #include <linux/slab.h> #include <linux/device.h> #include