Index: test/win/child_launcher.cc |
diff --git a/test/win/child_launcher.cc b/test/win/child_launcher.cc |
new file mode 100644 |
index 0000000000000000000000000000000000000000..cc95255c58d59c9393e7be72412e7794717b4fae |
--- /dev/null |
+++ b/test/win/child_launcher.cc |
@@ -0,0 +1,114 @@ |
+// Copyright 2015 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. |
+ |
+#include "test/win/child_launcher.h" |
+ |
+#include "gtest/gtest.h" |
+ |
+namespace crashpad { |
+namespace test { |
+ |
+ChildLauncher::ChildLauncher(const std::wstring& executable, |
+ const std::wstring& command_line) |
+ : executable_(executable), |
+ command_line_(command_line), |
+ process_handle_(), |
+ main_thread_handle_(), |
+ stdout_read_handle_() { |
+} |
+ |
+ChildLauncher::~ChildLauncher() { |
+ EXPECT_EQ(WAIT_OBJECT_0, |
+ WaitForSingleObject(process_handle_.get(), INFINITE)); |
+} |
+ |
+void ChildLauncher::Start() { |
+ ASSERT_FALSE(process_handle_.is_valid()); |
+ ASSERT_FALSE(main_thread_handle_.is_valid()); |
+ ASSERT_FALSE(stdout_read_handle_.is_valid()); |
+ |
+ // Create a pipe for the 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)); |
+ stdout_read_handle_.reset(stdout_read); |
+ ScopedFileHANDLE write_handle(stdout_write); |
+ ASSERT_TRUE( |
+ SetHandleInformation(stdout_read_handle_.get(), HANDLE_FLAG_INHERIT, 0)); |
+ |
+ STARTUPINFO startup_info = {0}; |
+ startup_info.cb = sizeof(startup_info); |
+ startup_info.hStdInput = GetStdHandle(STD_INPUT_HANDLE); |
+ startup_info.hStdOutput = write_handle.get(); |
+ startup_info.hStdError = GetStdHandle(STD_ERROR_HANDLE); |
+ startup_info.dwFlags = STARTF_USESTDHANDLES; |
+ PROCESS_INFORMATION process_information; |
+ std::wstring command_line; |
+ AppendCommandLineArgument(executable_, &command_line); |
+ command_line += L" "; |
+ command_line += command_line_; |
+ ASSERT_TRUE(CreateProcess(executable_.c_str(), |
+ &command_line[0], |
+ nullptr, |
+ nullptr, |
+ true, |
+ 0, |
+ nullptr, |
+ nullptr, |
+ &startup_info, |
+ &process_information)); |
+ // Take ownership of the two process handles returned. |
+ main_thread_handle_.reset(process_information.hThread); |
+ process_handle_.reset(process_information.hProcess); |
+} |
+ |
+// Ref: http://blogs.msdn.com/b/twistylittlepassagesallalike/archive/2011/04/23/everyone-quotes-arguments-the-wrong-way.aspx |
+void AppendCommandLineArgument(const std::wstring& argument, |
+ std::wstring* command_line) { |
+ // Don't bother quoting if unnecessary. |
+ if (!argument.empty() && |
+ argument.find_first_of(L" \t\n\v\"") == std::wstring::npos) { |
+ command_line->append(argument); |
+ } else { |
+ command_line->push_back(L'"'); |
+ for (std::wstring::const_iterator i = argument.begin();; ++i) { |
+ size_t backslash_count = 0; |
+ while (i != argument.end() && *i == L'\\') { |
+ ++i; |
+ ++backslash_count; |
+ } |
+ if (i == argument.end()) { |
+ // Escape all backslashes, but let the terminating double quotation mark |
+ // we add below be interpreted as a metacharacter. |
+ command_line->append(backslash_count * 2, L'\\'); |
+ break; |
+ } else if (*i == L'"') { |
+ // Escape all backslashes and the following double quotation mark. |
+ command_line->append(backslash_count * 2 + 1, L'\\'); |
+ command_line->push_back(*i); |
+ } else { |
+ // Backslashes aren't special here. |
+ command_line->append(backslash_count, L'\\'); |
+ command_line->push_back(*i); |
+ } |
+ } |
+ command_line->push_back(L'"'); |
+ } |
+} |
+ |
+} // namespace test |
+} // namespace crashpad |