如何编写一个linux内核模块

概述

内核模块是运行在内核空间的,区别于应用程序运行在用户空间,大多数内核模块是完成对硬件设备或者总线的驱动,文件名后缀为.ko,本篇文章记录一下该如何编写一个简单的内核模块并让其能在Linux系统中运行。

环境

首先需要有一个可用的Linux系统,因为我们大部分的电脑都是装的windown系统,所以需要在电脑上装一个虚拟机,在安装linux系统例如Ubuntu或者Centos等成熟的Linux发行版,假如是win10系统,也可以使用系统内置的Ubuntu系统。

本文采用的环境是在虚拟机中安装了Ubuntu18.04发行版。

源码

#include <linux/init.h>
#include <linux/module.h>
MODULE_LICENSE(Dual BSD/GPL);

static int hello_init(void)
{
    printk("KERN_ALERT" "hello world\n");
    return 0;
}

static void hello_exit(void)
{
    printk("KERN_ALERT "goodbye world");
}

module_init(hello_init);
module_exit(hello_exit);

MODULE_LICENSE宏定义表示该源码是基于什么许可证下的

module_init();当模块插入系统时会执行

module_exit();当模块从系统中移除时候会执行

printk是类似于用户空间使用的printf打印函数,内核中编程不能使用C库函数,内核自己实现了一套类似于C库的函数

Makfile

 ifeq ($(KERNELRELEASE),)
 
     # Assume the source tree is where the running kernel was built
     # You should set KERNELDIR in the environment if it's elsewhere
     KERNELDIR ?= /lib/modules/$(shell uname -r)/build
     # The current directory is passed to sub-makes as argument
     PWD := $(shell pwd)
 
 modules:
         $(MAKE) -C $(KERNELDIR) M=$(PWD) modules
 
 modules_install:
         $(MAKE) -C $(KERNELDIR) M=$(PWD) modules_install
 clean:
         rm -rf *.o *~ core .depend .*.cmd *.ko *.mod.c .tmp_versions *.mod modules.order *.symvers

 .PHONY: modules modules_install clean
 
 else
     # called from kernel build system: just declare what our modules are
     obj-m := hello.o
endif

uname -r 命令得到当前系统的内核版本

obj-m 表示将生成ko文件,可插拔的内核模块

只要执行make命令,它将按照makefile的规则,将上面的源码编译成可执行的模块

运行结果

insmod hello.ko 加载模块

rmmod hello.ko 卸载模块

打开/var/log/syslog文件可以看到下面的日志记录,printk打印的信息

Jun 10 09:39:53 localhost kernel: [168826.316966] hello world
Jun 10 09:40:32 localhost kernel: [168865.183984] goodbye world

欢迎访问我的博客

©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

友情链接更多精彩内容