编译生成新内核
1、实践原理
Linux模块是一些能够做为独立程序来编译的函数和数据类型的集合。之因此提供模块机制,是由于Linux自己是一个单内核。单内核因为全部内容都集成在一块儿,效率很高,但可扩展性和可维护性相对较差,模块机制可弥补这一缺陷。linux
Linux模块能够经过静态或动态的方法加载到内核空间,静态加载是指在内核启动过程当中加载;动态加载是指在内核运行的过程当中随时加载。shell
一个模块被加载到内核中时,就成为内核代码的一部分。模块加载入系统时,系统修改内核中的符号表,将新加载的模块提供的资源和符号添加到内核符号表中,以便模块间的通讯。函数
2、实践过程
(一)简单模块——姓名post
1.编写模块代码测试
2.编译模块this
接下来写Makefile。spa
三、加载测试卸载模块指针
sudo insmod lsp.kocode
dmesgblog
(二)进程
1.编写模块代码
模块构造函数:执行insmod或modprobe指令加载内核模块时会调用的初始化函数。函数原型必须是module_init(),括号内是函数指针
模块析构函数:执行rmmod指令卸载模块时调用的函数。函数原型是module_exit()
模块许可声明:函数原型是MODULE_LICENSE(),告诉内核该程序使用的许可证,否则在加载时它会提示该模块污染内核。通常会写GPL。
头文件module.h,必须包含此文件;
头文件kernel.h,包含经常使用的内核函数;
头文件init.h包含宏_init和_exit,容许释放内核占用的内存。
写一个简单的代码,用来向内核输出进程信息。
#include<linux/kernel.h> #include<linux/module.h> #include<linux/init.h> #include<linux/sched.h> static struct task_struct *pcurrent; static int __init print_init(void) { printk(KERN_INFO "print current task info\n"); printk("pid\ttgid\tprio\tstate\n"); for_each_process(pcurrent){ printk("%d\t",pcurrent->pid); printk("%d\t",pcurrent->tgid); printk("%d\t",pcurrent->prio); printk("%ld\n",pcurrent->state); } return 0; } static void __exit print_exit(void) { printk(KERN_INFO "Finished\n"); } module_init(print_init); module_exit(print_exit);
2.编译模块
接下来写Makefile。
(其中,all到make的过程当中要使用“回车+Tab”键)
obj-m:=proclist.o CURRENT_PATH:=$(shell pwd) LINUX_KERNEL_PATH:= /usr/src/linux-headers-3.13.0-32-generic all: make -C $(LINUX_KERNEL_PATH) M=$(CURRENT_PATH) modules clean: make -C $(LINUX_KERNEL_PATH) M=$(CURRENT_PATH) clean
第一行:本身写的.c的文件名+”.o”。
第三行的LINUX_KERNEL_PATH后面要写你本身的内核版本对应的内核源码包地址.
解释一下make命令:
make -C $(LINUX_KERNEL_PATH) 指明跳转到内核源码目录下读取那里的Makefile
M=$(CURRENT_PATH) 代表返回到当前目录继续执行当前的Makefile。
make以后的执行时这样的:
生成了好多文件:
三、加载模块
sudo insmod proclist.ko
输入密码后便可。此时已经加载了模块。
四、测试模块
Dmesg:看内核信息
3、遇到的问题
1. linux的内核版本
uname –r
位数
2.内核位置