Ubuntu16.04下的modules模块编译加载

一、首先编写对应的驱动程序的相关内容:(最简单的hello.c程序)

 1 #include<linux/init.h>
 2 #include<linux/module.h>
 3 MODULE_LICENSE("Dual BSD/GPL");
 4 MODULE_AUTHOR("MDAXIA");
 5 
 6 static int __init hello_init(void)
 7 {
 8         printk(KERN_ALERT "Hello world!
");
 9         return 0;
10 }
11 static void __exit hello_exit(void)
12 {
13         printk(KERN_ALERT "Goodbye,cruel world!");
14 }
15 module_init(hello_init);
16 module_exit(hello_exit);

二、编写对应Makefile文件:(注意事项Makefile,首字母大写M)

 1 ifeq ($(KERNELRELEASE),)
 2 KDIR:=/lib/modules/$(shell uname -r)/build
 3 PWD:=$(shell pwd)
 4 modules:
 5         $(MAKE) -C $(KDIR) M=$(PWD) modules
 6 modules_install:
 7         $(MAKE) -C $(KDIR) M=$(PWD) modules_install
 8 clean:
 9         rm -rf *.o *.ko .depend *.mod.o *.mod.c Module.* modules.* 
10 .PHONY:modules modules_install clean
11 else
12 obj-m :=hello.o
13 endif

三、使用make指令对程序进行编译生成目标文件hello.ko

sudo make

使用的是sudo make的指令来保证运行和文件的执行权限等等,编译的结果如下所示:

这里成功生成了我们需要的.ko文件

使用sudo make clean命令来清除相关的中间文件以及目标文件:

sudo make clean

这样就清除了所有的文件了~

四、安装加载模块,需要的是root权限:

sudo insmod ./hello.ko

这里的路径变了一下,是因为我的Ubuntu16.04的实体主机加载模块的时候,需要数字签名,但是数字签名之后还是不能正确的加载,之后就在我的虚拟机Ubuntu16.04上实验了一下,这样居然成功了,因此路径有所改变,但是驱动成功加载了。

驱动加载成功的验证方法:

cat /var/log/syslog | grep Hello

这样就显示驱动成功加载了

也可以使用lsmod来查看模块的加载:

lsmod | grep hello

使用rmmod指令来卸载驱动模块:

sudo rmmod hello

这里顺便推荐一个linux相关指令的中文查询网站:http://man.linuxde.net/

相关错误集合如下所示(确实不知道怎么解决,最后歪打正着就解决了,不过回过头来想,这些错误提示的分析也的确有很大的帮助吧):

Question1:这里重点关注一下第三行,Error的修改往往是从上到下的修改方式,所以这里提示的是-fstack-protector-strong not supported by compiler,意思很明确了,我们当前的gcc编译器版本不支持-fstack-protector-strong这个参数,这个参数的调用时在linux-headers-4.4.0-96-generic相关文件中的,既然不支持,那我们就找个支持的gcc版本来编译运行就可以了,通过查阅相关资料得到:‘-fstack-protector-strong’ 选项是gcc4.9以后的版本才加入的,也就是说需要安装gcc4.9以后的版本才可以编译通过,所以直接安装gcc5.4就行了,关于安装多个版本的gcc的内容请参考我的这篇文章:Ubuntu16.04多个版本GCC编译器的安装和切换完成安装之后直接make编译就不会出现上述问题了。

 1 make -C /lib/modules/4.4.0-96-generic/build M=/home/ubuntu-mm/ARMTools/Code_arm/modulesprogram modules
 2 make[1]: Entering directory '/usr/src/linux-headers-4.4.0-96-generic'
 3 Makefile:702: Cannot use CONFIG_CC_STACKPROTECTOR_STRONG: -fstack-protector-strong not supported by compiler
 4 scripts/Makefile.build:44: /home/ubuntu-mm/ARMTools/Code_arm/modulesprogram/Makefile: No such file or directory
 5 make[2]: *** No rule to make target '/home/ubuntu-mm/ARMTools/Code_arm/modulesprogram/Makefile'.  Stop.
 6 Makefile:1423: recipe for target '_module_/home/ubuntu-mm/ARMTools/Code_arm/modulesprogram' failed
 7 make[1]: *** [_module_/home/ubuntu-mm/ARMTools/Code_arm/modulesprogram] Error 2
 8 make[1]: Leaving directory '/usr/src/linux-headers-4.4.0-96-generic'
 9 makefile:7: recipe for target 'modules' failed
10 make: *** [modules] Error 2

Question2:这里重点关注了第四行的描述,也就是第一个make[1]的错误提示:/home/ubuntu-mm/ARMTools/Code_arm/modulesprogram/Makefile: No such file or directory,仔细一看,发现这里的文件名字是Makefile,回想起我的文件名是makefile,可能是这个问题,于是就rename一下:mv makefile Makefile 结果成功编译了~Cheers!所以说以后最好还是用Makefile来写吧~

 1 ubuntu-mm@ubuntu-mm:~/ARMTools/Code_arm/modulesprogram$ sudo make
 2 make -C /lib/modules/4.4.0-96-generic/build M=/home/ubuntu-mm/ARMTools/Code_arm/modulesprogram modules
 3 make[1]: Entering directory '/usr/src/linux-headers-4.4.0-96-generic'
 4 scripts/Makefile.build:44: /home/ubuntu-mm/ARMTools/Code_arm/modulesprogram/Makefile: No such file or directory
 5 make[2]: *** No rule to make target '/home/ubuntu-mm/ARMTools/Code_arm/modulesprogram/Makefile'. Stop.
 6 Makefile:1423: recipe for target '_module_/home/ubuntu-mm/ARMTools/Code_arm/modulesprogram' failed
 7 make[1]: *** [_module_/home/ubuntu-mm/ARMTools/Code_arm/modulesprogram] Error 2
 8 make[1]: Leaving directory '/usr/src/linux-headers-4.4.0-96-generic'
 9 makefile:7: recipe for target 'modules' failed
10 make: *** [modules] Error 2

Question3:insmod在插入模块的过程中出现下面的问题,查阅相关博客是模块数字签名的相关问题,详细参见:http://blog.csdn.net/hui872370036/article/details/69950869

insmod: ERROR: could not insert module ./hello.ko: Required key not available

内核从3.7后开始支持模块签名,这个功能使能以后,内核只允许安装特定key签名的模块,内核配置项如下所示:

 CONFIG_MODULE_SIG=y 表示开启了签名机制,但是这时候模块签名或不签名都可以使用。

 CONFIG_MODULE_SIG_FORCE=y 如果上述配置项使能,则模块必须有正确的签名才能正常使用。

 CONFIG_MODULE_SIG_ALL=y 内核在编译的时候,并不会主动去给模块签名,除非你把上述配置项打开。 

根据下面的英文教程的相关提示,按照下面的步骤进行操作:

/usr/src/linux-headers-4.4.0-96-generic/scripts/sign-file sha512 /usr/src/linux-headers-4.4.0-96-generic/certs/signing_key.pem /usr/src/linux-headers-4.4.0-96-generic/certs/signing_key.x509 hello.ko

这里显示数字签名注册成功~注册的方式是使用了对应ubuntu16.04的linux内核版本的sign-file文件,注册使用的方式是 哈希表sha512格式  注册的账户保存在signing_key.pem文件当中,对应的密钥保存在signing_key.x509文件当中。

如果你在对应的certs目录当中没有找到对应的两个文件,请用vim指令创建两个文件,并在.pem文件中输入如下的内容(这是pem文件的基本格式,所以要添上,否则会出现无法识别pem文件问题):

-----BEGIN CERTIFICATE-----
-----END CERTIFICATE-----

这样就完成了.pem文件的创建。

接下来就是加载对应的.ko模块的文件:

结果还是如此有待解决~

目前在虚拟机上直接插入和安装是没有问题的(虚拟机内核版本:4.4.0-83-generic,实体机内核版本:4.4.0-96-generic)

关于签名证书的问题,现在也还没有很好的解决,后面再尝试找到对应的解决办法吧~

详细的module数字签名的英文文档参照:https://wiki.gentoo.org/wiki/Signed_kernel_module_support#Enabling_module_signature_verification

完~

原文地址:https://www.cnblogs.com/uestc-mm/p/7644966.html