summaryrefslogtreecommitdiffstats
path: root/content/zygote/zygote_linux.cc
diff options
context:
space:
mode:
Diffstat (limited to 'content/zygote/zygote_linux.cc')
-rw-r--r--content/zygote/zygote_linux.cc488
1 files changed, 488 insertions, 0 deletions
diff --git a/content/zygote/zygote_linux.cc b/content/zygote/zygote_linux.cc
new file mode 100644
index 0000000..9c6a329
--- /dev/null
+++ b/content/zygote/zygote_linux.cc
@@ -0,0 +1,488 @@
+// Copyright (c) 2012 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#include "content/zygote/zygote_linux.h"
+
+#include <string.h>
+#include <sys/socket.h>
+#include <sys/types.h>
+#include <sys/wait.h>
+
+#include "base/command_line.h"
+#include "ipc/ipc_switches.h"
+#include "content/public/common/sandbox_linux.h"
+#include "base/process_util.h"
+#include "content/public/common/result_codes.h"
+#include "ipc/ipc_channel.h"
+#include "base/debug/trace_event.h"
+#include "base/file_util.h"
+#include "base/linux_util.h"
+#include "base/eintr_wrapper.h"
+#include "base/global_descriptors_posix.h"
+#include "base/logging.h"
+#include "base/pickle.h"
+#include "content/common/chrome_descriptors.h"
+#include "content/common/set_process_title.h"
+#include "content/common/unix_domain_socket_posix.h"
+#include "content/public/common/zygote_fork_delegate_linux.h"
+#include "content/common/sandbox_methods_linux.h"
+#include "content/common/zygote_commands_linux.h"
+
+// See http://code.google.com/p/chromium/wiki/LinuxZygote
+
+namespace content {
+
+namespace {
+
+const int kZygoteIdDescriptor = 7;
+
+// NOP function. See below where this handler is installed.
+void SIGCHLDHandler(int signal) {
+}
+
+} // namespace
+
+Zygote::Zygote(int sandbox_flags,
+ ZygoteForkDelegate* helper,
+ int proc_fd_for_seccomp)
+ : sandbox_flags_(sandbox_flags),
+ helper_(helper),
+ proc_fd_for_seccomp_(proc_fd_for_seccomp),
+ initial_uma_sample_(0),
+ initial_uma_boundary_value_(0) {
+ if (helper_) {
+ helper_->InitialUMA(&initial_uma_name_,
+ &initial_uma_sample_,
+ &initial_uma_boundary_value_);
+ }
+}
+
+Zygote::~Zygote() {
+}
+
+bool Zygote::ProcessRequests() {
+ // A SOCK_SEQPACKET socket is installed in fd 3. We get commands from the
+ // browser on it.
+ // A SOCK_DGRAM is installed in fd 5. This is the sandbox IPC channel.
+ // See http://code.google.com/p/chromium/wiki/LinuxSandboxIPC
+
+ // We need to accept SIGCHLD, even though our handler is a no-op because
+ // otherwise we cannot wait on children. (According to POSIX 2001.)
+ struct sigaction action;
+ memset(&action, 0, sizeof(action));
+ action.sa_handler = &SIGCHLDHandler;
+ CHECK(sigaction(SIGCHLD, &action, NULL) == 0);
+
+ if (UsingSUIDSandbox()) {
+ // Let the ZygoteHost know we are ready to go.
+ // The receiving code is in content/browser/zygote_host_linux.cc.
+ std::vector<int> empty;
+ bool r = UnixDomainSocket::SendMsg(kBrowserDescriptor,
+ kZygoteHelloMessage,
+ sizeof(kZygoteHelloMessage), empty);
+#if defined(OS_CHROMEOS)
+ LOG_IF(WARNING, !r) << "Sending zygote magic failed";
+ // Exit normally on chromeos because session manager may send SIGTERM
+ // right after the process starts and it may fail to send zygote magic
+ // number to browser process.
+ if (!r)
+ _exit(RESULT_CODE_NORMAL_EXIT);
+#else
+ CHECK(r) << "Sending zygote magic failed";
+#endif
+ }
+
+ for (;;) {
+ // This function call can return multiple times, once per fork().
+ if (HandleRequestFromBrowser(kBrowserDescriptor))
+ return true;
+ }
+}
+
+bool Zygote::UsingSUIDSandbox() const {
+ return sandbox_flags_ & kSandboxLinuxSUID;
+}
+
+bool Zygote::HandleRequestFromBrowser(int fd) {
+ std::vector<int> fds;
+ static const unsigned kMaxMessageLength = 2048;
+ char buf[kMaxMessageLength];
+ const ssize_t len = UnixDomainSocket::RecvMsg(fd, buf, sizeof(buf), &fds);
+
+ if (len == 0 || (len == -1 && errno == ECONNRESET)) {
+ // EOF from the browser. We should die.
+ _exit(0);
+ return false;
+ }
+
+ if (len == -1) {
+ PLOG(ERROR) << "Error reading message from browser";
+ return false;
+ }
+
+ Pickle pickle(buf, len);
+ PickleIterator iter(pickle);
+
+ int kind;
+ if (pickle.ReadInt(&iter, &kind)) {
+ switch (kind) {
+ case kZygoteCommandFork:
+ // This function call can return multiple times, once per fork().
+ return HandleForkRequest(fd, pickle, iter, fds);
+
+ case kZygoteCommandReap:
+ if (!fds.empty())
+ break;
+ HandleReapRequest(fd, pickle, iter);
+ return false;
+ case kZygoteCommandGetTerminationStatus:
+ if (!fds.empty())
+ break;
+ HandleGetTerminationStatus(fd, pickle, iter);
+ return false;
+ case kZygoteCommandGetSandboxStatus:
+ HandleGetSandboxStatus(fd, pickle, iter);
+ return false;
+ default:
+ NOTREACHED();
+ break;
+ }
+ }
+
+ LOG(WARNING) << "Error parsing message from browser";
+ for (std::vector<int>::const_iterator
+ i = fds.begin(); i != fds.end(); ++i)
+ close(*i);
+ return false;
+}
+
+void Zygote::HandleReapRequest(int fd,
+ const Pickle& pickle,
+ PickleIterator iter) {
+ base::ProcessId child;
+ base::ProcessId actual_child;
+
+ if (!pickle.ReadInt(&iter, &child)) {
+ LOG(WARNING) << "Error parsing reap request from browser";
+ return;
+ }
+
+ if (UsingSUIDSandbox()) {
+ actual_child = real_pids_to_sandbox_pids[child];
+ if (!actual_child)
+ return;
+ real_pids_to_sandbox_pids.erase(child);
+ } else {
+ actual_child = child;
+ }
+
+ base::EnsureProcessTerminated(actual_child);
+}
+
+void Zygote::HandleGetTerminationStatus(int fd,
+ const Pickle& pickle,
+ PickleIterator iter) {
+ base::ProcessHandle child;
+
+ if (!pickle.ReadInt(&iter, &child)) {
+ LOG(WARNING) << "Error parsing GetTerminationStatus request "
+ << "from browser";
+ return;
+ }
+
+ base::TerminationStatus status;
+ int exit_code;
+ if (UsingSUIDSandbox())
+ child = real_pids_to_sandbox_pids[child];
+ if (child) {
+ status = base::GetTerminationStatus(child, &exit_code);
+ } else {
+ // Assume that if we can't find the child in the sandbox, then
+ // it terminated normally.
+ status = base::TERMINATION_STATUS_NORMAL_TERMINATION;
+ exit_code = RESULT_CODE_NORMAL_EXIT;
+ }
+
+ Pickle write_pickle;
+ write_pickle.WriteInt(static_cast<int>(status));
+ write_pickle.WriteInt(exit_code);
+ ssize_t written =
+ HANDLE_EINTR(write(fd, write_pickle.data(), write_pickle.size()));
+ if (written != static_cast<ssize_t>(write_pickle.size()))
+ PLOG(ERROR) << "write";
+}
+
+int Zygote::ForkWithRealPid(const std::string& process_type,
+ std::vector<int>& fds,
+ const std::string& channel_switch,
+ std::string* uma_name,
+ int* uma_sample,
+ int* uma_boundary_value) {
+ const bool use_helper = (helper_ && helper_->CanHelp(process_type,
+ uma_name,
+ uma_sample,
+ uma_boundary_value));
+ if (!(use_helper || UsingSUIDSandbox())) {
+ return fork();
+ }
+
+ int dummy_fd;
+ ino_t dummy_inode;
+ int pipe_fds[2] = { -1, -1 };
+ base::ProcessId pid = 0;
+
+ dummy_fd = socket(PF_UNIX, SOCK_DGRAM, 0);
+ if (dummy_fd < 0) {
+ LOG(ERROR) << "Failed to create dummy FD";
+ goto error;
+ }
+ if (!base::FileDescriptorGetInode(&dummy_inode, dummy_fd)) {
+ LOG(ERROR) << "Failed to get inode for dummy FD";
+ goto error;
+ }
+ if (pipe(pipe_fds) != 0) {
+ LOG(ERROR) << "Failed to create pipe";
+ goto error;
+ }
+
+ if (use_helper) {
+ fds.push_back(dummy_fd);
+ fds.push_back(pipe_fds[0]);
+ pid = helper_->Fork(fds);
+ } else {
+ pid = fork();
+ }
+ if (pid < 0) {
+ goto error;
+ } else if (pid == 0) {
+ // In the child process.
+ close(pipe_fds[1]);
+ base::ProcessId real_pid;
+ // Wait until the parent process has discovered our PID. We
+ // should not fork any child processes (which the seccomp
+ // sandbox does) until then, because that can interfere with the
+ // parent's discovery of our PID.
+ if (!file_util::ReadFromFD(pipe_fds[0],
+ reinterpret_cast<char*>(&real_pid),
+ sizeof(real_pid))) {
+ LOG(FATAL) << "Failed to synchronise with parent zygote process";
+ }
+ if (real_pid <= 0) {
+ LOG(FATAL) << "Invalid pid from parent zygote";
+ }
+#if defined(OS_LINUX)
+ // Sandboxed processes need to send the global, non-namespaced PID when
+ // setting up an IPC channel to their parent.
+ IPC::Channel::SetGlobalPid(real_pid);
+ // Force the real PID so chrome event data have a PID that corresponds
+ // to system trace event data.
+ base::debug::TraceLog::GetInstance()->SetProcessID(
+ static_cast<int>(real_pid));
+#endif
+ close(pipe_fds[0]);
+ close(dummy_fd);
+ return 0;
+ } else {
+ // In the parent process.
+ close(dummy_fd);
+ dummy_fd = -1;
+ close(pipe_fds[0]);
+ pipe_fds[0] = -1;
+ base::ProcessId real_pid;
+ if (UsingSUIDSandbox()) {
+ uint8_t reply_buf[512];
+ Pickle request;
+ request.WriteInt(LinuxSandbox::METHOD_GET_CHILD_WITH_INODE);
+ request.WriteUInt64(dummy_inode);
+
+ const ssize_t r = UnixDomainSocket::SendRecvMsg(
+ kMagicSandboxIPCDescriptor, reply_buf, sizeof(reply_buf), NULL,
+ request);
+ if (r == -1) {
+ LOG(ERROR) << "Failed to get child process's real PID";
+ goto error;
+ }
+
+ Pickle reply(reinterpret_cast<char*>(reply_buf), r);
+ PickleIterator iter(reply);
+ if (!reply.ReadInt(&iter, &real_pid))
+ goto error;
+ if (real_pid <= 0) {
+ // METHOD_GET_CHILD_WITH_INODE failed. Did the child die already?
+ LOG(ERROR) << "METHOD_GET_CHILD_WITH_INODE failed";
+ goto error;
+ }
+ real_pids_to_sandbox_pids[real_pid] = pid;
+ }
+ if (use_helper) {
+ real_pid = pid;
+ if (!helper_->AckChild(pipe_fds[1], channel_switch)) {
+ LOG(ERROR) << "Failed to synchronise with zygote fork helper";
+ goto error;
+ }
+ } else {
+ int written =
+ HANDLE_EINTR(write(pipe_fds[1], &real_pid, sizeof(real_pid)));
+ if (written != sizeof(real_pid)) {
+ LOG(ERROR) << "Failed to synchronise with child process";
+ goto error;
+ }
+ }
+ close(pipe_fds[1]);
+ return real_pid;
+ }
+
+ error:
+ if (pid > 0) {
+ if (waitpid(pid, NULL, WNOHANG) == -1)
+ LOG(ERROR) << "Failed to wait for process";
+ }
+ if (dummy_fd >= 0)
+ close(dummy_fd);
+ if (pipe_fds[0] >= 0)
+ close(pipe_fds[0]);
+ if (pipe_fds[1] >= 0)
+ close(pipe_fds[1]);
+ return -1;
+}
+
+base::ProcessId Zygote::ReadArgsAndFork(const Pickle& pickle,
+ PickleIterator iter,
+ std::vector<int>& fds,
+ std::string* uma_name,
+ int* uma_sample,
+ int* uma_boundary_value) {
+ std::vector<std::string> args;
+ int argc = 0;
+ int numfds = 0;
+ base::GlobalDescriptors::Mapping mapping;
+ std::string process_type;
+ std::string channel_id;
+ const std::string channel_id_prefix = std::string("--")
+ + switches::kProcessChannelID + std::string("=");
+
+ if (!pickle.ReadString(&iter, &process_type))
+ return -1;
+ if (!pickle.ReadInt(&iter, &argc))
+ return -1;
+
+ for (int i = 0; i < argc; ++i) {
+ std::string arg;
+ if (!pickle.ReadString(&iter, &arg))
+ return -1;
+ args.push_back(arg);
+ if (arg.compare(0, channel_id_prefix.length(), channel_id_prefix) == 0)
+ channel_id = arg;
+ }
+
+ if (!pickle.ReadInt(&iter, &numfds))
+ return -1;
+ if (numfds != static_cast<int>(fds.size()))
+ return -1;
+
+ for (int i = 0; i < numfds; ++i) {
+ base::GlobalDescriptors::Key key;
+ if (!pickle.ReadUInt32(&iter, &key))
+ return -1;
+ mapping.push_back(std::make_pair(key, fds[i]));
+ }
+
+ mapping.push_back(std::make_pair(
+ static_cast<uint32_t>(kSandboxIPCChannel), kMagicSandboxIPCDescriptor));
+
+ // Returns twice, once per process.
+ base::ProcessId child_pid = ForkWithRealPid(process_type, fds, channel_id,
+ uma_name, uma_sample,
+ uma_boundary_value);
+ if (!child_pid) {
+ // This is the child process.
+#if defined(SECCOMP_SANDBOX)
+ if (SeccompSandboxEnabled() && proc_fd_for_seccomp_ >= 0) {
+ // Try to open /proc/self/maps as the seccomp sandbox needs access to it
+ int proc_self_maps = openat(proc_fd_for_seccomp_, "self/maps", O_RDONLY);
+ if (proc_self_maps >= 0) {
+ SeccompSandboxSetProcSelfMaps(proc_self_maps);
+ } else {
+ PLOG(ERROR) << "openat(/proc/self/maps)";
+ }
+ close(proc_fd_for_seccomp_);
+ proc_fd_for_seccomp_ = -1;
+ }
+#endif
+
+ close(kBrowserDescriptor); // Our socket from the browser.
+ if (UsingSUIDSandbox())
+ close(kZygoteIdDescriptor); // Another socket from the browser.
+ base::GlobalDescriptors::GetInstance()->Reset(mapping);
+
+#if defined(CHROMIUM_SELINUX)
+ SELinuxTransitionToTypeOrDie("chromium_renderer_t");
+#endif
+
+ // Reset the process-wide command line to our new command line.
+ CommandLine::Reset();
+ CommandLine::Init(0, NULL);
+ CommandLine::ForCurrentProcess()->InitFromArgv(args);
+
+ // Update the process title. The argv was already cached by the call to
+ // SetProcessTitleFromCommandLine in ChromeMain, so we can pass NULL here
+ // (we don't have the original argv at this point).
+ SetProcessTitleFromCommandLine(NULL);
+ } else if (child_pid < 0) {
+ LOG(ERROR) << "Zygote could not fork: process_type " << process_type
+ << " numfds " << numfds << " child_pid " << child_pid;
+ }
+ return child_pid;
+}
+
+bool Zygote::HandleForkRequest(int fd,
+ const Pickle& pickle,
+ PickleIterator iter,
+ std::vector<int>& fds) {
+ std::string uma_name;
+ int uma_sample;
+ int uma_boundary_value;
+ base::ProcessId child_pid = ReadArgsAndFork(pickle, iter, fds,
+ &uma_name, &uma_sample,
+ &uma_boundary_value);
+ if (child_pid == 0)
+ return true;
+ for (std::vector<int>::const_iterator
+ i = fds.begin(); i != fds.end(); ++i)
+ close(*i);
+ if (uma_name.empty()) {
+ // There is no UMA report from this particular fork.
+ // Use the initial UMA report if any, and clear that record for next time.
+ // Note the swap method here is the efficient way to do this, since
+ // we know uma_name is empty.
+ uma_name.swap(initial_uma_name_);
+ uma_sample = initial_uma_sample_;
+ uma_boundary_value = initial_uma_boundary_value_;
+ }
+ // Must always send reply, as ZygoteHost blocks while waiting for it.
+ Pickle reply_pickle;
+ reply_pickle.WriteInt(child_pid);
+ reply_pickle.WriteString(uma_name);
+ if (!uma_name.empty()) {
+ reply_pickle.WriteInt(uma_sample);
+ reply_pickle.WriteInt(uma_boundary_value);
+ }
+ if (HANDLE_EINTR(write(fd, reply_pickle.data(), reply_pickle.size())) !=
+ static_cast<ssize_t> (reply_pickle.size()))
+ PLOG(ERROR) << "write";
+ return false;
+}
+
+bool Zygote::HandleGetSandboxStatus(int fd,
+ const Pickle& pickle,
+ PickleIterator iter) {
+ if (HANDLE_EINTR(write(fd, &sandbox_flags_, sizeof(sandbox_flags_))) !=
+ sizeof(sandbox_flags_)) {
+ PLOG(ERROR) << "write";
+ }
+
+ return false;
+}
+
+} // namespace content