2014-08-26 17:10:19 -04:00
|
|
|
|
// Copyright 2014 The Crashpad Authors. All rights reserved.
|
|
|
|
|
//
|
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
|
//
|
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
//
|
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
|
// limitations under the License.
|
|
|
|
|
|
test: Move util/test to its own top-level directory, test.
After 9e79ea1da719, it no longer makes sense for crashpad_util_test_lib
to “hide” in util/util_test.gyp. All of util/test is moved to its own
top-level directory, test, which all other test code is allowed to
depend on. test, too, is allowed to depend on all other non-test code.
In a future change, when crashpad_util_test_lib gains a dependency on
crashpad_client, it won’t look so weird for something in util (even
though it’s in util/test) to depend on something in client, because the
thing that needs to depend on client will live in test, not util.
BUG=crashpad:33
R=scottmg@chromium.org
Review URL: https://codereview.chromium.org/1051533002
2015-03-31 17:44:14 -04:00
|
|
|
|
#include "test/multiprocess.h"
|
2014-08-26 17:10:19 -04:00
|
|
|
|
|
|
|
|
|
#include <signal.h>
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
#include <sys/wait.h>
|
2015-01-28 14:49:42 -08:00
|
|
|
|
#include <unistd.h>
|
2014-08-26 17:10:19 -04:00
|
|
|
|
|
2016-04-25 12:13:07 -07:00
|
|
|
|
#include <memory>
|
2014-08-26 17:10:19 -04:00
|
|
|
|
#include <string>
|
|
|
|
|
|
|
|
|
|
#include "base/auto_reset.h"
|
|
|
|
|
#include "base/files/scoped_file.h"
|
2014-09-22 13:06:12 -04:00
|
|
|
|
#include "base/logging.h"
|
2014-09-22 13:17:17 -04:00
|
|
|
|
#include "base/posix/eintr_wrapper.h"
|
2014-08-26 17:10:19 -04:00
|
|
|
|
#include "base/strings/stringprintf.h"
|
|
|
|
|
#include "gtest/gtest.h"
|
test: Move util/test to its own top-level directory, test.
After 9e79ea1da719, it no longer makes sense for crashpad_util_test_lib
to “hide” in util/util_test.gyp. All of util/test is moved to its own
top-level directory, test, which all other test code is allowed to
depend on. test, too, is allowed to depend on all other non-test code.
In a future change, when crashpad_util_test_lib gains a dependency on
crashpad_client, it won’t look so weird for something in util (even
though it’s in util/test) to depend on something in client, because the
thing that needs to depend on client will live in test, not util.
BUG=crashpad:33
R=scottmg@chromium.org
Review URL: https://codereview.chromium.org/1051533002
2015-03-31 17:44:14 -04:00
|
|
|
|
#include "test/errors.h"
|
2014-08-26 17:10:19 -04:00
|
|
|
|
#include "util/misc/scoped_forbid_return.h"
|
|
|
|
|
|
|
|
|
|
namespace crashpad {
|
|
|
|
|
namespace test {
|
|
|
|
|
|
|
|
|
|
namespace internal {
|
|
|
|
|
|
|
|
|
|
struct MultiprocessInfo {
|
|
|
|
|
MultiprocessInfo()
|
|
|
|
|
: pipe_c2p_read(-1),
|
|
|
|
|
pipe_c2p_write(-1),
|
|
|
|
|
pipe_p2c_read(-1),
|
|
|
|
|
pipe_p2c_write(-1),
|
|
|
|
|
child_pid(0) {}
|
|
|
|
|
|
|
|
|
|
base::ScopedFD pipe_c2p_read; // child to parent
|
|
|
|
|
base::ScopedFD pipe_c2p_write; // child to parent
|
|
|
|
|
base::ScopedFD pipe_p2c_read; // parent to child
|
|
|
|
|
base::ScopedFD pipe_p2c_write; // parent to child
|
|
|
|
|
pid_t child_pid; // valid only in parent
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
} // namespace internal
|
|
|
|
|
|
2014-09-09 17:04:47 -04:00
|
|
|
|
Multiprocess::Multiprocess()
|
2014-10-14 11:10:45 -04:00
|
|
|
|
: info_(nullptr),
|
2014-09-09 17:04:47 -04:00
|
|
|
|
code_(EXIT_SUCCESS),
|
|
|
|
|
reason_(kTerminationNormal) {
|
2014-08-26 17:10:19 -04:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Multiprocess::Run() {
|
2014-10-14 11:10:45 -04:00
|
|
|
|
ASSERT_EQ(nullptr, info_);
|
2016-04-25 12:13:07 -07:00
|
|
|
|
std::unique_ptr<internal::MultiprocessInfo> info(
|
|
|
|
|
new internal::MultiprocessInfo);
|
2014-08-26 17:10:19 -04:00
|
|
|
|
base::AutoReset<internal::MultiprocessInfo*> reset_info(&info_, info.get());
|
|
|
|
|
|
2014-10-09 15:08:54 -04:00
|
|
|
|
ASSERT_NO_FATAL_FAILURE(PreFork());
|
2014-08-26 17:10:19 -04:00
|
|
|
|
|
|
|
|
|
pid_t pid = fork();
|
|
|
|
|
ASSERT_GE(pid, 0) << ErrnoMessage("fork");
|
|
|
|
|
|
|
|
|
|
if (pid > 0) {
|
|
|
|
|
info_->child_pid = pid;
|
|
|
|
|
|
|
|
|
|
RunParent();
|
|
|
|
|
|
|
|
|
|
// Waiting for the child happens here instead of in RunParent() because even
|
|
|
|
|
// if RunParent() returns early due to a gtest fatal assertion failure, the
|
|
|
|
|
// child should still be reaped.
|
|
|
|
|
|
|
|
|
|
// This will make the parent hang up on the child as much as would be
|
|
|
|
|
// visible from the child’s perspective. The child’s side of the pipe will
|
|
|
|
|
// be broken, the child’s remote port will become a dead name, and an
|
|
|
|
|
// attempt by the child to look up the service will fail. If this weren’t
|
|
|
|
|
// done, the child might hang while waiting for a parent that has already
|
|
|
|
|
// triggered a fatal assertion failure to do something.
|
|
|
|
|
info.reset();
|
2014-10-14 11:10:45 -04:00
|
|
|
|
info_ = nullptr;
|
2014-08-26 17:10:19 -04:00
|
|
|
|
|
|
|
|
|
int status;
|
2014-09-22 13:17:17 -04:00
|
|
|
|
pid_t wait_pid = HANDLE_EINTR(waitpid(pid, &status, 0));
|
2014-08-26 17:10:19 -04:00
|
|
|
|
ASSERT_EQ(pid, wait_pid) << ErrnoMessage("waitpid");
|
2014-09-09 17:04:47 -04:00
|
|
|
|
|
|
|
|
|
TerminationReason reason;
|
|
|
|
|
int code;
|
|
|
|
|
std::string message;
|
|
|
|
|
if (WIFEXITED(status)) {
|
|
|
|
|
reason = kTerminationNormal;
|
|
|
|
|
code = WEXITSTATUS(status);
|
2016-06-20 13:41:40 -04:00
|
|
|
|
message = base::StringPrintf("Child exited with code %d", code);
|
2014-09-09 17:04:47 -04:00
|
|
|
|
} else if (WIFSIGNALED(status)) {
|
|
|
|
|
reason = kTerminationSignal;
|
|
|
|
|
code = WTERMSIG(status);
|
|
|
|
|
message =
|
2016-06-20 13:41:40 -04:00
|
|
|
|
base::StringPrintf("Child terminated by signal %d (%s)%s",
|
2014-09-09 17:04:47 -04:00
|
|
|
|
code,
|
|
|
|
|
strsignal(code),
|
|
|
|
|
WCOREDUMP(status) ? " (core dumped)" : "");
|
|
|
|
|
} else {
|
|
|
|
|
FAIL() << "Unknown termination reason";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (reason_ == kTerminationNormal) {
|
2016-06-20 13:41:40 -04:00
|
|
|
|
message += base::StringPrintf(", expected exit with code %d", code_);
|
|
|
|
|
} else if (reason_ == kTerminationSignal) {
|
|
|
|
|
message += base::StringPrintf(", expected termination by signal %d (%s)",
|
|
|
|
|
code_,
|
|
|
|
|
strsignal(code_));
|
2014-09-09 17:04:47 -04:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (reason != reason_ || code != code_) {
|
|
|
|
|
ADD_FAILURE() << message;
|
2014-08-26 17:10:19 -04:00
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
RunChild();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-09-09 17:04:47 -04:00
|
|
|
|
void Multiprocess::SetExpectedChildTermination(TerminationReason reason,
|
|
|
|
|
int code) {
|
|
|
|
|
reason_ = reason;
|
|
|
|
|
code_ = code;
|
|
|
|
|
}
|
|
|
|
|
|
2014-08-26 17:10:19 -04:00
|
|
|
|
Multiprocess::~Multiprocess() {
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Multiprocess::PreFork() {
|
|
|
|
|
int pipe_fds_c2p[2];
|
|
|
|
|
int rv = pipe(pipe_fds_c2p);
|
|
|
|
|
ASSERT_EQ(0, rv) << ErrnoMessage("pipe");
|
|
|
|
|
|
|
|
|
|
info_->pipe_c2p_read.reset(pipe_fds_c2p[0]);
|
|
|
|
|
info_->pipe_c2p_write.reset(pipe_fds_c2p[1]);
|
|
|
|
|
|
|
|
|
|
int pipe_fds_p2c[2];
|
|
|
|
|
rv = pipe(pipe_fds_p2c);
|
|
|
|
|
ASSERT_EQ(0, rv) << ErrnoMessage("pipe");
|
|
|
|
|
|
|
|
|
|
info_->pipe_p2c_read.reset(pipe_fds_p2c[0]);
|
|
|
|
|
info_->pipe_p2c_write.reset(pipe_fds_p2c[1]);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pid_t Multiprocess::ChildPID() const {
|
|
|
|
|
EXPECT_NE(0, info_->child_pid);
|
|
|
|
|
return info_->child_pid;
|
|
|
|
|
}
|
|
|
|
|
|
2015-01-28 14:49:42 -08:00
|
|
|
|
FileHandle Multiprocess::ReadPipeHandle() const {
|
2014-08-26 17:10:19 -04:00
|
|
|
|
int fd = info_->child_pid ? info_->pipe_c2p_read.get()
|
|
|
|
|
: info_->pipe_p2c_read.get();
|
2014-09-22 13:06:12 -04:00
|
|
|
|
CHECK_NE(fd, -1);
|
2014-08-26 17:10:19 -04:00
|
|
|
|
return fd;
|
|
|
|
|
}
|
|
|
|
|
|
2015-01-28 14:49:42 -08:00
|
|
|
|
FileHandle Multiprocess::WritePipeHandle() const {
|
2014-08-26 17:10:19 -04:00
|
|
|
|
int fd = info_->child_pid ? info_->pipe_p2c_write.get()
|
|
|
|
|
: info_->pipe_c2p_write.get();
|
2014-09-22 13:06:12 -04:00
|
|
|
|
CHECK_NE(fd, -1);
|
2014-08-26 17:10:19 -04:00
|
|
|
|
return fd;
|
|
|
|
|
}
|
|
|
|
|
|
2014-09-22 13:06:12 -04:00
|
|
|
|
void Multiprocess::CloseReadPipe() {
|
|
|
|
|
if (info_->child_pid) {
|
|
|
|
|
info_->pipe_c2p_read.reset();
|
|
|
|
|
} else {
|
|
|
|
|
info_->pipe_p2c_read.reset();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Multiprocess::CloseWritePipe() {
|
|
|
|
|
if (info_->child_pid) {
|
|
|
|
|
info_->pipe_p2c_write.reset();
|
|
|
|
|
} else {
|
|
|
|
|
info_->pipe_c2p_write.reset();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-08-26 17:10:19 -04:00
|
|
|
|
void Multiprocess::RunParent() {
|
|
|
|
|
// The parent uses the read end of c2p and the write end of p2c.
|
|
|
|
|
info_->pipe_c2p_write.reset();
|
|
|
|
|
info_->pipe_p2c_read.reset();
|
|
|
|
|
|
|
|
|
|
MultiprocessParent();
|
|
|
|
|
|
|
|
|
|
info_->pipe_c2p_read.reset();
|
|
|
|
|
info_->pipe_p2c_write.reset();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Multiprocess::RunChild() {
|
|
|
|
|
ScopedForbidReturn forbid_return;
|
|
|
|
|
|
|
|
|
|
// The child uses the write end of c2p and the read end of p2c.
|
|
|
|
|
info_->pipe_c2p_read.reset();
|
|
|
|
|
info_->pipe_p2c_write.reset();
|
|
|
|
|
|
|
|
|
|
MultiprocessChild();
|
|
|
|
|
|
|
|
|
|
info_->pipe_c2p_write.reset();
|
|
|
|
|
info_->pipe_p2c_read.reset();
|
|
|
|
|
|
2014-10-07 17:28:50 -04:00
|
|
|
|
if (testing::Test::HasFailure()) {
|
2014-08-26 17:10:19 -04:00
|
|
|
|
// Trigger the ScopedForbidReturn destructor.
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2015-03-11 13:59:46 -04:00
|
|
|
|
// In a forked child, exit() is unsafe. Use _exit() instead.
|
|
|
|
|
_exit(EXIT_SUCCESS);
|
2014-08-26 17:10:19 -04:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} // namespace test
|
|
|
|
|
} // namespace crashpad
|