tt.c
来自「LINUX 2.6.17.4的源码」· C语言 代码 · 共 198 行
C
198 行
/* * Copyright (C) 2000, 2001, 2002 Jeff Dike (jdike@karaya.com) * Licensed under the GPL */#include <stdio.h>#include <unistd.h>#include <signal.h>#include <sched.h>#include <errno.h>#include <stdarg.h>#include <stdlib.h>#include <setjmp.h>#include <sys/time.h>#include <sys/ptrace.h>#include <linux/ptrace.h>#include <sys/wait.h>#include <sys/mman.h>#include <asm/ptrace.h>#include <asm/unistd.h>#include <asm/page.h>#include "user_util.h"#include "kern_util.h"#include "user.h"#include "signal_kern.h"#include "sysdep/ptrace.h"#include "sysdep/sigcontext.h"#include "irq_user.h"#include "ptrace_user.h"#include "init.h"#include "os.h"#include "uml-config.h"#include "choose-mode.h"#include "mode.h"#include "tempfile.h"int protect_memory(unsigned long addr, unsigned long len, int r, int w, int x, int must_succeed){ int err; err = os_protect_memory((void *) addr, len, r, w, x); if(err < 0){ if(must_succeed) panic("protect failed, err = %d", -err); else return(err); } return(0);}void kill_child_dead(int pid){ kill(pid, SIGKILL); kill(pid, SIGCONT); do { int n; CATCH_EINTR(n = waitpid(pid, NULL, 0)); if (n > 0) kill(pid, SIGCONT); else break; } while(1);}void stop(void){ while(1) sleep(1000000);}int wait_for_stop(int pid, int sig, int cont_type, void *relay){ sigset_t *relay_signals = relay; int status, ret; while(1){ CATCH_EINTR(ret = waitpid(pid, &status, WUNTRACED)); if((ret < 0) || !WIFSTOPPED(status) || (WSTOPSIG(status) != sig)){ if(ret < 0){ printk("wait failed, errno = %d\n", errno); } else if(WIFEXITED(status)) printk("process %d exited with status %d\n", pid, WEXITSTATUS(status)); else if(WIFSIGNALED(status)) printk("process %d exited with signal %d\n", pid, WTERMSIG(status)); else if((WSTOPSIG(status) == SIGVTALRM) || (WSTOPSIG(status) == SIGALRM) || (WSTOPSIG(status) == SIGIO) || (WSTOPSIG(status) == SIGPROF) || (WSTOPSIG(status) == SIGCHLD) || (WSTOPSIG(status) == SIGWINCH) || (WSTOPSIG(status) == SIGINT)){ ptrace(cont_type, pid, 0, WSTOPSIG(status)); continue; } else if((relay_signals != NULL) && sigismember(relay_signals, WSTOPSIG(status))){ ptrace(cont_type, pid, 0, WSTOPSIG(status)); continue; } else printk("process %d stopped with signal %d\n", pid, WSTOPSIG(status)); panic("wait_for_stop failed to wait for %d to stop " "with %d\n", pid, sig); } return(status); }}void forward_ipi(int fd, int pid){ int err; err = os_set_owner(fd, pid); if(err < 0) printk("forward_ipi: set_owner failed, fd = %d, me = %d, " "target = %d, err = %d\n", fd, os_getpid(), pid, -err);}/* *------------------------- * only for tt mode (will be deleted in future...) *------------------------- */struct tramp { int (*tramp)(void *); void *tramp_data; unsigned long temp_stack; int flags; int pid;};/* See above for why sigkill is here */int sigkill = SIGKILL;int outer_tramp(void *arg){ struct tramp *t; int sig = sigkill; t = arg; t->pid = clone(t->tramp, (void *) t->temp_stack + page_size()/2, t->flags, t->tramp_data); if(t->pid > 0) wait_for_stop(t->pid, SIGSTOP, PTRACE_CONT, NULL); kill(os_getpid(), sig); _exit(0);}int start_fork_tramp(void *thread_arg, unsigned long temp_stack, int clone_flags, int (*tramp)(void *)){ struct tramp arg; unsigned long sp; int new_pid, status, err; /* The trampoline will run on the temporary stack */ sp = stack_sp(temp_stack); clone_flags |= CLONE_FILES | SIGCHLD; arg.tramp = tramp; arg.tramp_data = thread_arg; arg.temp_stack = temp_stack; arg.flags = clone_flags; /* Start the process and wait for it to kill itself */ new_pid = clone(outer_tramp, (void *) sp, clone_flags, &arg); if(new_pid < 0) return(new_pid); CATCH_EINTR(err = waitpid(new_pid, &status, 0)); if(err < 0) panic("Waiting for outer trampoline failed - errno = %d", errno); if(!WIFSIGNALED(status) || (WTERMSIG(status) != SIGKILL)) panic("outer trampoline didn't exit with SIGKILL, " "status = %d", status); return(arg.pid);}void forward_pending_sigio(int target){ sigset_t sigs; if(sigpending(&sigs)) panic("forward_pending_sigio : sigpending failed"); if(sigismember(&sigs, SIGIO)) kill(target, SIGIO);}
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?