一、linux内核模块简介

linux内核整体结构非常庞大,其包含的组件也非常多。我们怎么把需要的部分都包含在内核中呢?

     一种办法是把所有的需要的功能都编译到内核中。这会导致两个问题,一是生成的内核会很大,二是如果我们要在现有的内核中新增或删除功能,不得不重新编译内核,工作效率会非常的低,同时如果编译的模块不是很完善,很有可能会造成内核崩溃。

    linux提供了另一种机制来解决这个问题,这种集中被称为模块,可以实现编译出的内核本身并不含有所有功能,而在这些功能需要被使用的时候,其对应的代码可以被动态的加载到内核中。

二、模块特点:

    1)模块本身并不被编译入内核,从而控制了内核的大小。

    2)模块一旦被加载,他就和内核中的其他部分完全一样。

    注意:模块并不是驱动的必要形式:即:驱动不一定必须是模块,有些驱动是直接编译进内核的;同时模块也不全是驱动,例如我们写的一些很小的算法可以作为模块编译进内核,但它并不是驱动。就像烧饼不一定是圆的,圆的也不都是烧饼一样。

三、最简单的模块分析

1)以下是一个最简单的模块例子

[cpp] view plaincopy

  1. #include <linux/init.h>         /* printk() */

  2. #include <linux/module.h>       /* __init __exit */

  3. static int  __init  hello_init(void)      /*模块加载函数,通过insmod命令加载模块时,被自动执行*/

  4. {

  5. printk(KERN_INFO " Hello World enter\n");

  6. return 0;

  7. }

  8. static void  __exit  hello_exit(void)    /*模块卸载函数,当通过rmmod命令卸载时,会被自动执行*/

  9. {

  10. printk(KERN_INFO " Hello World exit\n ");

  11. }

  12. module_init(hello_init);

  13. module_exit(hello_exit);

  14. MODULE_AUTHOR("dengwei");           /*模块作者,可选*/

  15. MODULE_LICENSE("Dual BSD/GPL");     /*模块许可证明,描述内核模块的许可权限,必须*/

  16. MODULE_DESCRIPTION("A simple Hello World Module"); /*模块说明,可选*/

  17. MODULE_ALIAS("a simplest module");                  /*模块说明,可选*/<span style="font-family:SimSun;font-size:18px;color:#FF0000;"><strong>

  18. </strong></span>

2) 以下是编译上述模块所需的编写的makefile

[cpp] view plaincopy

  1. obj-m :=hello.o                     //目标文件

  2. #module-objs := file1.o file.o      //当模块有多个文件组成时,添加本句

  3. KDIR :=/usr/src/linux               //内核路径,根据实际情况换成自己的内核路径,嵌入式的换成嵌入式,PC机的指定PC机路径

  4. PWD := $(shell pwd)                 //模块源文件路径

  5. all:

  6. $(MAKE)  -C  $(KDIR)  SUBDIRS=$(PWD)  modules

  7. @rm -rf *.mod.*

  8. @rm -rf .*.cmd

  9. @rm -rf *.o

  10. @rm -rf Module.*

  11. clean:

  12. rm -rf *.ko

最终会编译得到:hello.ko文件

使用insmodhello.ko将模块插入内核,然后使用dmesg即可看到输出提示信息。


常用的几种模块操作:

insmod XXX.ko    加载指定模块

lsmod                      列举当前系统中的所有模块

rmmod  XXX         卸载指定模块(注意没有.ko后缀)

dmesg                    当打印等级低于默认输出等级时,采用此命令查看系统日志

3)linux内核模块的程序结构

1.模块加载函数:

Linux内核模块一般以__init标示声明,典型的模块加载函数的形式如下:

[cpp] view plaincopy

  1. static int __init myModule_init(void)

  2. {

  3. /* Module init code */

  4. PRINTK("myModule_init\n");

  5. return 0;

  6. }

  7. module_init(myModule_init);

模块加载函数的名字可以随便取,但必须以“module_init(函数名)”的形式被指定;

执行insmod命令时被执行,用于初始化模块所必需资源,比如内存空间、硬件设备等;

它返回整形值,若初始化成功,应返回0,初始化失败返回负数。

2.模块卸载函数

典型的模块卸载函数形式如下:

[cpp] view plaincopy

  1. static void __exit myModule_exit(void)

  2. {

  3. /* Module exit code */

  4. PRINTK("myModule_exit\n");

  5. return;

  6. }

  7. module_exit(myModule_exit);

模块卸载函数在模块卸载的时候执行,不返回任何值,需用”module_exit(函数名)”的形式被指定。

卸载模块完成与加载函数相反的功能:

   若加载函数注册了XXX,则卸载函数应当注销XXX

   若加载函数申请了内存空间,则卸载函数应当释放相应的内存空间

   若加载函数申请了某些硬件资源(中断、DMA、I/0端口、I/O内存等),则卸载函数应当释放相应的硬件资源

   若加载函数开启了硬件,则卸载函数应当关闭硬件。


其中__init 、__exit 为系统提供的两种宏,表示其所修饰的函数在调用完成后会自动回收内存,即内核认为这种函数只会被执行1次,然后他所占用的资源就会被释放。


3.模块声明与描述

    在linux内核模块中,我们可以用MODULE_AUTHOR、MODULE_DESCRIPTION、MODULE_VERSION、MODULE_TABLE、MODULE_ALIA,分别描述模块的作者、描述、版本、设备表号、别名等。

[cpp] view plaincopy

  1. MODULE_AUTHOR("dengwei");

  2. MODULE_LICENSE("Dual BSD/GPL");

  3. MODULE_DESCRIPTION("A simple Hello World Module");

  4. MODULE_ALIAS("a simplest module");

四、有关模块的其它特性

1)模块参数:

我们可以利用module_param(参数名、参数类型、参数读写属性) 为模块定义一个参数,例如:

[cpp] view plaincopy

  1. static char *string_test = “this is a test”;

  2. static num_test = 1000;

  3. module_param (num_test,int,S_IRUGO);

  4. module_param (steing_test,charp,S_ITUGO);

    在装载模块时,用户可以给模块传递参数,形式为:”insmod 模块名 参数名=参数值”,如果不传递,则参数使用默认的参数值

    参数的类型可以是:byte,short,ushort,int,uint,long,ulong,charp,bool;

    权限:定义在linux/stat.h中,控制存取权限,S_IRUGO表示所有用户只读;

    模块被加载后,在sys/module/下会出现以此模块命名的目录,当读写权限为零时:表示此参数不存在sysfs文件系统下的文件节点,当读写权限不为零时:此模块的目录下会存在parameters目录,包含一系列以参数名命名的文件节点,这些文件节点的权限值就是传入module_param()的“参数读/写权限“,而该文件的内容为参数的值。

    除此之外,模块也可以拥有参数数组,形式为:”module_param_array(数组名、数组类型、数组长、参数读写权限等)”,当不需要保存实际的输入的数组元素的个数时,可以设置“数组长“为0。

    运行insmod时,使用逗号分隔输入的数组元素。

下面是一个实际的例子,来说明模块传参的过程。

[cpp] view plaincopy

  1. #include <linux/module.h>    /*module_init()*/

  2. #include <linux/kernel.h> /* printk() */

  3. #include <linux/init.h>       /* __init __exit */

  4. #define DEBUG   //open debug message

  5. #ifdef DEBUG

  6. #define PRINTK(fmt, arg...)     printk(KERN_WARNING fmt, ##arg)

  7. #else

  8. #define PRINTK(fmt, arg...)     printk(KERN_DEBUG fmt, ##arg)

  9. #endif

  10. static char *string_test="default paramater";

  11. static int num_test=1000;

  12. static int __init hello_init(void)

  13. {

  14. PRINTK("\nthe  string_test is : %s\n",string_test);

  15. PRINTK("the  num_test is : %d\n",num_test);

  16. return 0;

  17. }

  18. static void __exit hello_exit(void)

  19. {

  20. PRINTK(" input paramater module exit\n ");

  21. }

  22. module_init(hello_init);

  23. module_exit(hello_exit);

  24. module_param(num_test,int,S_IRUGO);

  25. module_param(string_test,charp,S_IRUGO);

  26. MODULE_AUTHOR("dengwei");

  27. MODULE_LICENSE("GPL");

当执行 insmod hello_param.ko时,执行dmesg 查看内核输出信息:

[cpp] view plaincopy

  1. Hello World enter

  2. the test string is: this is a test

  3. the test num is :1000


当执行insmod  hello_param.ko num_test=2000 string_test=“edit by dengwei”,执行dmesg查看内核输出信息:

[cpp] view plaincopy

  1. Hello World enter

  2. the test string is: edit by dengwei

  3. the test num is :2000

2)导出模块及符号的相互引用

Linux2.6内核的“/proc/kallsyms“文件对应内核符号表,它记录了符号以及符号所在的内存地址,模块可以使用下列宏导到内核符号表中。

EXPORT_SYMBOL(符号名);       任意模块均可

EXPORT_SYMBOL_GPL(符号名);   只使用于包含GPL许可权的模块

导出的符号可以被其它模块使用,使用前声明一下即可。

下面给出一个简单的例子:将add sub符号导出到内核符号表中,这样其它的模块就可以利用其中的函数

[cpp] view plaincopy

  1. #include <linux/module.h>    /*module_init()*/

  2. #include <linux/kernel.h> /* printk() */

  3. #include <linux/init.h>       /* __init __exit */

  4. int add_test(int a ,int b)

  5. {

  6. return a + b;

  7. }

  8. int sub_test(int a,int b)

  9. {

  10. return a - b;

  11. }

  12. EXPORT_SYMBOL(add_test);

  13. EXPORT_SYMBOL(sub_test);

  14. MODULE_AUTHOR("dengwei");

  15. MODULE_LICENSE("GPL");

执行 cat/proc/kallsyms | grep test 即可找到以下信息,表示模块确实被加载到内核表中。

[cpp] view plaincopy

  1. f88c9008 r __ksymtab_sub_integar        [export_symb]

  2. f88c9020 r __kstrtab_sub_integar         [export_symb]

  3. f88c9018 r __kcrctab_sub_integar         [export_symb]

  4. f88c9010 r __ksymtab_add_integar        [export_symb]

  5. f88c902c r __kstrtab_add_integar          [export_symb]

  6. f88c901c r __kcrctab_add_integar         [export_symb]

  7. f88c9000 T add_tes                [export_symb]

  8. f88c9004 T sub_tes                [export_symb]

  9. 13db98c9 a __crc_sub_integar           [export_symb]

  10. e1626dee a __crc_add_integar           [export_symb]


在其它模块中可以引用此符号

[cpp] view plaincopy

  1. #include <linux/module.h>    /*module_init()*/

  2. #include <linux/kernel.h> /* printk() */

  3. #include <linux/init.h>       /* __init __exit */

  4. #define DEBUG   //open debug message

  5. #ifdef DEBUG

  6. #define PRINTK(fmt, arg...)     printk(KERN_WARNING fmt, ##arg)

  7. #else

  8. #define PRINTK(fmt, arg...)     printk(KERN_DEBUG fmt, ##arg)

  9. #endif

  10. extern int add_test(int a ,int b);

  11. extern int sub_test(int a,int b);

  12. static int __init hello_init(void)

  13. {

  14. int a,b;

  15. a = add_test(10,20);

  16. b = sub_test(30,20);

  17. PRINTK("the add test result is %d",a);

  18. PRINTK("the sub test result is %d\n",b);

  19. return 0;

  20. }

  21. static void __exit hello_exit(void)

  22. {

  23. PRINTK(" Hello World exit\n ");

  24. }

  25. module_init(hello_init);

  26. module_exit(hello_exit);

  27. MODULE_AUTHOR("dengwei");

  28. MODULE_LICENSE("GPL");

linux驱动基础开发2——linux 驱动开发前奏(模块编程)-转相关推荐

  1. Linux入门基础教程之Linux下软件安装

    Linux入门基础教程之Linux下软件安装 一.在线安装: sudo apt-get install 即可安装 如果在安装完后无法用Tab键补全命令,可以执行: source ~/.zshrc AP ...

  2. linux系统初级管理书,Linux系统管理基础--超级适合Linux新手的书

    liuyongqing 于 2012-11-15 11:53:18发表: 谢谢分享 liuyongqing 于 2012-11-15 11:53:11发表: 谢谢分享 liuyongqing 于 20 ...

  3. linux系统基础与应用,Linux操作系统:基础、原理与应用

    <Linux操作系统:基础.原理与应用> 第1部分基础篇 第1章操作系统概述/3 1.1认识操作系统3 1.1.1操作系统的概念3 1.1.2操作系统的功能4 1.2操作系统的发展与现状5 ...

  4. Linux学习基础文章1:Linux一句话精彩问答

    0001 修改主机名(陈绪) vi /etc/sysconfig/network,修改HOSTNAME一行为"HOSTNAME=主机名"(没有这行?那就添加这一行吧),然后运行命令 ...

  5. Linux 学习基础入门之Linux发展史

    [daodu] Linux发展史1. 什么是操作系统我们在使用电脑时候,一般是使用应用程序的,你比如说我现在在Chrome浏览器访问云栖社区.Chrome运行在操作系统上,操作系统驱动硬件,也就是我们 ...

  6. linux系统基础命令使用,linux基础命令(linux必学的60个命令)

    本文主要是讲解Linux系统上最常用.最基本的10个命令. 如果您习惯于通过一个漂亮的图形界面来完成所有的工作,那么在Linux CLI(命令行界面)世界中入门可能会有些困难.对于初学者,有时很难决定 ...

  7. 安装linux出现基础系统出错,Linux系统出错提示信息详解

    ERROR PCI: cannot allocate(无法指派) 这样的错误有许多,他们主要在启动系统时出现.他们有一个共同的起因:错误的电源管理行为.罪魁祸首是一个叫做ACPI的东西,即高级配置与电 ...

  8. 【Linux入门基础知识】Linux 脚本编写基础

    1. Linux 脚本编写基础 1.1 语法基本介绍 1.1.1 开头 程序必须以下面的行开始(必须放在文件的第一行): #!/bin/sh 符号#!用来告诉系统它后面的参数是用来执行该文件的程序.在 ...

  9. linux部分基础命令总结,Linux常用基础命令总结

    近期自己学习了一下Linux,写这篇博客以便于对自己的一个总结,记录自己的学习情况,奥利给! 想对Linux熟练掌握,就必须学会它的操作命令,虽然可能会花费一些时间,不过从长远的角度来说,这的确是一件 ...

  10. linux的基础简答题,Linux认证考试试题及答案「简答题」

    Linux认证考试试题及答案「简答题」 1.论述实时信号.非实时信号.可靠信号.不可靠信号四个概念. 答:实时信号目前未用,非实时信号编号1-31.0表示空信号 1分 非实时信号没有排队机制,可能丢失 ...

最新文章

  1. FastAdmin扩展PHPEXCEL,PHP7.3高版本兼容问题
  2. 【渗透技术】一个渗透测试工具人是怎样操作的
  3. VTK:可视化之ChooseTextColor
  4. 我所理解的设计模式(C++实现)——策略模式(Strategy Pattern)
  5. python scrapy框架爬虫_Python Scrapy爬虫框架
  6. php reactphp wss_Node和React中如何进行实时通信?
  7. 感应联动不是梦,穿透屏幕“闻”见花香你敢信?
  8. python itertools_itertools
  9. spring - ioc和aop
  10. [LevelDB] 编译和使用
  11. 【附PDF下载】2021年上半年信息系统项目管理师上午综合知识真题
  12. B2B网站平台建设:优势、功能、模块三大方向解析
  13. win 10 arm iso 文件下载
  14. 计算机如何分屏操作步骤,win7怎么设置电脑分屏显示|win7分屏显示设置方法
  15. c语言 字符串提取连续数字,c语言一串字符串中提取数字并相加的问题
  16. qq授权登录。微信授权登录、微博授权登录
  17. js正则验证手机号格式
  18. 计嵌 廖峻 20178303040 C++作业
  19. GitHub下载加速利器
  20. listview 的首行固定内容标题且加粗显示(类似于表格的首行)的实现方法

热门文章

  1. 严防ARP病毒的六个步骤
  2. datagridview列 值提取_提取符合条件的多个记录,VLOOKUP:盘他!
  3. qc是什么职位_质量管理部门该干什么?又该怎么干?
  4. 使用XmlPullParser解析XML
  5. Android编程之LocalBroadcastManager源码详解
  6. Eclipse中@author的修改
  7. Greenplum:你不可不知的实施与维护最佳实践
  8. window.location.href跳转无效 IE Bug【转载】
  9. SnagIt的Visual Studio Team System插件
  10. Sudo bug 可导致非权限 Linux 和 MacOS 用户以根身份运行命令