summaryrefslogtreecommitdiff
path: root/src/process.cpp
blob: e4cf0213cac7dc764fc201da8ceeb46130d16e34 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
#include "xorg/gtest/process.h"

#include <sys/types.h>
#include <sys/wait.h>
#include <unistd.h>

#include <algorithm>
#include <cerrno>
#include <csignal>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <stdexcept>
#include <vector>

struct xorg::testing::Process::Private {
  pid_t pid;
};

xorg::testing::Process::Process() : d_(new Private) {
  d_->pid = -1;
}

void xorg::testing::Process::Start(const std::string& program, va_list args) {
  if (d_->pid != -1)
    throw std::runtime_error("Attempting to start an already started process");

  d_->pid = vfork();

  if (d_->pid == -1) {
    throw std::runtime_error("Failed to fork child process");
  } else if (d_->pid == 0) { /* Child */
    close(0);
    close(1);
    close(2);

    std::vector<char*> argv;
    char* arg;
    do {
      arg = va_arg(args, char*);

      if (arg) {
        arg = strdup(arg);
        if (!arg) {
          std::for_each(argv.begin(), argv.end(), free);
          throw std::bad_alloc();
        }
      }

      argv.push_back(arg);
    } while (arg);

    execvp(program.c_str(), &argv[0]);

    std::for_each(argv.begin(), argv.end(), free);
    throw std::runtime_error("Failed to start process");
  }
}

void xorg::testing::Process::Start(const std::string& program, ...) {
  va_list list;
  va_start(list, program);
  Start(program, list);
  va_end(list); /* Shouldn't get here */
}

bool xorg::testing::Process::Terminate() {
  if (d_->pid == -1) {
    return false;
  } else if (d_->pid == 0) {
    /* Child */
    throw std::runtime_error("Child process tried to terminate itself");
  } else { /* Parent */
    if (kill(d_->pid, SIGTERM) < 0) {
      return false;
    }
  }
  return true;
}

bool xorg::testing::Process::Kill() {
  if (d_->pid == -1) {
    return false;
  } else if (d_->pid == 0) {
    /* Child */
    throw std::runtime_error("Child process tried to kill itself");
  } else { /* Parent */
    if (kill(d_->pid, SIGKILL) < 0) {
      return false;
    }
  }
  return true;
}

void xorg::testing::Process::SetEnv(const char* name, const char* value,
                                    bool overwrite) {
  if (setenv(name, value, overwrite) != 0)
    throw std::runtime_error("Failed to set environment variable in process");

  return;
}

const char* xorg::testing::Process::GetEnv(const char* name) {
  return getenv(name);
}

pid_t xorg::testing::Process::Pid() const {
  return d_->pid;
}