linux内核结构以及内核模块编程

1、linux内核结构

1.1、单内核与微内核结构

1.1.1、什么是单内核结构和微内核结构

  • linux操作系统是一个单内核的结构,它的各个子系统之间可以直接调用
    比如说文件系统、内存管理、进程管理以及网络系统和进程间通信它们互相之间可以直接调用
  • 只有一些核心的代码它在微内核里面
    比如说核心的进程间通信、中断以及调度程序放在微内核,而其他的像文件系统、内存管理可能以服务器的方式放在外部

1.1.2、单内核与微内核的区别

  • 单内核因为它各个子系统之间可以直接调用,所以性能比较高,但是它的可维护性比较差
  • 微内核与服务器之间是一种通信的关系,比如说当要发出一个请求的时候文件系统可能要与微内核进行通信,而这个通信是有代价的,所以微内核的效率比较低,但是因为这种架构各个服务器之间相对独立,所以它的维护性比较好

1.1.3、linux内核源码目录结构

其中各个目录的主要代码类别如下:

  • arch:架构相关
  • block:块设备管理相关
  • drivers:驱动相关
  • fs:文件系统相关
  • include:头文件相关
  • init:linux内核初始化相关
  • ipc:进程间通信
  • kernel:linux内核相关
  • lib:库函数相关
  • mm:内存管理相关
  • net:网络协议相关
    … …

linux-5.4:
在这里插入图片描述

2、linux内核模块

从上面linux源码目录也可以发现,linux内核是单内核结构,因此其维护性较差,因此为了能够让我们的驱动代码不需要每次都重新编译一次linux内核,linux提供了一种模块——可加装的linux内核模块(LKM)。因为linux使用C语言写的,由于C语言是模块化编程的,添加模块很麻烦,所以为了给Linux添加模块,于是诞生了LKM,就是可以使使用户随时添加内核模块到内核,也可以随时卸载,不和内核建立捆绑。
为了更好的理解该模块,下面写一个简单的linux内核驱动模块:

2.1、驱动模块的编译

  • hello.c文件
#include <linux/kernel.h>
#include <linux/init.h>
#include <linux/module.h>
//module.h包含了对模块的版本控制
//kernel.h包含了常用的内核函数
//init.h包含了宏__init和__exit
/*
 * 模块的初始化函数 hello_init()
 * __init是用于初始化的修饰符
 * insmod模块的时候执行
 */
static int __init hello_init(void)
{
  //<1>是输出的级别,表示立即在终端输出
  printk("<1>hello,Kernal!...\n!");
  return 0;
}
/*
 * 模块的退出和清理函数 hello_exit()
 * rmmod 模块的时候执行
 */
static void __exit hello_exit(void)
{
  printk("<1>Goodbye,world!...\n");
}

module_init(hello_init);
module_exit(hello_exit);
/*
 * 模块的许可证声明GPL
 */
MODULE_LICENSE("GPL");

ps:
printk相比printf来说还多了个:日志级别的设置,用来控制printk打印的这条信息是否在终端上显示的,当日志级别的数值小于控制台级别时,printk要打印的信息才会在控制台打印出来,否则不会显示在控制台!


#define    KERN_EMERG    "<0>"    /* system is unusable            */
#define    KERN_ALERT    "<1>"    /* action must be taken immediately    */
#define    KERN_CRIT    "<2>"    /* critical conditions            */
#define    KERN_ERR    "<3>"    /* error conditions            */
#define    KERN_WARNING    "<4>"    /* warning conditions            */
#define    KERN_NOTICE    "<5>"    /* normal but significant condition    */
#define    KERN_INFO    "<6>"    /* informational            */
#define    KERN_DEBUG    "<7>"    /* debug-level messages            */
  • Makefile文件
    内核模块不是独立的可执行文件,但在运行时其目标文件被链接到内核中,并且只有root用户才能insmod和rmmod模块
obj-m :=hello.o                    #产生hello模块的目标文件,这里的hello必须和c语言的文件名对应,
CURRENT_PATH := $(shell pwd)       #模块当前所在的当前路径
LINUX_KERNEL := $(shell uname -r)  #linux内核源代码的当前版本,uame -r 查看内核的linux命令
LINUX_KERNEL_PATH := /usr/src/linux-headers-$(LINUX_KERNEL)
                                   #linux内核源代码的绝对路径
all:
        make -C $(LINUX_KERNEL_PATH) M=$(CURRENT_PATH) modules #编译模块,Makefile文件中若一行是命令,必须以tab键开头
clean:
        make -C $(LINUX_KERNEL_PATH) M=$(CURRENT_PATH) clean   #清理模块

**

**ps:
obj-m := .o
obj-y := .o
上面两者的区别在于,前者才会生成ko文件,后者只是代码编译进内核,并不生成ko文件。

  • 如何安装和卸载驱动模块
    insmod hello.ko:安装hello.ko模块
    rmmod hello.ko:卸载hello.ko模块
    dmesg:查看hello.ko的模块信息
    lsmod:查看hello.ko模块,可搭配grep使用,例如,lsmod | grep hello

3、linux内核模块与C应用的对比

上面已经简单实现了一个linux内核模块,但是它与C应用编程有哪些区别呢,如下:
在这里插入图片描述

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值