#include <linux/kernel.h> #include <linux/sched.h> #include <linux/module.h> #include <linux/init.h> #include <linux/moduleparam.h> static pid_t pid=1; module_param(pid, int, 0644); static int module1_init(void) { struct task_struct *p; struct list_head *pp; struct task_struct *psibling; //当前进程的PID p = pid_task(find_vpid(pid),PIDTYPE_PID); printk("me: %d %ld %s \n",p->pid,p->state,p->comm); //父进程 if(p->parent == NULL){ printk("No Parent\n"); } else{ printk("Parent:%d %ld %s \n",p->parent->pid,p->parent->state,p->parent->comm); } //兄弟进程 list_for_each(pp,&p->parent->children) { psibling = list_entry(pp,struct task_struct,sibling); printk("Brother %d %ld %s \n",psibling->pid,psibling->state,psibling->comm); } //子进程 list_for_each(pp,&p->children) { psibling = list_entry(pp,struct task_struct,sibling); printk("Children %d %ld %s \n",psibling->pid,psibling->state,psibling->comm); } return 0; } static void module1_exit(void) { printk(KERN_ALERT"goodbye!\n"); } module_init(module1_init); module_exit(module1_exit); MODULE_LICENSE("GPL");
时间: 2023-06-18 17:08:38 浏览: 316
这是一个 Linux 内核模块,可以获取指定进程的父进程、兄弟进程、子进程的信息,并打印输出。具体实现是通过调用 Linux 内核中的进程任务结构体 `task_struct` 中的成员变量来获取这些信息的。
其中,`module_param` 宏用于接收命令行参数,`module_init` 和 `module_exit` 宏分别用于指定模块的初始化函数和退出函数。`pid_task` 函数用于获取指定 PID 对应的进程任务结构体指针,而 `list_for_each` 宏则用于遍历双向链表。
需要注意的是,该模块需要在 Linux 内核源码目录下进行编译并加载,而且只能在特定的内核版本中运行。同时,也需要在加载时指定要查询的进程的 PID,例如:
```
insmod module1.ko pid=1234
```
这里的 `1234` 就是要查询的进程的 PID。
相关问题
#ifndef __KERNEL__ #define __KERNEL__ #endif #ifndef MODULE #define MODULE #endif #include <linuxersion.h> #include <linux/module.h> #include <linux/kernel.h> #include <linux.h> #include <linuxpes.h> #include <linux/errno.h> #include <linux/fs.h> #include <linux/sched.h> #include <linux/ioport.h> #include <linuxab.h> #include <linuxring.h> #include <asm/io.h> #include <asmgment.h> #include <asm/uaccess.h> #include "scull.h"
这段代码中包含了多个头文件,其中可能有一个错误的头文件名字。具体来说,`<asmgment.h>` 是一个非标准的头文件,不属于 Linux 内核或者标准的 C/C++ 库。如果你的代码中包含了这个头文件,编译器就会找不到它并报错。
你需要检查代码中的头文件名字是否正确,如果有误,需要将其更正为正确的头文件名字。如果你不确定应该包含哪个头文件,可以查看相关函数或变量的文档或者示例代码,确定正确的头文件名字。
为下面这段代码写一个makefile文件,要求把all/install/uninstall 写入 Makefile 文件中。#include <linux/init.h> #include <linux/module.h> #include <linux/workqueue.h> #include <linux/sched.h> #include <linux/delay.h> #include <linux/timekeeping.h> MODULE_LICENSE("GPL"); static struct workqueue_struct* workqueue; static struct work_struct work1; static struct delayed_work work2; static int times; module_param(times, int, 5); int get_time(char* buff, int len) { struct timespec64 ts; struct tm tm_res; char time_string[40]; ktime_get_real_ts64(&ts); time64_to_tm(ts.tv_sec, 0, &tm_res); buff=tm_res.tm_year + 1900, tm_res.tm_mon + 1, tm_res.tm_mday,tm_res.tm_hour, tm_res.tm_min, tm_res.tm_sec; return 0; } static void real_time_work(struct work_struct* work) { char buff[40]; get_time(buff, sizeof(buff)); printk(KERN_INFO "%s\n",buff); } static void delayed_work(struct work_struct* work) { char buff[40]; get_time(buff, sizeof(buff)); printk(KERN_INFO "this is a delay work :%s\n", buff); } static int __init module_init(void) { printk(KERN_INFO "Initializing workqueue module.\n"); workqueue = create_workqueue("workqueue"); INIT_WORK(&work1, real_time_work); INIT_DELAYED_WORK(&work2, delayed_work); queue_work(workqueue, &work1); queue_delayed_work(workqueue, &work2, msecs_to_jiffies(10000)); return 0; } static void __exit module_exit(void) { printk(KERN_INFO "Exiting workqueue module.\n"); flush_workqueue(workqueue); destroy_workqueue(workqueue); } module_init(module_init); module_exit(module_exit);
以下是makefile文件的内容:
```
obj-m := workqueue.o
all:
make -C /lib/modules/$(shell uname -r)/build M=$(PWD) modules
install:
make -C /lib/modules/$(shell uname -r)/build M=$(PWD) modules_install
depmod -a
uninstall:
rm /lib/modules/$(shell uname -r)/kernel/drivers/misc/workqueue.ko
depmod -a
```
其中,obj-m定义了需要编译的模块文件,即workqueue.o。all指令使用make命令编译模块文件,install指令安装模块文件并更新依赖,uninstall指令删除已安装的模块文件并更新依赖。
阅读全文