操作系统实验二

这篇博客介绍了两个Linux内核模块的实现,第一个模块`show_all_kernel_thread`用于列出所有内核线程的详细信息,包括程序名、PID、状态和优先级。第二个模块`show_task_family`则展示指定PID进程的家族信息,包括父进程、兄弟进程和子进程。博客还探讨了内核线程与普通线程的区别,以及涉及的宏定义和`task_struct`结构。
摘要由CSDN通过智能技术生成

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
信号头文件。定义信号符号常量,信号结构以及信号操作函数原型。

宏定义 for_each_pro

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值