1. Linux - 内核模块

一、什么是内核模块

      内核模块是Linux内核向外部提供的一个插口,其全称为动态可加载内核模块(Loadable Kernel Module,LKM),我们简称为模块。Linux内核之所以提供模块机制,是因为它本身是一个单内核(monolithic kernel)。单内核的最大优点是效率高,因为所有的内容都集成在一起,但其缺点是可扩展性和可维护性相对较差,模块机制就是为了弥补这一缺陷。
      模块是具有独立功能的程序,它可以被单独编译,但不能独立运行。它在运行时被链接到内核作为内核的一部分在内核空间运行,这与运行在用户空间的进程是不同的。模块通常由一组函数和数据结构组成,用来实现一种文件系统、一个驱动程序或其他内核上层的功能。

二、为什么使用内核模块

      在我们使用make menuconfig 配置内核时,可以看到,其实内核其实是由很多组件组成。通过该配置,将需要的选项功能配置进内核文件(zImage);在内核启动过程中,zImage被解压到内存中,相应组件也被解压都内存中了。这样,相对比较占内存空间和耗启动时间。这时,就引入了内核模块的概念了;可以将不需要一开机就必须加载好的模块,不编译进内核,等待系统起来后再加载;由此来缩减系统启动时间;或将不经常使用或极少使用的模块,不编译进内核;等待需要使用时再提早安装进内核,不需要使用时卸载掉,以减少内存空间的使用;

三、安装与卸载内核模块

  • 安装 insmod
insmod module.ko

*卸载 rmmod

rmmod module

*查看内核模块

lsmod

四、内核模块示例

  • 1、编写Makefile
#声明编译的是模块
obj-m := helloworld.o

#内核代码路径
KDIR := /home/work/project/02-source/iMX-Linux

all:
    make -C $(KDIR) M=$(PWD) modules CROSS_COMPILE=arm-linux-gnueabihf- ARCH=arm
    
clean:
    rm -f *.o *.ko *.order *.symvers
  • 2、编译内核模块(以hello word为例)
#include <linux/init.h>
#include <linux/module.h>


static int hello_init(void)
{
    printk(KERN_WARNING"Hello world!\n");
    return 0;   
}

static void hello_exit(void)
{
    printk(KERN_WARNING"hello exit!\n");    
}

module_init(hello_init);    /*入口*/
module_exit(hello_exit);    /*出口*/
示例
  • 3、内核模块可选项
    • 模块声明
      1、MODULE_LICENSE( ” 遵守的协议 ” )
      申明该模块遵守的许可证协议,如: “GPL“ 、 ”GPL v2“ 等
      2、MODULE_AUTHOR(“ 作者 ”)
      申明模块的作者
      3、MODULE_DESCRIPTION(“ 模块的功能描述")
      申明模块的功能
      4、MODULE_VERSION("V1.0")
      申明模块的版本

    • 模块参数
      在应用程序中,int main(int argc, char** argv),argc表示命令行输入的参数个数,argv中保存输入的参数;
      1.那么内核模块中可以通过命令行输入参数么?
      2.参数怎么传入,传入后保存在哪里?
      通过宏module_param指定保存模块参数的变量。模块参数用于在加载模块时传递参数给模块。

module_param(name,type,perm)
* name:变量的名称
* type:变量类型 bool:布尔型 int:整型 charp:字符串型
* perm是访问权限。 S_IRUGO:读权限 S_IWUSR:写权限
例:
int a = 3;
char *st;
module_param(a,int, S_IRUGO);
module_param(st,charp, S_IRUGO);
  • 符号输出
    先来思考下两个问题:
    什么是内核符号?
    为什么要导出模块中的内核符号?
    通俗的说,就是为了内核模块A的属性或方法能够让内核模块B使用。
    内核符号的导出使用宏:
    <1> EXPORT_SYMBOL(符号名)
    <2> EXPORT_SYMBOL_GPL(符号名)
    说明:
    其中EXPORT_SYMBOL_GPL只能用于包含GPL许可证的模块。
#include <linux/init.h>
#include <linux/module.h>

static int val=0;
char *p;

/*使用模块参数*/
module_param(val,int,S_IRUGO|S_IWUSR);
module_param(p,charp,S_IRUGO|S_IWUSR);

int func(void)
{
  return 0;
}

static int hello_init(void)
{
    printk(KERN_WARNING"Hello world!\n");
    printk("val=%d \n",val);
    printk("p is = %s \n",p);
    
    return 0;   
}
[图片上传中...(image.png-adb027-1547298574867-0)]

static void hello_exit(void)
{
    printk(KERN_WARNING"hello exit!\n");    
}

/*导出内核符号*/
EXPORT_SYMBOL(func);
MODULE_LICENSE("GPL");
module_init(hello_init);
module_exit(hello_exit);
运行示例
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 1 Linux 内核模块简介 Linux 内核是一个十分庞大的系统,如何能够为其瘦身,订制适合自己应用场景的 li...
    守拙圆阅读 1,106评论 0 2
  • 1 总体设计思路 Linux内核是单体式结构,相对于微内核结构而言,其运行效率高,但是系统的可维护性和可扩展性较差...
    ggdd5151阅读 405评论 0 0
  • 1. /proc/kallsyms列出了linux内核导出的所有符号及对应的地址。 基本格式是: 逻辑地址 标识 ...
    WebSSO阅读 428评论 0 0
  • 妈妈 对你的一切 我习以为常 以为你就是 平凡的模样 动画片里的女王 挥动她的魔法杖 闪耀星星的光芒 把你带回 让...
    花花的樱花阅读 295评论 0 4
  • 一滴滴雨滴落下时,我会特别的想去海边,看看浪花,汹涌磅礴。特别想要呐喊!可是又碍于面子。我就是一个如此矛...
    玲珑寒雪阅读 193评论 0 0