操作系统-实验二
1、模块一:show_all_kernel_thread
实验要求:设计一个不带参数的模块,要求列出系统中所有内核线程的程序名、PID、进程状态、进程优先级、父进程的PID。
模块源码
#include <linux/init.h>
#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/sched/signal.h>
MODULE_LICENSE("GPL");
/*模块的许可证声明, "GPL" 是指明了 这是GNU General Public License的任意版本;
从2.4.10版本内核开始,模块必须通过MODULE_LICENSE宏声明此模块的许可证,
否则在加载此模块时,会收到内核被污染 “kernel tainted” 的警告。*/
// 初始化函数,模块在加载的时候会运行init函数
static int __init show_all_kernel_thread_init(void)
{
// 格式化输出头
struct task_struct *p;
printk("%-20s%-6s%-6s%-6s%-6s", "Name", "PID", "State", "Prio", "PPID");
printk("--------------------------------------------");
// for_each_process(p)的作用是从0开始遍历进程链表中的所有进程
for_each_process(p)
// p最开始指向进程链表中第一个进程,随着循环不断进行p也不断后移直至链表尾,详见下文
{
if (p->mm == NULL)
//判断是否是内核线程,mm成员为空则是内核线程。
//因为内核线程没有独立的地址空间,所以内核线程地址空间的mm成员为空NULL。
{
// 打印进程p的相关信息
printk("%-20s%-6d%-6d%-6d%-6d", p->comm, p->pid, p->state, p->prio,
p->parent->pid);
}
}
return 0;
}
// 卸载函数,模块在加载的时候会运行exit函数
static void __exit show_all_kernel_thread_exit(void)
{
printk("[ShowAllKernelThread] Module Uninstalled.");
}
module_init(show_all_kernel_thread_init);
module_exit(show_all_kernel_thread_exit);
头文件
linux/init.h
包含了模块的初始化的宏定义,以及一些其他函数的初始化函数。
内核模块的初始化和注销函数就在这个文件中。
linux/module.h
包含了许多与加载模块有关的符号与函数的定义。
写内核驱动的时候 必须加载这个头文件,作用是动态的将模块加载到内核中去。
常用的宏定义如 MODULE_LICESENCE(),MODULE_AUTHOR(),等在此文件中。
linux/kernel.h
包含了内核打印函数等
linux/sched/signal.h
信号头文件。定义信号符号常量,信号结构以及信号操作函数原型。