gpt4 book ai didi

C SECCOMP 阻塞或关闭 STDIN/STDOUT

转载 作者:太空狗 更新时间:2023-10-29 12:38:55 27 4
gpt4 key购买 nike

我现在正在实现在 fork 之后在子进程中运行另一个程序。

int main(int argc, char *argv[]) {
pid_t pid = 0;
int status;

struct user_regs_struct regs;

prctl(PR_SET_NO_NEW_PRIVS, 1, 0, 0, 0);
prctl(PR_SET_DUMPABLE, 0);

scmp_filter_ctx ctx;
ctx = seccomp_init(SCMP_ACT_KILL); // default action: kill

// build rules for whitelist of system calls
for (int i = 0; i < size_of_whitelist_syscall; i++) {
seccomp_rule_add(ctx, SCMP_ACT_ALLOW, whitelist_syscall[i], 0);
}

pid = fork();

if (pid != 0) {
while (waitpid(pid, &status, 0)) {
if (WIFEXITED(status)) {
fprintf(stderr, "terminated with code %d\n", WEXITSTATUS(status));
break;
} else if (WIFSIGNALED(status)) {
if (WTERMSIG(status) == 31) {
fprintf(stderr, "terminated by system call violation\n");
} else {
fprintf(stderr, "terminated by signal %d\n", WTERMSIG(status));
}
break;
}

ptrace(PTRACE_GETREGS, pid, NULL, &regs);


//fprintf(stderr, "%s(%lld) from pid %d\n", callname(REG(regs)), REG(regs), pid);

ptrace(PTRACE_SYSCALL, pid, NULL, NULL);
}
} else if (pid == 0) {
FILE *fp_in = freopen("std.in", "r", stdin);
FILE *fp_out = freopen("std.out", "w", stdout);
FILE *fp_error = freopen("err.out", "a+", stderr);

ptrace(PTRACE_TRACEME, 0, NULL, NULL);

seccomp_load(ctx);
execl("/usr/bin/java", "/usr/bin/java", "Test", NULL);
//execl("/usr/bin/python3", "python", "./test.py", NULL);
//execl("./test.out", "./test.out", NULL);
seccomp_release(ctx);

fclose(fp_in);
fclose(fp_out);
fclose(fp_error);

exit(0);
} else {
perror("failed to fork");
}

return 0;
}


int whitelist_syscall[] = {
SCMP_SYS(access),
SCMP_SYS(arch_prctl),
SCMP_SYS(brk),
SCMP_SYS(clone),
SCMP_SYS(close),
SCMP_SYS(dup),
SCMP_SYS(execve),
SCMP_SYS(exit_group),
SCMP_SYS(fcntl),
SCMP_SYS(fstat),
SCMP_SYS(futex),
SCMP_SYS(getcwd),
SCMP_SYS(getdents),
SCMP_SYS(getegid),
SCMP_SYS(geteuid),
SCMP_SYS(getgid),
SCMP_SYS(getpid),
SCMP_SYS(getrandom),
SCMP_SYS(getuid),
SCMP_SYS(ioctl),
SCMP_SYS(lseek),
SCMP_SYS(lstat),
SCMP_SYS(mmap),
SCMP_SYS(mprotect),
SCMP_SYS(munmap),
SCMP_SYS(openat),
SCMP_SYS(prlimit64),
SCMP_SYS(read),
SCMP_SYS(readlink),
SCMP_SYS(rt_sigaction),
SCMP_SYS(rt_sigprocmask),
SCMP_SYS(set_robust_list),
SCMP_SYS(set_tid_address),
SCMP_SYS(sigaltstack),
SCMP_SYS(stat),
SCMP_SYS(sysinfo),
SCMP_SYS(write)
};

int size_of_whitelist_syscall = sizeof(whitelist_syscall) / sizeof(int);

如上所示,我正在测试运行只是简单的“Hello world”的 C/Python/Java 程序。

seccomp_load(ctx);

execl("/usr/bin/java", "/usr/bin/java", "Test", NULL);
//execl("/usr/bin/python3", "python", "./test.py", NULL);
//execl("./test.out", "./test.out", NULL);

我允许使用 seccomp 进行一些白名单系统调用。

运行 Python 和 C 程序并将它们的输出重定向到文件。

在 Java 的情况下,子进程正常终止,但如果我加载 seccomp,输出就会消失。否则,我会看到“Hello World”消息。

谢谢。

最佳答案

我自己回答。

  1. 我尝试在 JVM 上运行 Java 程序。它没有向 parent 发送正确的信号,所以看起来一切都很好。

  2. 我的问题与 I/O 问题、缓冲 I/O、刷新或文件描述符无关。

  3. 我通过逐行删除系统调用来测试 ALLOW 规则。

以下是 JVM 的最少系统调用:

    SCMP_SYS(access),
SCMP_SYS(arch_prctl),
SCMP_SYS(brk),
SCMP_SYS(clock_getres),
SCMP_SYS(clone),
SCMP_SYS(close),
SCMP_SYS(connect),
SCMP_SYS(execve),
SCMP_SYS(exit_group),
SCMP_SYS(fchdir),
SCMP_SYS(fcntl),
SCMP_SYS(fstat),
SCMP_SYS(ftruncate),
SCMP_SYS(futex),
SCMP_SYS(getcwd),
SCMP_SYS(getdents),
SCMP_SYS(geteuid),
SCMP_SYS(getpid),
SCMP_SYS(gettid),
SCMP_SYS(getuid),
SCMP_SYS(kexec_load),
SCMP_SYS(kill),
SCMP_SYS(lseek),
SCMP_SYS(lstat),
SCMP_SYS(mkdir),
SCMP_SYS(mmap),
SCMP_SYS(mprotect),
SCMP_SYS(munmap),
SCMP_SYS(openat),
SCMP_SYS(prctl),
SCMP_SYS(pread64),
SCMP_SYS(prlimit64),
SCMP_SYS(pselect6),
SCMP_SYS(read),
SCMP_SYS(readlink),
SCMP_SYS(rt_sigaction),
SCMP_SYS(rt_sigprocmask),
SCMP_SYS(rt_sigreturn),
SCMP_SYS(sched_getaffinity),
SCMP_SYS(sched_yield),
SCMP_SYS(set_robust_list),
SCMP_SYS(set_tid_address),
SCMP_SYS(socket),
SCMP_SYS(stat),
SCMP_SYS(sysinfo),
SCMP_SYS(uname),
SCMP_SYS(unlink),
SCMP_SYS(write)

关于C SECCOMP 阻塞或关闭 STDIN/STDOUT,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/57100617/

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