LDD学习笔记 -- Linux内核模块

2024-01-04 02:44

本文主要是介绍LDD学习笔记 -- Linux内核模块,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

LDD学习笔记 -- 内核模块

    • 简介
    • LKM类型
    • Static Linux Kernel Module
    • Dynamic Linux Kernel Module
    • LKM编写语法 @syntax@
    • 详细描述
      • 内核头文件
      • 用户空间头文件
      • Module Initialization Function
      • Module Cleanup Function
      • Keyword & Tag
      • 宏 __init __exit
      • LKM入口注册
      • Module Metadate(Module Description)
    • 构建内核模块 Building
      • In tree module
      • Out of tree module
      • Kernel Build System
      • 在host系统上构建并安装内核模块
      • 为Target系统构建并安装内核模块

简介

在运行时添加 / 删除的代码

Linux支持在系统启动和运行时从内核中动态的插入或移除代码。在运行时添加/删除的代码叫做内核模块。

Linux内核模块通过向内核引入新的功能(安全、设备驱动、文件系统驱动、系统调用、其他)来动态的扩展内核的功能。模块化的方法,像正在运行的内核的插件。

嵌入式Linux系统可以由 最小的基本内核镜像 + 可选的设备驱动/其他功能通过模块插入按需提供

例如一个热拔插的USB设备,它的驱动程序(Linux内核模块),当其插入后会动态加载到内核中

LKM类型

  • Static Linux Kernel Module
  • Dynamic Linux Kernel Module

Static Linux Kernel Module

当构建Linux内核时,将模块静态的链接到内核镜像。模块成为最终内核镜像的一部分,内核镜像size变大。
不能卸载,运行中模块永久的占据内存。

Dynamic Linux Kernel Module

不会在编译期间嵌入内核镜像,单独编译和链接生成.ko文件
可以使用用户空间程序insmod, modprobe, rmmod从内核中加载/卸载,从而允许更加灵活的管理内核模块。

LKM编写语法 @syntax@

  • Header section
  • Code
  • Registration
  • Module description
#include <linux/module.h>/* This is module initialization entry point */
static int __init my_kernel_module_init(void)
{pr_info("hello kernel~\n");return 0;
}/* This is module clean-up entry point */
static void __exit my_kernel_module_exit(void)
{pr_info("my hello module exit\n");
}/* registration */
module_init(my_kernel_module_init);
module_exit(my_kernel_module_exit);/* This is description information about the module */
MODULE_LICENSE("GPL");
MODULE_AUTHOR("NAME");
MODULE_DESCRIPTION("A kernel module to print hello");

详细描述

运行在内核空间的。

内核头文件

内核源码树目录下 /include/linux 有所有的内核头文件,例如module.h
内核模块代码是运行在内核空间的,需要使用内核头文件,在内核构建过程中,没有用户空间库链接到
内核模块。不能将内核模块与C标准库链接。因此在写内核模块的时候不能使用任何用户空间的头文件。

用户空间头文件

例如C标准库里的stdio.h

Module Initialization Function

like main,做与初始化相关的事情
初始化模块,在模块插入 / 静态链接模块的系统引导期间调用。
返回 0 表示成功;返回 非0 表示失败从而阻止模块加载。
内容:设备的一些初始化,初始化设备私有数据结构,动态地为各种内核数据结构和服务请求一些内存,请求分配 major number, minor number, 创建设备文件, 在函数中执行各种操作。 (后面的字符设备驱动里)

Module Cleanup Function

复杂清理模块任务,在模块被移除时候调用。

Keyword & Tag

  • static就是常用的意思,修饰的函数只在这个文件中有效
  • __init 标记表示此代码是模块初始化特有的,模块初始化完成后可以从内存中丢弃,以节省内核内存。对于减少动态加载模块中的内核内存很有用。

宏 __init __exit

__init__exit是Linux kernel中使用的C语言宏。
定义在LINUX_SRC/include/linux/init.h
是编译器指令或属性

#define __init			__section(.init.text)
#define __initdata		__section(.init.data)
#define __initconst 	__section(.init.rodata)#define __exit			__section(.exit.text)

编译器指令,指示编译器将数据或代码保存在称为"init."的输出段中(在最终的内核印象中)。
只对静态模块有意义!

__init目的释放内核运行时内存。初始化函数执行后,在引导期间,内核将从内存中释放.init部分,只对静态模块有用,函数只会在boot引导的时候调用一次,无法卸载的内置驱动程序不需要再内存中保存其对init函数的引用。
因为有10万个内置模块~

通过使用__init技术,将函数推入init段,这是一个特殊的段,内核后面可以释放它

__exit目的:内建模块,不需要清理函数,当其与清理函数一起使用时,内核构建系统将在构建过程中排除这些函数,作为构建系统的标记,将清理函数从最终内核印象中排除。不编译了呗~,

LKM入口注册

module_init module_exir Macros,定义在linux/module.h
在内核编程中,需要向内核注册初始化和清理函数。
使用内核提供的宏 module_initmodule_exit来完成
分别将参数添加到内核init入口点数据库(the init entry point database of the kernel)和内核出口点数据库(the exit entry point database of the hernel)

Module Metadate(Module Description)

MODULE_LICENSE 该内核模块的开源许可类型
MODULE_AUTHOR
MODULE_DESCRIPTION

可以使用objdump modinfo 提取内核模块的元数据,即模块信息

构建内核模块 Building

  • 与内核镜像静态链接
  • 作为可动态加载的模块
    • In tree module
    • out of tree module

In tree module

使用make modules 命令去构建Linux内核所有的动态可加载的内核模块,都是In tree module,他们在Linux kernel tree内部。意味着得到内核开发人员和维护者的认可。

Out of tree module

加载它会污染内核,加载该模块到内核中时内核会发出一个警告,内核中也会设置一个taint flag,可忽略。指在内核源码树之外的。

Kernel Build System

调用"kbuild"构建内核,不用关心使用哪种编译器开关或参数。

在编译外部模块前,需要有预构建的完整内核源码,因为它包含配置项和构建时需要的头文件。注:这个linux kernel源码的版本必须和目标板运行的版本一样哦~

  • 下载完整的内核源码,并build它;

make -C <path to linux kernel tree> M=<path to your module> [target]
[Target] :modules modules_install clean help
C是为了使用linux kernel源码的顶层Makefile,指定kbuild的编译开关,依赖列表,版本符号
本地Makefile需要指定kbuild的变量: obj-<X>:=<module_name>.o
X = n ,不编译模块; =y,编译模块连接到kernel image; =m,编译动态内核模块

在host系统上构建并安装内核模块

uname -r 查看机器上的linux版本 5.4.0-150-generic 这里是预编译的内核源码和内核头文件路径/lib/modules/5.4.0-150-generic/
在这里插入图片描述
make -C /lib/modules/5.4.0-150-generic/build/ M=$PWD modules
在这里插入图片描述
sudo insmod hello.ko
sudo rmmod hello.ko
dmesg
在这里插入图片描述

为Target系统构建并安装内核模块

注意已经在~/.bashrc中添加了目标板的编译工具链路径,目标板也预编译了。
make ARCH=arm CROSS_COMPILE=arm-buildroot-linux-gnueabihf- -C ../../Linux-4.9.88/ M=$PWD modules
make ARCH=arm CROSS_COMPILE=arm-buildroot-linux-gnueabihf- -C ../../Linux-4.9.88/ M=$PWD clean

file hello.ko 查看模块的体系结构
modinfo hello.k o 查看模块的信息
在这里插入图片描述
arm-buildroot-linux-gnueabihf-objdump -h hello.ko 分析内核对象文件的各个部分
在这里插入图片描述

在目标板上安装模块
这里使用最方便的方法,使用WindTerm工具SSH方式登陆目标板,把hello.ko文件拖到板子里 。

sudo insmod hello.ko
sudo rmmod hello.ko
dmesg

SSH连接的dmesg
在这里插入图片描述
串口的输出
在这里插入图片描述
在目标系统板上测试内核模块,成功~

这篇关于LDD学习笔记 -- Linux内核模块的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/567922

相关文章

windows和Linux安装Jmeter与简单使用方式

《windows和Linux安装Jmeter与简单使用方式》:本文主要介绍windows和Linux安装Jmeter与简单使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录Windows和linux安装Jmeter与简单使用一、下载安装包二、JDK安装1.windows设

Kali Linux安装实现教程(亲测有效)

《KaliLinux安装实现教程(亲测有效)》:本文主要介绍KaliLinux安装实现教程(亲测有效),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、下载二、安装总结一、下载1、点http://www.chinasem.cn击链接 Get Kali | Kal

linux服务之NIS账户管理服务方式

《linux服务之NIS账户管理服务方式》:本文主要介绍linux服务之NIS账户管理服务方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、所需要的软件二、服务器配置1、安装 NIS 服务2、设定 NIS 的域名 (NIS domain name)3、修改主

Linux实现简易版Shell的代码详解

《Linux实现简易版Shell的代码详解》本篇文章,我们将一起踏上一段有趣的旅程,仿照CentOS–Bash的工作流程,实现一个功能虽然简单,但足以让你深刻理解Shell工作原理的迷你Sh... 目录一、程序流程分析二、代码实现1. 打印命令行提示符2. 获取用户输入的命令行3. 命令行解析4. 执行命令

重新对Java的类加载器的学习方式

《重新对Java的类加载器的学习方式》:本文主要介绍重新对Java的类加载器的学习方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1、介绍1.1、简介1.2、符号引用和直接引用1、符号引用2、直接引用3、符号转直接的过程2、加载流程3、类加载的分类3.1、显示

ubuntu16.04如何部署dify? 在Linux上安装部署Dify的技巧

《ubuntu16.04如何部署dify?在Linux上安装部署Dify的技巧》随着云计算和容器技术的快速发展,Docker已经成为现代软件开发和部署的重要工具之一,Dify作为一款优秀的云原生应用... Dify 是一个基于 docker 的工作流管理工具,旨在简化机器学习和数据科学领域的多步骤工作流。它

Linux高并发场景下的网络参数调优实战指南

《Linux高并发场景下的网络参数调优实战指南》在高并发网络服务场景中,Linux内核的默认网络参数往往无法满足需求,导致性能瓶颈、连接超时甚至服务崩溃,本文基于真实案例分析,从参数解读、问题诊断到优... 目录一、问题背景:当并发连接遇上性能瓶颈1.1 案例环境1.2 初始参数分析二、深度诊断:连接状态与

Linux系统调试之ltrace工具使用与调试过程

《Linux系统调试之ltrace工具使用与调试过程》:本文主要介绍Linux系统调试之ltrace工具使用与调试过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐... 目录一、ltrace 定义与作用二、ltrace 工作原理1. 劫持进程的 PLT/GOT 表2. 重定

Linux区分SSD和机械硬盘的方法总结

《Linux区分SSD和机械硬盘的方法总结》在Linux系统管理中,了解存储设备的类型和特性是至关重要的,不同的存储介质(如固态硬盘SSD和机械硬盘HDD)在性能、可靠性和适用场景上有着显著差异,本文... 目录一、lsblk 命令简介基本用法二、识别磁盘类型的关键参数:ROTA查询 ROTA 参数ROTA

嵌入式Linux之使用设备树驱动GPIO的实现方式

《嵌入式Linux之使用设备树驱动GPIO的实现方式》:本文主要介绍嵌入式Linux之使用设备树驱动GPIO的实现方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐... 目录一、设备树配置1.1 添加 pinctrl 节点1.2 添加 LED 设备节点二、编写驱动程序2.1