Linux:如何突破内核模块验证的限制?

1、要解决的问题

Linux的内核模块(即ko文件)在编译时指定了可以运行在哪个内核版本。
例如,使用Ubuntu 16.04内核4.8.0-42-generic头文件编译的ko文件,只能运行这个内核版本中。

这就有个麻烦,Ubuntu内核版本更新速度非常快,如何内核更新为4.8.0-58-generic,那么这个ko文件就不能运行了,需要重新编译ko文件。

能否有简单的方法,让已有的ko文件运行不同的内核版本中呢?
有,本文介绍一种方法。

2、Linux内核是如何限制ko文件的执行?

ko文件运行在Linux内核中,运行时会调用Linux内核提供的接口。
由于Linux内核版本演进非常快,内核提供的接口可能经常发生变动。

为了保持Linux内核运行稳定,Linux采用的策略是:

  • 编译ko文件时,确定了它调用的所有Linux内核接口的版本。
  • 装载ko文件时,检查它调用的Linux内核接口版本,如果与当前Linux内核提供的接口版本完全致,则可以装载,否则拒绝装载。

2、内核接口版本是什么?

Linux内核为每个接口计算一个CRC整数值
例如:vfs_create 0xba70d5ae

编译ko时,针对是某个内核版本,把每个接口的CRC值都记录在ko文件中,即接口的版本信息。
装载ko时,与当前内核的接口CRC值进行比较,即接口版本比较。

参考:《Linux内核模块符号CRC检查机制》

3、修改ko文件中的CRC值能突破运行限制吗?

可以。

我在Ubuntu和CentOS上实验,通过修改ko文件中的接口CRC值,能够使之运行在不同的内核版本中。

4、如何实现呢?

Ubuntu和CentOS中有有相应的文本文件记录了指定版本内核的所有接口CRC值。

  • Ubuntu:/boot目录下有所有已安装版本的abi文件,例如abi-4.8.0-58-generic。
  • CentOS:/boot目录下前缀是symvers的文件。

只要把ko文件中所有使用的内核接口的CRC值改为与上述文件中的值,就能让ko文件运行了。

github上有一个开源程序能够修改ko文件的CRC值:
https://github.com/romeroperezabel/vermagic

我们可以修改这个程序,实现如下功能:

  • 列出ko文件调用的所有内核接口
  • 在记录CRC值的文件中,找到接口对应的CRC值
  • 修改ko文件中接口的CRC值

这样,ko文件就能运行在相应的内核版本中了。

5、注意

有可能修改成功,也有可能修改失败,例如新版本把某个接口取消了。
如果修改失败,就得重新编译ko文件了。

即使修改成功,也可能运行失败,严重的会导致系统无法启动,所以要特别注意,可以先在虚拟机上测试。
如果运行失败,也得重新编译ko文件了。

6、其它

最近遇到一个编译后ko文件无法运行的问题,解决方法如下。
repoline的问题:insmod一个简单ko,出现错误

  • 0
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 编写Linux内核模块需要掌握C语言编程和Linux内核的基本知识。以下是编写Linux内核模块的基本步骤: 1. 编写模块代码:使用C语言编写模块代码,包括模块初始化和清理函数。 2. 编写Makefile文件:Makefile文件用于编译和链接模块代码。 3. 编译模块:使用make命令编译模块代码,生成.ko文件。 4. 加载模块:使用insmod命令加载模块。 5. 卸载模块:使用rmmod命令卸载模块。 6. 查看模块信息:使用lsmod命令查看已加载的模块信息。 编写Linux内核模块需要注意以下几点: 1. 模块代码必须遵循Linux内核的编程规范。 2. 模块代码必须与当前内核版本兼容。 3. 模块代码必须具有正确的权限和安全性。 4. 模块代码必须经过充分的测试和验证。 总之,编写Linux内核模块需要认真学习和掌握相关知识,同时注重代码质量和安全性。 ### 回答2: Linux内核模块是向Linux内核添加新功能的一种方法。编写Linux内核模块需要一定的编程技能和一定的Linux操作系统知识。 首先,需要安装Linux操作系统、开发工具和编译器,如gcc和make。接着,需要了解Linux内核模块的编程API和编程语言,如C语言。可以通过阅读Linux内核源代码和其文档来了解这些方面的知识。 然后,可以开始编写内核模块代码。在编写Linux内核模块时,需要遵循特定的格式和样式。首先,需要包含Linux内核头文件,然后定义模块初始化函数和模块卸载函数。模块初始化函数将会在模块加载时启动,而模块卸载函数则会在模块被卸载时被调用。需要编写代码实现模块所需的功能。 在完成代码编写后,可以使用makefile文件来编译和构建模块。makefile定义编译规则和操作,如编译器选项和链接库。编译完成后,需要使用insmod命令将模块加载到内核中。可以使用lsmod命令检查模块是否加载成功。如果需要卸载模块,可以使用rmmod命令进行卸载。 总的来说,要编写Linux内核模块需要一定的技能和知识,但如果对Linux内核熟悉的话,编写内核模块并不是一件非常困难的事情。对于初学者来说,最好仔细阅读Linux内核源代码和文档,并花费时间去理解Linux的工作原理和内核API。 ### 回答3: 编写 Linux 内核模块可以实现 Linux 系统的定制化和扩展化,可以为系统添加新的功能或者修改系统现有的功能。创建内核模块是增加 Linux 内核灵活性的威力工具之一。 以下是编写 Linux 内核模块的步骤: 1. 安装编写内核模块所需的开发工具:通常情况下需要安装 Linux 内核头文件和编译工具包。 2. 创建一个新的内核模块源文件:编写一个包含必要模块初始化函数和清理函数的源文件。 3. 编写模块初始化函数: 初始化函数是一个从内核开始执行到模块加载时调用的函数,它在模块加载时初始化模块状态。 4. 实现模块功能: 在初始化函数中实现模块的功能,例如向系统添加新的设备驱动程序,注册新的文件系统或者添加扩展虚拟文件系统等。 5. 注册模块和退出: 在模块初始化函数中通过注册一些关键信息和环境变量来告诉内核如何正确处理模块,使用函数 module_init() 和 module_exit() 注册和退出模块。 6. 编译模块: 通过编译命令将源文件编译为动态的共享库,然后将目标文件移动到内核可加载模块目录中。 7. 加载内核模块: 使用 modprobe 命令加载编译好的内核模块,模块将会被加载到内核中并且驱动尝试去执行初始化函数。 8. 卸载内核模块: 通过 rmmod 命令卸载内核模块,该命令将会在模块退出函数中被执行,协助系统将资源交还给内核。 总而言之,编写 Linux 内核模块需要熟悉系统 API 接口和底层机制,才能设计和编写可靠高效的程序。编写好的内核模块可以定制化和扩展系统的功能,使该系统得以满足特定的应用需求。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值