chromium/third_party/crashpad/crashpad/test/win/child_launcher.cc

// Copyright 2015 The Crashpad Authors
//
// 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.

#include "test/win/child_launcher.h"

#include "gtest/gtest.h"
#include "test/errors.h"
#include "util/win/command_line.h"

namespace crashpad {
namespace test {

ChildLauncher::ChildLauncher(const base::FilePath& executable,
                             const std::wstring& command_line)
    : executable_(executable),
      command_line_(command_line),
      process_handle_(),
      main_thread_handle_(),
      stdout_read_handle_(),
      stdin_write_handle_() {}

ChildLauncher::~ChildLauncher() {
  if (process_handle_.is_valid())
    WaitForExit();
}

void ChildLauncher::Start() {
  ASSERT_FALSE(process_handle_.is_valid());
  ASSERT_FALSE(main_thread_handle_.is_valid());
  ASSERT_FALSE(stdout_read_handle_.is_valid());

  // Create pipes for the stdin/stdout of the child.
  SECURITY_ATTRIBUTES security_attributes = {0};
  security_attributes.nLength = sizeof(SECURITY_ATTRIBUTES);
  security_attributes.bInheritHandle = true;

  HANDLE stdout_read;
  HANDLE stdout_write;
  ASSERT_TRUE(CreatePipe(&stdout_read, &stdout_write, &security_attributes, 0))
      << ErrorMessage("CreatePipe");
  stdout_read_handle_.reset(stdout_read);
  ScopedFileHANDLE write_handle(stdout_write);
  ASSERT_TRUE(
      SetHandleInformation(stdout_read_handle_.get(), HANDLE_FLAG_INHERIT, 0))
      << ErrorMessage("SetHandleInformation");

  HANDLE stdin_read;
  HANDLE stdin_write;
  ASSERT_TRUE(CreatePipe(&stdin_read, &stdin_write, &security_attributes, 0))
      << ErrorMessage("CreatePipe");
  stdin_write_handle_.reset(stdin_write);
  ScopedFileHANDLE read_handle(stdin_read);
  ASSERT_TRUE(
      SetHandleInformation(stdin_write_handle_.get(), HANDLE_FLAG_INHERIT, 0))
      << ErrorMessage("SetHandleInformation");

  STARTUPINFO startup_info = {0};
  startup_info.cb = sizeof(startup_info);
  startup_info.hStdInput = read_handle.get();
  startup_info.hStdOutput = write_handle.get();
  startup_info.hStdError = GetStdHandle(STD_ERROR_HANDLE);
  EXPECT_NE(startup_info.hStdError, INVALID_HANDLE_VALUE)
      << ErrorMessage("GetStdHandle");
  startup_info.dwFlags = STARTF_USESTDHANDLES;
  PROCESS_INFORMATION process_information;
  std::wstring command_line;
  AppendCommandLineArgument(executable_.value(), &command_line);
  command_line += L" ";
  command_line += command_line_;
  ASSERT_TRUE(CreateProcess(executable_.value().c_str(),
                            &command_line[0],
                            nullptr,
                            nullptr,
                            true,
                            0,
                            nullptr,
                            nullptr,
                            &startup_info,
                            &process_information))
      << ErrorMessage("CreateProcess");
  // Take ownership of the two process handles returned.
  main_thread_handle_.reset(process_information.hThread);
  process_handle_.reset(process_information.hProcess);
}

DWORD ChildLauncher::WaitForExit() {
  EXPECT_TRUE(process_handle_.is_valid());
  EXPECT_EQ(WaitForSingleObject(process_handle_.get(), INFINITE), WAIT_OBJECT_0)
      << ErrorMessage("WaitForSingleObject");
  DWORD exit_code = 0;
  EXPECT_TRUE(GetExitCodeProcess(process_handle_.get(), &exit_code))
      << ErrorMessage("GetExitCodeProcess");
  process_handle_.reset();
  return exit_code;
}

}  // namespace test
}  // namespace crashpad