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
|
|
|
|
2014-09-09 17:04:47 -04:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <sys/signal.h>
|
2014-08-26 17:10:19 -04:00
|
|
|
#include <unistd.h>
|
|
|
|
|
2016-01-06 12:22:50 -05:00
|
|
|
#include "base/macros.h"
|
2014-08-26 17:10:19 -04:00
|
|
|
#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/gtest_death_check.h"
|
2014-12-17 14:35:18 -08:00
|
|
|
#include "util/file/file_io.h"
|
2014-08-26 17:10:19 -04:00
|
|
|
|
2014-10-07 17:28:50 -04:00
|
|
|
namespace crashpad {
|
|
|
|
namespace test {
|
2014-08-26 17:10:19 -04:00
|
|
|
namespace {
|
|
|
|
|
|
|
|
class TestMultiprocess final : public Multiprocess {
|
|
|
|
public:
|
|
|
|
TestMultiprocess() : Multiprocess() {}
|
|
|
|
|
|
|
|
~TestMultiprocess() {}
|
|
|
|
|
|
|
|
private:
|
2014-09-22 13:06:12 -04:00
|
|
|
// Multiprocess:
|
|
|
|
|
2014-10-14 11:11:57 -04:00
|
|
|
void MultiprocessParent() override {
|
2015-01-28 14:49:42 -08:00
|
|
|
FileHandle read_handle = ReadPipeHandle();
|
2014-08-26 17:10:19 -04:00
|
|
|
char c;
|
2015-01-28 14:49:42 -08:00
|
|
|
CheckedReadFile(read_handle, &c, 1);
|
2014-08-26 17:10:19 -04:00
|
|
|
EXPECT_EQ('M', c);
|
|
|
|
|
|
|
|
pid_t pid;
|
2015-01-28 14:49:42 -08:00
|
|
|
CheckedReadFile(read_handle, &pid, sizeof(pid));
|
2014-08-26 17:10:19 -04:00
|
|
|
EXPECT_EQ(pid, ChildPID());
|
|
|
|
|
|
|
|
c = 'm';
|
2015-01-28 14:49:42 -08:00
|
|
|
CheckedWriteFile(WritePipeHandle(), &c, 1);
|
2014-08-26 17:10:19 -04:00
|
|
|
|
|
|
|
// The child will close its end of the pipe and exit. Make sure that the
|
|
|
|
// parent sees EOF.
|
2015-01-28 14:49:42 -08:00
|
|
|
CheckedReadFileAtEOF(read_handle);
|
2014-08-26 17:10:19 -04:00
|
|
|
}
|
|
|
|
|
2014-10-14 11:11:57 -04:00
|
|
|
void MultiprocessChild() override {
|
2015-01-28 14:49:42 -08:00
|
|
|
FileHandle write_handle = WritePipeHandle();
|
2014-08-26 17:10:19 -04:00
|
|
|
|
|
|
|
char c = 'M';
|
2015-01-28 14:49:42 -08:00
|
|
|
CheckedWriteFile(write_handle, &c, 1);
|
2014-08-26 17:10:19 -04:00
|
|
|
|
|
|
|
pid_t pid = getpid();
|
2015-01-28 14:49:42 -08:00
|
|
|
CheckedWriteFile(write_handle, &pid, sizeof(pid));
|
2014-08-26 17:10:19 -04:00
|
|
|
|
2015-01-28 14:49:42 -08:00
|
|
|
CheckedReadFile(ReadPipeHandle(), &c, 1);
|
2014-08-26 17:10:19 -04:00
|
|
|
EXPECT_EQ('m', c);
|
|
|
|
}
|
|
|
|
|
|
|
|
DISALLOW_COPY_AND_ASSIGN(TestMultiprocess);
|
|
|
|
};
|
|
|
|
|
|
|
|
TEST(Multiprocess, Multiprocess) {
|
|
|
|
TestMultiprocess multiprocess;
|
|
|
|
multiprocess.Run();
|
|
|
|
}
|
|
|
|
|
2014-09-09 17:04:47 -04:00
|
|
|
class TestMultiprocessUnclean final : public Multiprocess {
|
|
|
|
public:
|
|
|
|
enum TerminationType {
|
|
|
|
kExitSuccess = 0,
|
|
|
|
kExitFailure,
|
|
|
|
kExit2,
|
|
|
|
kAbort,
|
|
|
|
};
|
|
|
|
|
|
|
|
explicit TestMultiprocessUnclean(TerminationType type)
|
|
|
|
: Multiprocess(),
|
|
|
|
type_(type) {
|
|
|
|
if (type_ == kAbort) {
|
|
|
|
SetExpectedChildTermination(kTerminationSignal, SIGABRT);
|
|
|
|
} else {
|
|
|
|
SetExpectedChildTermination(kTerminationNormal, ExitCode());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
~TestMultiprocessUnclean() {}
|
|
|
|
|
|
|
|
private:
|
|
|
|
int ExitCode() const {
|
|
|
|
return type_;
|
|
|
|
}
|
|
|
|
|
2014-09-22 13:06:12 -04:00
|
|
|
// Multiprocess:
|
|
|
|
|
2014-10-14 11:11:57 -04:00
|
|
|
void MultiprocessParent() override {
|
2014-09-09 17:04:47 -04:00
|
|
|
}
|
|
|
|
|
2014-10-14 11:11:57 -04:00
|
|
|
void MultiprocessChild() override {
|
2014-09-09 17:04:47 -04:00
|
|
|
if (type_ == kAbort) {
|
|
|
|
abort();
|
|
|
|
} else {
|
|
|
|
_exit(ExitCode());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TerminationType type_;
|
|
|
|
|
|
|
|
DISALLOW_COPY_AND_ASSIGN(TestMultiprocessUnclean);
|
|
|
|
};
|
|
|
|
|
2014-09-22 13:06:12 -04:00
|
|
|
TEST(Multiprocess, SuccessfulExit) {
|
2014-09-09 17:04:47 -04:00
|
|
|
TestMultiprocessUnclean multiprocess(TestMultiprocessUnclean::kExitSuccess);
|
|
|
|
multiprocess.Run();
|
|
|
|
}
|
|
|
|
|
2014-09-22 13:06:12 -04:00
|
|
|
TEST(Multiprocess, UnsuccessfulExit) {
|
2014-09-09 17:04:47 -04:00
|
|
|
TestMultiprocessUnclean multiprocess(TestMultiprocessUnclean::kExitFailure);
|
|
|
|
multiprocess.Run();
|
|
|
|
}
|
|
|
|
|
2014-09-22 13:06:12 -04:00
|
|
|
TEST(Multiprocess, Exit2) {
|
2014-09-09 17:04:47 -04:00
|
|
|
TestMultiprocessUnclean multiprocess(TestMultiprocessUnclean::kExit2);
|
|
|
|
multiprocess.Run();
|
|
|
|
}
|
|
|
|
|
2014-09-22 13:06:12 -04:00
|
|
|
TEST(Multiprocess, AbortSignal) {
|
2014-09-09 17:04:47 -04:00
|
|
|
TestMultiprocessUnclean multiprocess(TestMultiprocessUnclean::kAbort);
|
|
|
|
multiprocess.Run();
|
|
|
|
}
|
|
|
|
|
2014-09-22 13:06:12 -04:00
|
|
|
class TestMultiprocessClosePipe final : public Multiprocess {
|
|
|
|
public:
|
|
|
|
enum WhoCloses {
|
|
|
|
kParentCloses = 0,
|
|
|
|
kChildCloses,
|
|
|
|
};
|
|
|
|
enum WhatCloses {
|
|
|
|
kReadCloses = 0,
|
|
|
|
kWriteCloses,
|
|
|
|
kReadAndWriteClose,
|
|
|
|
};
|
|
|
|
|
|
|
|
TestMultiprocessClosePipe(WhoCloses who_closes, WhatCloses what_closes)
|
|
|
|
: Multiprocess(),
|
|
|
|
who_closes_(who_closes),
|
|
|
|
what_closes_(what_closes) {
|
|
|
|
}
|
|
|
|
|
|
|
|
~TestMultiprocessClosePipe() {}
|
|
|
|
|
|
|
|
private:
|
|
|
|
void VerifyInitial() {
|
2015-01-28 14:49:42 -08:00
|
|
|
ASSERT_NE(-1, ReadPipeHandle());
|
|
|
|
ASSERT_NE(-1, WritePipeHandle());
|
2014-09-22 13:06:12 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// Verifies that the partner process did what it was supposed to do. This must
|
|
|
|
// only be called when who_closes_ names the partner process, not this
|
|
|
|
// process.
|
|
|
|
//
|
|
|
|
// If the partner was supposed to close its write pipe, the read pipe will be
|
|
|
|
// checked to ensure that it shows end-of-file.
|
|
|
|
//
|
|
|
|
// If the partner was supposed to close its read pipe, the write pipe will be
|
|
|
|
// checked to ensure that a checked write causes death. This can only be done
|
|
|
|
// if the partner also provides some type of signal when it has closed its
|
|
|
|
// read pipe, which is done in the form of it closing its write pipe, causing
|
|
|
|
// the read pipe in this process to show end-of-file.
|
|
|
|
void VerifyPartner() {
|
|
|
|
if (what_closes_ == kWriteCloses) {
|
2015-01-28 14:49:42 -08:00
|
|
|
CheckedReadFileAtEOF(ReadPipeHandle());
|
2014-09-22 13:06:12 -04:00
|
|
|
} else if (what_closes_ == kReadAndWriteClose) {
|
2015-01-28 14:49:42 -08:00
|
|
|
CheckedReadFileAtEOF(ReadPipeHandle());
|
2014-09-22 13:06:12 -04:00
|
|
|
char c = '\0';
|
|
|
|
|
|
|
|
// This will raise SIGPIPE. If fatal (the normal case), that will cause
|
|
|
|
// process termination. If SIGPIPE is being handled somewhere, the write
|
2014-12-17 14:35:18 -08:00
|
|
|
// will still fail and set errno to EPIPE, and CheckedWriteFile() will
|
|
|
|
// abort execution. Regardless of how SIGPIPE is handled, the process will
|
|
|
|
// be terminated. Because the actual termination mechanism is not known,
|
|
|
|
// no regex can be specified.
|
2015-03-09 18:02:14 -04:00
|
|
|
EXPECT_DEATH_CHECK(CheckedWriteFile(WritePipeHandle(), &c, 1), "");
|
2014-09-22 13:06:12 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Close() {
|
|
|
|
switch (what_closes_) {
|
|
|
|
case kReadCloses:
|
|
|
|
CloseReadPipe();
|
2015-01-28 14:49:42 -08:00
|
|
|
EXPECT_NE(-1, WritePipeHandle());
|
2015-03-09 18:02:14 -04:00
|
|
|
EXPECT_DEATH_CHECK(ReadPipeHandle(), "fd");
|
2014-09-22 13:06:12 -04:00
|
|
|
break;
|
|
|
|
case kWriteCloses:
|
|
|
|
CloseWritePipe();
|
2015-01-28 14:49:42 -08:00
|
|
|
EXPECT_NE(-1, ReadPipeHandle());
|
2015-03-09 18:02:14 -04:00
|
|
|
EXPECT_DEATH_CHECK(WritePipeHandle(), "fd");
|
2014-09-22 13:06:12 -04:00
|
|
|
break;
|
|
|
|
case kReadAndWriteClose:
|
|
|
|
CloseReadPipe();
|
|
|
|
CloseWritePipe();
|
2015-03-09 18:02:14 -04:00
|
|
|
EXPECT_DEATH_CHECK(ReadPipeHandle(), "fd");
|
|
|
|
EXPECT_DEATH_CHECK(WritePipeHandle(), "fd");
|
2014-09-22 13:06:12 -04:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Multiprocess:
|
|
|
|
|
2014-10-14 11:11:57 -04:00
|
|
|
void MultiprocessParent() override {
|
2014-10-09 15:08:54 -04:00
|
|
|
ASSERT_NO_FATAL_FAILURE(VerifyInitial());
|
2014-09-22 13:06:12 -04:00
|
|
|
|
|
|
|
if (who_closes_ == kParentCloses) {
|
|
|
|
Close();
|
|
|
|
} else {
|
|
|
|
VerifyPartner();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-10-14 11:11:57 -04:00
|
|
|
void MultiprocessChild() override {
|
2014-10-09 15:08:54 -04:00
|
|
|
ASSERT_NO_FATAL_FAILURE(VerifyInitial());
|
2014-09-22 13:06:12 -04:00
|
|
|
|
|
|
|
if (who_closes_ == kChildCloses) {
|
|
|
|
Close();
|
|
|
|
} else {
|
|
|
|
VerifyPartner();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
WhoCloses who_closes_;
|
|
|
|
WhatCloses what_closes_;
|
|
|
|
|
|
|
|
DISALLOW_COPY_AND_ASSIGN(TestMultiprocessClosePipe);
|
|
|
|
};
|
|
|
|
|
|
|
|
TEST(MultiprocessDeathTest, ParentClosesReadPipe) {
|
|
|
|
TestMultiprocessClosePipe multiprocess(
|
|
|
|
TestMultiprocessClosePipe::kParentCloses,
|
|
|
|
TestMultiprocessClosePipe::kReadCloses);
|
|
|
|
multiprocess.Run();
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(MultiprocessDeathTest, ParentClosesWritePipe) {
|
|
|
|
TestMultiprocessClosePipe multiprocess(
|
|
|
|
TestMultiprocessClosePipe::kParentCloses,
|
|
|
|
TestMultiprocessClosePipe::kWriteCloses);
|
|
|
|
multiprocess.Run();
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(MultiprocessDeathTest, ParentClosesReadAndWritePipe) {
|
|
|
|
TestMultiprocessClosePipe multiprocess(
|
|
|
|
TestMultiprocessClosePipe::kParentCloses,
|
|
|
|
TestMultiprocessClosePipe::kReadAndWriteClose);
|
|
|
|
multiprocess.Run();
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(MultiprocessDeathTest, ChildClosesReadPipe) {
|
|
|
|
TestMultiprocessClosePipe multiprocess(
|
|
|
|
TestMultiprocessClosePipe::kChildCloses,
|
|
|
|
TestMultiprocessClosePipe::kReadCloses);
|
|
|
|
multiprocess.Run();
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(MultiprocessDeathTest, ChildClosesWritePipe) {
|
|
|
|
TestMultiprocessClosePipe multiprocess(
|
|
|
|
TestMultiprocessClosePipe::kChildCloses,
|
|
|
|
TestMultiprocessClosePipe::kWriteCloses);
|
|
|
|
multiprocess.Run();
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(MultiprocessDeathTest, ChildClosesReadAndWritePipe) {
|
|
|
|
TestMultiprocessClosePipe multiprocess(
|
|
|
|
TestMultiprocessClosePipe::kChildCloses,
|
|
|
|
TestMultiprocessClosePipe::kReadAndWriteClose);
|
|
|
|
multiprocess.Run();
|
|
|
|
}
|
|
|
|
|
2014-08-26 17:10:19 -04:00
|
|
|
} // namespace
|
2014-10-07 17:28:50 -04:00
|
|
|
} // namespace test
|
|
|
|
} // namespace crashpad
|