Blame SOURCES/0004-strace.c-introduce-struct-tcb_wait_data.patch

2eae47
From acdd2e8d3d1551b41170a24951addb80b0b0d423 Mon Sep 17 00:00:00 2001
6cf6ff
From: "Dmitry V. Levin" <ldv@altlinux.org>
6cf6ff
Date: Tue, 14 Aug 2018 13:43:34 +0000
2eae47
Subject: [PATCH 04/27] strace.c: introduce struct tcb_wait_data
6cf6ff
6cf6ff
Introduce a new structure to pass information between next_event(),
6cf6ff
restart_delayed_tcb(), and dispatch_event().
6cf6ff
6cf6ff
This is going to be used by a subsequent change of next_event().
6cf6ff
6cf6ff
* strace.c (struct tcb_wait_data): New type.
6cf6ff
(next_event): Remove parameters, return a pointer
6cf6ff
to const struct tcb_wait_data.  Return NULL instead of TE_BREAK.
6cf6ff
(dispatch_event): Replace all parameters with a pointer
6cf6ff
to const struct tcb_wait_data, obtain the trace event, siginfo,
6cf6ff
and status from its fields.
6cf6ff
(restart_delayed_tcb): Add local struct tcb_wait_data variable
6cf6ff
with te field set to TE_RESTART, pass it to dispatch_event().
6cf6ff
(main): Remove status and si variables, update next_event()
6cf6ff
and dispatch_event() invocations.
6cf6ff
6cf6ff
Co-Authored-by: Eugene Syromyatnikov <evgsyr@gmail.com>
6cf6ff
---
6cf6ff
 strace.c | 107 ++++++++++++++++++++++++++++++++++++++++-----------------------
6cf6ff
 1 file changed, 69 insertions(+), 38 deletions(-)
6cf6ff
6cf6ff
diff --git a/strace.c b/strace.c
6cf6ff
index cd04b98..6d70d20 100644
6cf6ff
--- a/strace.c
6cf6ff
+++ b/strace.c
6cf6ff
@@ -158,6 +158,12 @@ static bool open_append;
6cf6ff
 struct tcb *printing_tcp;
6cf6ff
 static struct tcb *current_tcp;
6cf6ff
 
6cf6ff
+struct tcb_wait_data {
6cf6ff
+	enum trace_event te; /**< Event passed to dispatch_event() */
6cf6ff
+	int status;          /**< status, returned by wait4() */
6cf6ff
+	siginfo_t si;        /**< siginfo, returned by PTRACE_GETSIGINFO */
6cf6ff
+};
6cf6ff
+
6cf6ff
 static struct tcb **tcbtab;
6cf6ff
 static unsigned int nprocs;
6cf6ff
 static size_t tcbtabsize;
6cf6ff
@@ -2226,16 +2232,19 @@ print_event_exit(struct tcb *tcp)
6cf6ff
 	line_ended();
6cf6ff
 }
6cf6ff
 
6cf6ff
-static enum trace_event
6cf6ff
-next_event(int *pstatus, siginfo_t *si)
6cf6ff
+static const struct tcb_wait_data *
6cf6ff
+next_event(void)
6cf6ff
 {
6cf6ff
+	static struct tcb_wait_data wait_data;
6cf6ff
+
6cf6ff
 	int pid;
6cf6ff
 	int status;
6cf6ff
 	struct tcb *tcp;
6cf6ff
+	struct tcb_wait_data *wd = &wait_data;
6cf6ff
 	struct rusage ru;
6cf6ff
 
6cf6ff
 	if (interrupted)
6cf6ff
-		return TE_BREAK;
6cf6ff
+		return NULL;
6cf6ff
 
6cf6ff
 	/*
6cf6ff
 	 * Used to exit simply when nprocs hits zero, but in this testcase:
6cf6ff
@@ -2255,7 +2264,7 @@ next_event(int *pstatus, siginfo_t *si)
6cf6ff
 		 * on exit. Oh well...
6cf6ff
 		 */
6cf6ff
 		if (nprocs == 0)
6cf6ff
-			return TE_BREAK;
6cf6ff
+			return NULL;
6cf6ff
 	}
6cf6ff
 
6cf6ff
 	const bool unblock_delay_timer = is_delay_timer_armed();
6cf6ff
@@ -2278,7 +2287,7 @@ next_event(int *pstatus, siginfo_t *si)
6cf6ff
 	 * then the system call will be interrupted and
6cf6ff
 	 * the expiration will be handled by the signal handler.
6cf6ff
 	 */
6cf6ff
-	pid = wait4(-1, pstatus, __WALL, (cflag ? &ru : NULL));
6cf6ff
+	pid = wait4(-1, &status, __WALL, (cflag ? &ru : NULL));
6cf6ff
 	const int wait_errno = errno;
6cf6ff
 
6cf6ff
 	/*
6cf6ff
@@ -2292,14 +2301,16 @@ next_event(int *pstatus, siginfo_t *si)
6cf6ff
 		sigprocmask(SIG_BLOCK, &timer_set, NULL);
6cf6ff
 
6cf6ff
 		if (restart_failed)
6cf6ff
-			return TE_BREAK;
6cf6ff
+			return NULL;
6cf6ff
 	}
6cf6ff
 
6cf6ff
 	if (pid < 0) {
6cf6ff
-		if (wait_errno == EINTR)
6cf6ff
-			return TE_NEXT;
6cf6ff
+		if (wait_errno == EINTR) {
6cf6ff
+			wd->te = TE_NEXT;
6cf6ff
+			return wd;
6cf6ff
+		}
6cf6ff
 		if (nprocs == 0 && wait_errno == ECHILD)
6cf6ff
-			return TE_BREAK;
6cf6ff
+			return NULL;
6cf6ff
 		/*
6cf6ff
 		 * If nprocs > 0, ECHILD is not expected,
6cf6ff
 		 * treat it as any other error here:
6cf6ff
@@ -2308,12 +2319,13 @@ next_event(int *pstatus, siginfo_t *si)
6cf6ff
 		perror_msg_and_die("wait4(__WALL)");
6cf6ff
 	}
6cf6ff
 
6cf6ff
-	status = *pstatus;
6cf6ff
+	wd->status = status;
6cf6ff
 
6cf6ff
 	if (pid == popen_pid) {
6cf6ff
 		if (!WIFSTOPPED(status))
6cf6ff
 			popen_pid = 0;
6cf6ff
-		return TE_NEXT;
6cf6ff
+		wd->te = TE_NEXT;
6cf6ff
+		return wd;
6cf6ff
 	}
6cf6ff
 
6cf6ff
 	if (debug_flag)
6cf6ff
@@ -2324,8 +2336,10 @@ next_event(int *pstatus, siginfo_t *si)
6cf6ff
 
6cf6ff
 	if (!tcp) {
6cf6ff
 		tcp = maybe_allocate_tcb(pid, status);
6cf6ff
-		if (!tcp)
6cf6ff
-			return TE_NEXT;
6cf6ff
+		if (!tcp) {
6cf6ff
+			wd->te = TE_NEXT;
6cf6ff
+			return wd;
6cf6ff
+		}
6cf6ff
 	}
6cf6ff
 
6cf6ff
 	clear_regs(tcp);
6cf6ff
@@ -2342,11 +2356,15 @@ next_event(int *pstatus, siginfo_t *si)
6cf6ff
 		tcp->stime = stime;
6cf6ff
 	}
6cf6ff
 
6cf6ff
-	if (WIFSIGNALED(status))
6cf6ff
-		return TE_SIGNALLED;
6cf6ff
+	if (WIFSIGNALED(status)) {
6cf6ff
+		wd->te = TE_SIGNALLED;
6cf6ff
+		return wd;
6cf6ff
+	}
6cf6ff
 
6cf6ff
-	if (WIFEXITED(status))
6cf6ff
-		return TE_EXITED;
6cf6ff
+	if (WIFEXITED(status)) {
6cf6ff
+		wd->te = TE_EXITED;
6cf6ff
+		return wd;
6cf6ff
+	}
6cf6ff
 
6cf6ff
 	/*
6cf6ff
 	 * As WCONTINUED flag has not been specified to wait4,
6cf6ff
@@ -2373,19 +2391,19 @@ next_event(int *pstatus, siginfo_t *si)
6cf6ff
 		if (sig == SIGSTOP && (tcp->flags & TCB_IGNORE_ONE_SIGSTOP)) {
6cf6ff
 			debug_func_msg("ignored SIGSTOP on pid %d", tcp->pid);
6cf6ff
 			tcp->flags &= ~TCB_IGNORE_ONE_SIGSTOP;
6cf6ff
-			return TE_RESTART;
6cf6ff
+			wd->te = TE_RESTART;
6cf6ff
 		} else if (sig == syscall_trap_sig) {
6cf6ff
-			return TE_SYSCALL_STOP;
6cf6ff
+			wd->te = TE_SYSCALL_STOP;
6cf6ff
 		} else {
6cf6ff
-			*si = (siginfo_t) {};
6cf6ff
+			memset(&wd->si, 0, sizeof(wd->si));
6cf6ff
 			/*
6cf6ff
 			 * True if tracee is stopped by signal
6cf6ff
 			 * (as opposed to "tracee received signal").
6cf6ff
 			 * TODO: shouldn't we check for errno == EINVAL too?
6cf6ff
 			 * We can get ESRCH instead, you know...
6cf6ff
 			 */
6cf6ff
-			bool stopped = ptrace(PTRACE_GETSIGINFO, pid, 0, si) < 0;
6cf6ff
-			return stopped ? TE_GROUP_STOP : TE_SIGNAL_DELIVERY_STOP;
6cf6ff
+			bool stopped = ptrace(PTRACE_GETSIGINFO, pid, 0, &wd->si) < 0;
6cf6ff
+			wd->te = stopped ? TE_GROUP_STOP : TE_SIGNAL_DELIVERY_STOP;
6cf6ff
 		}
6cf6ff
 		break;
6cf6ff
 	case PTRACE_EVENT_STOP:
6cf6ff
@@ -2398,16 +2416,23 @@ next_event(int *pstatus, siginfo_t *si)
6cf6ff
 		case SIGTSTP:
6cf6ff
 		case SIGTTIN:
6cf6ff
 		case SIGTTOU:
6cf6ff
-			return TE_GROUP_STOP;
6cf6ff
+			wd->te = TE_GROUP_STOP;
6cf6ff
+			break;
6cf6ff
+		default:
6cf6ff
+			wd->te = TE_RESTART;
6cf6ff
 		}
6cf6ff
-		return TE_RESTART;
6cf6ff
+		break;
6cf6ff
 	case PTRACE_EVENT_EXEC:
6cf6ff
-		return TE_STOP_BEFORE_EXECVE;
6cf6ff
+		wd->te = TE_STOP_BEFORE_EXECVE;
6cf6ff
+		break;
6cf6ff
 	case PTRACE_EVENT_EXIT:
6cf6ff
-		return TE_STOP_BEFORE_EXIT;
6cf6ff
+		wd->te = TE_STOP_BEFORE_EXIT;
6cf6ff
+		break;
6cf6ff
 	default:
6cf6ff
-		return TE_RESTART;
6cf6ff
+		wd->te = TE_RESTART;
6cf6ff
 	}
6cf6ff
+
6cf6ff
+	return wd;
6cf6ff
 }
6cf6ff
 
6cf6ff
 static int
6cf6ff
@@ -2436,12 +2461,18 @@ trace_syscall(struct tcb *tcp, unsigned int *sig)
6cf6ff
 
6cf6ff
 /* Returns true iff the main trace loop has to continue. */
6cf6ff
 static bool
6cf6ff
-dispatch_event(enum trace_event ret, int *pstatus, siginfo_t *si)
6cf6ff
+dispatch_event(const struct tcb_wait_data *wd)
6cf6ff
 {
6cf6ff
 	unsigned int restart_op = PTRACE_SYSCALL;
6cf6ff
 	unsigned int restart_sig = 0;
6cf6ff
+	enum trace_event te = wd ? wd->te : TE_BREAK;
6cf6ff
+	/*
6cf6ff
+	 * Copy wd->status to a non-const variable to workaround glibc bugs
6cf6ff
+	 * around union wait fixed by glibc commit glibc-2.24~391
6cf6ff
+	 */
6cf6ff
+	int status = wd ? wd->status : 0;
6cf6ff
 
6cf6ff
-	switch (ret) {
6cf6ff
+	switch (te) {
6cf6ff
 	case TE_BREAK:
6cf6ff
 		return false;
6cf6ff
 
6cf6ff
@@ -2469,17 +2500,17 @@ dispatch_event(enum trace_event ret, int *pstatus, siginfo_t *si)
6cf6ff
 		break;
6cf6ff
 
6cf6ff
 	case TE_SIGNAL_DELIVERY_STOP:
6cf6ff
-		restart_sig = WSTOPSIG(*pstatus);
6cf6ff
-		print_stopped(current_tcp, si, restart_sig);
6cf6ff
+		restart_sig = WSTOPSIG(status);
6cf6ff
+		print_stopped(current_tcp, &wd->si, restart_sig);
6cf6ff
 		break;
6cf6ff
 
6cf6ff
 	case TE_SIGNALLED:
6cf6ff
-		print_signalled(current_tcp, current_tcp->pid, *pstatus);
6cf6ff
+		print_signalled(current_tcp, current_tcp->pid, status);
6cf6ff
 		droptcb(current_tcp);
6cf6ff
 		return true;
6cf6ff
 
6cf6ff
 	case TE_GROUP_STOP:
6cf6ff
-		restart_sig = WSTOPSIG(*pstatus);
6cf6ff
+		restart_sig = WSTOPSIG(status);
6cf6ff
 		print_stopped(current_tcp, NULL, restart_sig);
6cf6ff
 		if (use_seize) {
6cf6ff
 			/*
6cf6ff
@@ -2494,7 +2525,7 @@ dispatch_event(enum trace_event ret, int *pstatus, siginfo_t *si)
6cf6ff
 		break;
6cf6ff
 
6cf6ff
 	case TE_EXITED:
6cf6ff
-		print_exited(current_tcp, current_tcp->pid, *pstatus);
6cf6ff
+		print_exited(current_tcp, current_tcp->pid, status);
6cf6ff
 		droptcb(current_tcp);
6cf6ff
 		return true;
6cf6ff
 
6cf6ff
@@ -2577,13 +2608,15 @@ dispatch_event(enum trace_event ret, int *pstatus, siginfo_t *si)
6cf6ff
 static bool
6cf6ff
 restart_delayed_tcb(struct tcb *const tcp)
6cf6ff
 {
6cf6ff
+	const struct tcb_wait_data wd = { .te = TE_RESTART };
6cf6ff
+
6cf6ff
 	debug_func_msg("pid %d", tcp->pid);
6cf6ff
 
6cf6ff
 	tcp->flags &= ~TCB_DELAYED;
6cf6ff
 
6cf6ff
 	struct tcb *const prev_tcp = current_tcp;
6cf6ff
 	current_tcp = tcp;
6cf6ff
-	bool ret = dispatch_event(TE_RESTART, NULL, NULL);
6cf6ff
+	bool ret = dispatch_event(&wd;;
6cf6ff
 	current_tcp = prev_tcp;
6cf6ff
 
6cf6ff
 	return ret;
6cf6ff
@@ -2694,9 +2727,7 @@ main(int argc, char *argv[])
6cf6ff
 
6cf6ff
 	exit_code = !nprocs;
6cf6ff
 
6cf6ff
-	int status;
6cf6ff
-	siginfo_t si;
6cf6ff
-	while (dispatch_event(next_event(&status, &si), &status, &si))
6cf6ff
+	while (dispatch_event(next_event()))
6cf6ff
 		;
6cf6ff
 	terminate();
6cf6ff
 }
6cf6ff
-- 
6cf6ff
2.1.4
6cf6ff