gpt4 book ai didi

c - Linux 克隆调用的最小堆栈大小?

转载 作者:太空狗 更新时间:2023-10-29 11:12:42 24 4
gpt4 key购买 nike

我一直在摆弄 clone 调用,我注意到不同的子线程堆栈分配有三种不同的结果。以下演示分配一个 n 字节大的堆栈,其中 n 作为参数传递,然后尝试克隆。

foo.c:

#define _GNU_SOURCE
#include <stdlib.h>
#include <unistd.h>
#include <sched.h>
#include <errno.h>

int child(void *arg)
{
(void)arg;
write(STDOUT_FILENO, "carpe momentum\n", 15);
return 0;
}

int main(int argc, char **argv)
{
long stacksize;
pid_t pid;
void *stack;

if (argc < 2)
return 1;

errno = 0;
stacksize = strtol(argv[1], NULL, 0);
if (errno != 0)
return 1;

stack = malloc(stacksize);
if (stack == NULL)
return 1;

pid = clone(child, stack + stacksize, 0, NULL);
if (pid == -1)
return 1;

write(STDOUT_FILENO, "success\n", 8);

return 0;
}

这是我的观察:

$ cc -o foo foo.c
$ ./foo 0
Segmentation fault
$ ./foo 23
Segmentation fault
$ ./foo 24
success
$ ./foo 583
success
$ ./foo 584
success
carpe momentum
$ ./foo 1048576 #1024 * 1024, amount suggested by man-page example
success
carpe momentum

所有 0 到 23 之间的零散样本都发生了段错误,对于 24 到 583 之间的所有样本,父级成功但子级保持沉默。高于 584 的任何合理值都会导致两者都成功。

反汇编表明 child 仅使用 16 个字节的堆栈空间,另外至少还有 16 个字节用于调用 write。但这已经超过了停止段错误所需​​的 24 个字节。

$ objdump -d foo
# ...
080484cb <child>:
80484cb: 55 push %ebp
80484cc: 89 e5 mov %esp,%ebp
80484ce: 83 ec 08 sub $0x8,%esp
80484d1: 83 ec 04 sub $0x4,%esp
80484d4: 6a 0f push $0xf
80484d6: 68 50 86 04 08 push $0x8048650
80484db: 6a 01 push $0x1
80484dd: e8 be fe ff ff call 80483a0 <write@plt>
80484e2: 83 c4 10 add $0x10,%esp
80484e5: b8 00 00 00 00 mov $0x0,%eax
80484ea: c9 leave
80484eb: c3 ret
# ...

这提示了几个重叠的问题。

  • 为什么 clone 不会出现 24 到 583 字节堆栈之间的段错误?
  • child 如何在堆栈太少的情况下悄无声息地失败?
  • 堆栈空间有什么用?
  • 24 和 584 字节的意义是什么?它们在不同的系统和实现方式上有何不同?
  • 我可以计算最低堆栈要求吗?我应该吗?

我在 i686 Debian 系统上:

$ uname -a
Linux REDACTED 3.16.0-4-686-pae #1 SMP Debian 3.16.7-ckt25-2+deb8u3 (2016-07-02) i686 GNU/Linux

最佳答案

  • Why doesn't clone segfault between 24 and 583 bytes of stack?

确实如此,但因为它是一个单独的进程,所以您看不到它。在 24 之前,发生段错误的不是 child ,而是试图设置 child 的 parent 。尝试使用 strace -ff 来查看这种情况。

  • How does child fail silently with too little stack?

当 child 死亡时, parent 会收到通知。在这种情况下,父级(执行 clone() 调用的父级)不会对此通知执行任何操作。它在 24 以下不是“静默”的原因是因为那是 parent 去世的时候,在这种情况下你的 shell 会收到通知。

  • What is all that stack space used for?
  • What is the significance of 24 and 584 bytes? How do they vary on different systems and implementations?

前 24 个(还有一点)用于设置对 child 的函数调用。因为它是一个普通函数,完成后它会返回到调用函数。这意味着 clone 必须设置一个调用函数以返回(一个干净地终止子进程的函数)。

584(和一点点)显然是调用函数、您的函数、write 和任何 write 调用的局部变量所需的内存量。

我写“(和一点)”的原因是因为在 stack 之前可能有一些内存可用并被 clone 滥用child 当房间用完时。尝试在克隆之后添加一个 free(stack) 以查看滥用的结果。

  • Can I calculate a minimum stack requirement? Should I?

一般来说你可能不应该。它需要对您的功能和使用的外部功能进行非常深入的分析。就像“普通”程序一样,我建议使用默认值(如果我没记错的话,在 Linux 上是 8MB)。只有当你有严格的内存要求(或堆栈溢出问题)时,你才应该开始担心这些事情。

关于c - Linux 克隆调用的最小堆栈大小?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38939650/

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