gpt4 book ai didi

c - `struct task_struct current` 的兄弟总是包含一个 `pid = 0` 的进程

转载 作者:行者123 更新时间:2023-12-04 01:03:59 24 4
gpt4 key购买 nike

我正在破解 linux 内核并与 struct task_struct current 的 sibling 和 child 一起玩。

当输出 sibling 的pid和命令名时,似乎有一个格式错误的进程,pid = 0,命令名是胡言乱语。

进程的父进程也会发生同样的事情。

为什么兄弟之间出现pid=0的进程?那个进程不是为 swapper 保留的吗?

代码

// Loop over process and parents using something like:

/*
printk("--syscall ## Begin process results ##");

printk("--syscall // View children //");
my_list_head = &(current->children);

printk("--syscall // View siblings //");
my_list_head = &(current->sibling)

printk("--syscall results: ...");
*/


if (my_list_head == NULL) {
return 0;
}

list_for_each(tempNode, my_list_head) {
  tempTask = list_entry(tempNode, struct task_struct,
          sibling);
  printk("--syscall The %ld-th process's pid is %d and command %s",
         count, tempTask->pid, tempTask->comm);
 }

输出

带空格的格式

[ 2938.994084] --syscall ## Begin process results ##
[ 2938.994089] --syscall // View children //
[ 2938.994105] --syscall // View siblings //
[ 2938.994116] --syscall The 1-th process's pid is 0 and command \x80ݶE\x96\xff\xff
[ 2938.994133] --syscall results: pid=1400 name=process_ancesto state=0 uid=1000 nvcsw=1 nivcsw=0 num_children=0 num_siblings=1

[ 2938.994139] --syscall ## Begin process results ##
[ 2938.994144] --syscall // View children //
[ 2938.994149] --syscall The 1-th process's pid is 1400 and command process_ancesto
[ 2938.994158] --syscall // View siblings //
[ 2938.994163] --syscall The 1-th process's pid is 0 and command
[ 2938.994176] --syscall results: pid=1282 name=bash state=1 uid=1000 nvcsw=88 nivcsw=18 num_children=1 num_siblings=1

[ 2938.994180] --syscall ## Begin process results ##
[ 2938.994185] --syscall // View children //
[ 2938.994190] --syscall The 1-th process's pid is 1282 and command bash
[ 2938.994198] --syscall // View siblings //
[ 2938.994203] --syscall The 1-th process's pid is 1275 and command systemd
[ 2938.994210] --syscall The 2-th process's pid is 0 and command
[ 2938.994216] --syscall The 3-th process's pid is 117 and command systemd-journal
[ 2938.994222] --syscall The 4-th process's pid is 145 and command systemd-udevd
[ 2938.994227] --syscall The 5-th process's pid is 148 and command systemd-network
[ 2938.994233] --syscall The 6-th process's pid is 369 and command systemd-resolve
[ 2938.994239] --syscall The 7-th process's pid is 370 and command systemd-timesyn
[ 2938.994245] --syscall The 8-th process's pid is 412 and command accounts-daemon
[ 2938.994321] --syscall The 9-th process's pid is 413 and command dbus-daemon
[ 2938.994336] --syscall The 10-th process's pid is 417 and command irqbalance
[ 2938.994346] --syscall The 11-th process's pid is 418 and command rsyslogd
[ 2938.994352] --syscall The 12-th process's pid is 419 and command snapd
[ 2938.994359] --syscall The 13-th process's pid is 420 and command systemd-logind
[ 2938.994365] --syscall The 14-th process's pid is 439 and command cron
[ 2938.994372] --syscall The 15-th process's pid is 451 and command atd
[ 2938.994378] --syscall The 16-th process's pid is 456 and command agetty
[ 2938.994385] --syscall The 17-th process's pid is 461 and command sshd
[ 2938.994390] --syscall The 18-th process's pid is 491 and command unattended-upgr
[ 2938.994397] --syscall The 19-th process's pid is 501 and command polkitd
[ 2938.994413] --syscall results: pid=1200 name=login state=1 uid=0 nvcsw=31 nivcsw=33 num_children=1 num_siblings=19

最佳答案

下面是两个同胞子进程如何链接到其父进程的子进程列表的说明:

     PARENT              CHILD 1             CHILD 2
====== ======= =======

task_struct task_struct
+-------------+ +-------------+
| | | |
task_struct ~ ~ ~ ~
+-------------+ | | | |
| | |-------------| |-------------|
~ ~ | children | | children |
| | | | | |
. . |-------------| . . |-------------| . . |-------------| . .
| children | | sibling | | sibling |
X==>| prev | next |<===>| prev | next |<===>| prev | next |<==X
. . |-------------| . . |-------------| . . |-------------| . .
| sibling | | | | |
| | ~ ~ ~ ~
|-------------| | | | |
| | +-------------+ +-------------+
~ ~
| | 'X's are joined together, making
+-------------+ a doubly linked, circular list.

尽管 childrensibling 都是 struct list_head 类型,但是 children 被用作实际列表head(链接到它的子进程列表),而 sibling 被用作列表条目。

parent 的 children.next 链接指向 child 1 的 sibling 成员, child 1 的 sibling.next 链接指向 child 2 的 sibling 成员,而 child 2 的 sibling.next 链接指向父级的 children 成员(列表头)。类似地, parent 的 children.prev 链接指向 child 2 的 sibling 成员, child 2 的 sibling.prev 链接指向 child 1 的 兄弟 成员,子 1 的 sibling.prev 链接指向父级的 children 成员。

list_for_each(pos, head) 宏访问列表中的每个节点 pos,从 head->next 开始,而 pos != 头

通常情况下,list_for_each(pos, head)head参数应该是一个真正的链表头,但是宏无法区分链表头和链表入口。它们都是同一类型,所有节点都循环链接在一起。 (整个列表由一个列表头组成,其中零个或多个列表条目链接成一个圆圈。对于空列表,列表头仅链接回自身。)list_for_each 宏将围绕双向链表,直到它回到它开始的地方。

如果调用 list_for_each(pos, head)head 指向父级的 children 成员,则 pos 将在第一次迭代中指向 child 1 的 sibling 成员,并在第二次迭代中指向 child 2 的 sibling 成员,并使用 pos 终止循环 指向父级的 children 成员。在循环内,list_entry(pos, struct task_struct, sibling) 将正确指向子进程的 struct task_struct 的开头。

假设子进程 1 是当前进程。 OP 的代码使用 list_for_each(pos, head)head 指向 child 1 的 sibling 成员。因此,pos 将在第一次迭代中指向 child 2 的 sibling 成员,并在第二次迭代中指向父级的 children 成员,并且将终止循环,pos 指向子 1 的 sibling 成员。在循环内部,list_entry(pos, struct task_struct, sibling) 会在第一次迭代中正确指向 child 2 的 struct task_struct 的开头,但是 pos 将指向在第二次迭代中父级的 struct task_struct 开始之前的某个位置。这就是问题出在 OP 代码中的地方。

关于c - `struct task_struct current` 的兄弟总是包含一个 `pid = 0` 的进程,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/67119348/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com