crashpad/test/multiprocess_exec_test_child.cc

94 lines
2.6 KiB
C++
Raw Normal View History

// 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.
#include <errno.h>
#include <limits.h>
#include <stdio.h>
#include <sys/types.h>
#include <algorithm>
#include "base/logging.h"
#include "build/build_config.h"
#if defined(OS_POSIX)
#include <sys/resource.h>
#include <unistd.h>
#elif defined(OS_WIN)
#include <windows.h>
#endif
int main(int argc, char* argv[]) {
#if defined(OS_POSIX)
#if defined(OS_FUCHSIA)
// getrlimit() is not implemented on Fuchsia. By construction, the child only
// receieves specific fds that it's given, but check low values as mild
// verification.
int last_fd = 1024;
#else
rlimit rlimit_nofile;
if (getrlimit(RLIMIT_NOFILE, &rlimit_nofile) != 0) {
LOG(FATAL) << "getrlimit";
}
int last_fd = static_cast<int>(rlimit_nofile.rlim_cur);
#endif // OS_FUCHSIA
// Make sure that theres nothing open at any FD higher than 3. All FDs other
// than stdin, stdout, and stderr should have been closed prior to or at
// exec().
for (int fd = STDERR_FILENO + 1; fd < last_fd; ++fd) {
if (close(fd) == 0 || errno != EBADF) {
LOG(FATAL) << "close";
}
}
// Read a byte from stdin, expecting it to be a specific value.
char c;
ssize_t rv = read(STDIN_FILENO, &c, 1);
if (rv != 1 || c != 'z') {
LOG(FATAL) << "read";
}
// Write a byte to stdout.
c = 'Z';
rv = write(STDOUT_FILENO, &c, 1);
if (rv != 1) {
LOG(FATAL) << "write";
}
#elif defined(OS_WIN)
// TODO(scottmg): Verify that only the handles we expect to be open, are.
// Read a byte from stdin, expecting it to be a specific value.
char c;
DWORD bytes_read;
HANDLE stdin_handle = GetStdHandle(STD_INPUT_HANDLE);
if (!ReadFile(stdin_handle, &c, 1, &bytes_read, nullptr) ||
bytes_read != 1 || c != 'z') {
LOG(FATAL) << "ReadFile";
}
// Write a byte to stdout.
c = 'Z';
DWORD bytes_written;
if (!WriteFile(
GetStdHandle(STD_OUTPUT_HANDLE), &c, 1, &bytes_written, nullptr) ||
bytes_written != 1) {
LOG(FATAL) << "WriteFile";
}
#endif // OS_POSIX
return 0;
}