blob: a091426c89ac8e76554d507ec975169a4d9208a3 [file] [log] [blame]
/* -*- Mode: C++; tab-width: 8; c-basic-offset: 2; indent-tabs-mode: nil; -*- */
#ifndef RR_WAIT_STATUS_H_
#define RR_WAIT_STATUS_H_
#include <signal.h>
#include <memory>
#include <vector>
#include "core.h"
namespace rr {
class RecordTask;
class WaitStatus {
public:
explicit WaitStatus(int status = 0) : status(status) {}
explicit WaitStatus(const siginfo_t &info);
enum Type {
// Task exited normally.
EXIT,
// Task exited due to fatal signal.
FATAL_SIGNAL,
// Task is in a signal-delivery-stop.
SIGNAL_STOP,
// Task is in a group-stop. (See ptrace man page.)
// You must use PTRACE_SEIZE to generate PTRACE_EVENT_STOPs, or these
// will be treated as STOP_SIGNAL.
GROUP_STOP,
// Task is in a syscall-stop triggered by PTRACE_SYSCALL
// and PTRACE_O_TRACESYSGOOD.
SYSCALL_STOP,
// Task is in a PTRACE_EVENT stop, except for PTRACE_EVENT_STOP
// which is treated as GROUP_STOP.
PTRACE_EVENT
};
Type type() const;
// Exit code if type() == EXIT, otherwise -1.
int exit_code() const;
// Fatal signal if type() == FATAL_SIGNAL, otherwise zero.
int fatal_sig() const;
// Stop signal if type() == STOP_SIGNAL, otherwise zero. A zero signal
// (rare but observed via PTRACE_INTERRUPT) is converted to SIGSTOP.
int stop_sig() const;
// Stop signal if type() == GROUP_STOP, otherwise zero. A zero signal
// (rare but observed via PTRACE_INTERRUPT) is converted to SIGSTOP.
int group_stop() const;
bool is_syscall() const;
// ptrace event if type() == PTRACE_EVENT, otherwise zero.
int ptrace_event() const;
// For exit_code() and fatal_sig(), returns 0. For all other types
// returns the signal involved.
int ptrace_signal() const;
// True if this status means we reaped the exit status.
bool reaped() const { return exit_code() != -1 || fatal_sig() != 0; }
int get() const { return status; }
static WaitStatus for_exit_code(int code);
static WaitStatus for_fatal_sig(int sig);
static WaitStatus for_stop_sig(int sig);
static WaitStatus for_group_sig(int sig, RecordTask* t);
static WaitStatus for_syscall(RecordTask* t);
static WaitStatus for_ptrace_event(int ptrace_event);
// Fill in the si_status/si_code field of the siginfo_t struct to correspond
// this wait status.
template <typename Arch>
void fill_siginfo(typename Arch::siginfo_t *info, bool ptracer = true,
unsigned ptrace_options = 0);
private:
bool stopped() const;
int ptrace_event_code() const;
int stop_sig_code() const;
int status;
};
std::ostream& operator<<(std::ostream& stream, WaitStatus status);
} // namespace rr
#endif /* RR_WAIT_STATUS_H_ */