C - Linux - 用于迭代进程的自定义内核模块'子程序炸毁内核日志和计算机

问题描述 投票:1回答:1

我是linux内核模块的新手,我正在尝试在处理复杂的概念之前实现一些基本概念。我编写了一个代码,它接受一个模块参数(一个int)并检查是否有一个带有该pid的进程。如果存在,则将其子项作为列表并在打印子项的ID和描述时对其进行迭代。这是代码:

#include <linux/init.h>
#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/list.h>
#include <linux/types.h>
#include <linux/slab.h>
#include <linux/moduleparam.h>
#include <linux/sched/signal.h>

MODULE_LICENSE("Dual BSD/GPL");
MODULE_AUTHOR("Some guy");

int mypid = 0;

static int simple_init(void)
{

    struct task_struct *task;
    struct list_head *list;

    printk(KERN_ALERT "Loading Module\nThe process id: %d\n", mypid);


    for_each_process(task){
        printk(KERN_ALERT "PID/NAME: %d/%s\n", task->pid, task->comm);

        if(task->pid == mypid){

            printk(KERN_ALERT "The common pid found: %d/%s\n", task->pid, task->comm);


            list_for_each(list, &task->children){

            task = list_entry(list, struct task_struct, sibling);               
                //printk(KERN_INFO "Parent ID/NAME: %d/%s\n", task->parent->pid, task->parent->comm);               
                printk(KERN_ALERT "Child PID/NAME: %d/%s\n", task->pid, task->comm);
            }

    } 


    return 0;

}


static void simple_exit(void){

    printk(KERN_WARNING "Removing Module\n");

}

module_init(simple_init);
module_exit(simple_exit);
module_param(mypid, int, 0);

但是,当我运行此代码时

sudo insmod listtasks.ko mypid=1800(or a random pid)

它不会停止执行并占用所有内核日志内存,最终冻结计算机。我习惯在恢复模式下重新启动它并删除爆炸的日志文件,但我看不出如何解决问题。非常感谢所有的帮助。

亲切的问候,

c linux ubuntu linux-kernel kernel-module
1个回答
2
投票

我通过初始化一个名为childtask的新task_struct解决了这个问题:

struct task_struct *childtask;

然后将其分配给list_for_each循环中的list_entry:

childtask = list_entry(list, struct task_struct, sibling);  

所以task和childtask是不同的指针。

© www.soinside.com 2019 - 2024. All rights reserved.