// Copyright 2019 Google LLC // // 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. // Implementation file for the sandbox2::Client class. #include "sandboxed_api/sandbox2/client.h" #include <fcntl.h> #include <linux/filter.h> #include <linux/seccomp.h> #include <sys/prctl.h> #include <syscall.h> #include <unistd.h> #include <climits> #include <cstddef> #include <cstdint> #include <cstdlib> #include <cstring> #include <memory> #include <utility> #include "absl/base/internal/raw_logging.h" #include "absl/base/attributes.h" #include "absl/base/macros.h" #include "absl/memory/memory.h" #include "absl/strings/numbers.h" #include "absl/strings/str_cat.h" #include "absl/strings/str_join.h" #include "absl/strings/str_split.h" #include "sandboxed_api/sandbox2/comms.h" #include "sandboxed_api/sandbox2/sanitizer.h" #include "sandboxed_api/sandbox2/util/strerror.h" #include "sandboxed_api/util/raw_logging.h" namespace sandbox2 { constexpr uint32_t Client::kClient2SandboxReady; constexpr uint32_t Client::kSandbox2ClientDone; constexpr const char* Client::kFDMapEnvVar; Client::Client(Comms* comms) : comms_(comms) { char* fdmap_envvar = getenv(kFDMapEnvVar); if (!fdmap_envvar) { return; } std::map<absl::string_view, absl::string_view> vars = absl::StrSplit(fdmap_envvar, ',', absl::SkipEmpty()); for (const auto& var : vars) { int fd; SAPI_RAW_CHECK(absl::SimpleAtoi(var.second, &fd), "failed to parse fd map"); SAPI_RAW_CHECK(fd_map_.emplace(std::string{var.first}, fd).second, "could not insert mapping into fd map (duplicate)"); } unsetenv(kFDMapEnvVar); } std::string Client::GetFdMapEnvVar() const { return absl::StrCat(kFDMapEnvVar, "=", absl::StrJoin(fd_map_, ",", absl::PairFormatter(","))); } void Client::PrepareEnvironment() { SetUpIPC(); SetUpCwd(); } void Client::EnableSandbox() { ReceivePolicy(); ApplyPolicyAndBecomeTracee(); } void Client::SandboxMeHere() { PrepareEnvironment(); EnableSandbox(); } void Client::SetUpCwd() { { // Get the current working directory to check if we are in a mount // namespace. // Note: glibc 2.27 no longer returns a relative path in that case, but // fails with ENOENT and returns a nullptr instead. The code still // needs to run on lower version for the time being. char cwd_buf[PATH_MAX + 1] = {0}; char* cwd = getcwd(cwd_buf, ABSL_ARRAYSIZE(cwd_buf)); SAPI_RAW_PCHECK(cwd != nullptr || errno == ENOENT, "no current working directory"); // Outside of the mount namespace, the path is of the form // '(unreachable)/...'. Only check for the slash, since Linux might make up // other prefixes in the future. if (errno == ENOENT || cwd_buf[0] != '/') { SAPI_RAW_VLOG(1, "chdir into mount namespace, cwd was '%s'", cwd_buf); // If we are in a mount namespace but fail to chdir, then it can lead to a // sandbox escape -- we need to fail with FATAL if the chdir fails. SAPI_RAW_PCHECK(chdir("/") != -1, "corrective chdir"); } } // Receive the user-supplied current working directory and change into it. std::string cwd; SAPI_RAW_CHECK(comms_->RecvString(&cwd), "receiving working directory"); if (!cwd.empty()) { // On the other hand this chdir can fail without a sandbox escape. It will // probably not have the intended behavior though. if (chdir(cwd.c_str()) == -1) { SAPI_RAW_VLOG( 1, "chdir(%s) failed, falling back to previous cwd or / (with " "namespaces). Use Executor::SetCwd() to set a working directory: %s", cwd, StrError(errno)); } } } void Client::SetUpIPC() { uint32_t num_of_fd_pairs; SAPI_RAW_CHECK(comms_->RecvUint32(&num_of_fd_pairs), "receiving number of fd pairs"); SAPI_RAW_CHECK(fd_map_.empty(), "fd map not empty"); SAPI_RAW_VLOG(1, "Will receive %d file descriptor pairs", num_of_fd_pairs); for (uint32_t i = 0; i < num_of_fd_pairs; ++i) { int32_t requested_fd; int32_t fd; std::string name; SAPI_RAW_CHECK(comms_->RecvInt32(&requested_fd), "receiving requested fd"); SAPI_RAW_CHECK(comms_->RecvFD(&fd), "receiving current fd"); SAPI_RAW_CHECK(comms_->RecvString(&name), "receiving name string"); if (requested_fd != -1 && fd != requested_fd) { if (requested_fd > STDERR_FILENO && fcntl(requested_fd, F_GETFD) != -1) { // Dup2 will silently close the FD if one is already at requested_fd. // If someone is using the deferred sandbox entry, ie. SandboxMeHere, // the application might have something actually using that fd. // Therefore let's log a big warning if that FD is already in use. // Note: this check doesn't happen for STDIN,STDOUT,STDERR. SAPI_RAW_LOG( WARNING, "Cloning received fd %d over %d which is already open and will " "be silently closed. This may lead to unexpected behavior!", fd, requested_fd); } SAPI_RAW_VLOG(1, "Cloning received fd=%d onto fd=%d", fd, requested_fd); SAPI_RAW_PCHECK(dup2(fd, requested_fd) != -1, ""); // Close the newly received FD if it differs from the new one. close(fd); fd = requested_fd; } if (!name.empty()) { SAPI_RAW_CHECK(fd_map_.emplace(name, fd).second, "duplicate fd mapping"); } } } void Client::ReceivePolicy() { std::vector<uint8_t> bytes; SAPI_RAW_CHECK(comms_->RecvBytes(&bytes), "receive bytes"); policy_len_ = bytes.size(); policy_ = absl::make_unique<uint8_t[]>(policy_len_); memcpy(policy_.get(), bytes.data(), policy_len_); } void Client::ApplyPolicyAndBecomeTracee() { // When running under TSAN, we need to notify TSANs background thread that we // want it to exit and wait for it to be done. When not running under TSAN, // this function does nothing. sanitizer::WaitForTsan(); // Creds can be received w/o synchronization, once the connection is // established. pid_t cred_pid; uid_t cred_uid ABSL_ATTRIBUTE_UNUSED; gid_t cred_gid ABSL_ATTRIBUTE_UNUSED; SAPI_RAW_CHECK(comms_->RecvCreds(&cred_pid, &cred_uid, &cred_gid), "receiving credentials"); SAPI_RAW_CHECK(prctl(PR_SET_DUMPABLE, 1) == 0, "setting PR_SET_DUMPABLE flag"); if (prctl(PR_SET_PTRACER, cred_pid) == -1) { SAPI_RAW_VLOG(1, "No YAMA on this system. Continuing"); } SAPI_RAW_CHECK(prctl(PR_SET_NO_NEW_PRIVS, 1, 0, 0, 0) == 0, "setting PR_SET_NO_NEW_PRIVS flag"); SAPI_RAW_CHECK(prctl(PR_SET_KEEPCAPS, 0) == 0, "setting PR_SET_KEEPCAPS flag"); sock_fprog prog; prog.len = static_cast<uint16_t>(policy_len_ / sizeof(sock_filter)); prog.filter = reinterpret_cast<sock_filter*>(policy_.get()); SAPI_RAW_VLOG( 1, "Applying policy in PID %d, sock_fprog.len: %hd entries (%d bytes)", syscall(__NR_gettid), prog.len, policy_len_); // Signal executor we are ready to have limits applied on us and be ptraced. // We want limits at the last moment to avoid triggering them too early and we // want ptrace at the last moment to avoid synchronization deadlocks. SAPI_RAW_CHECK(comms_->SendUint32(kClient2SandboxReady), "receiving ready signal from executor"); uint32_t ret; // wait for confirmation SAPI_RAW_CHECK(comms_->RecvUint32(&ret), "receving confirmation from executor"); SAPI_RAW_CHECK(ret == kSandbox2ClientDone, "invalid confirmation from executor"); SAPI_RAW_CHECK( syscall(__NR_seccomp, SECCOMP_SET_MODE_FILTER, SECCOMP_FILTER_FLAG_TSYNC, reinterpret_cast<uintptr_t>(&prog)) == 0, "setting SECCOMP_FILTER_FLAG_TSYNC flag"); } int Client::GetMappedFD(const std::string& name) { auto it = fd_map_.find(name); SAPI_RAW_CHECK(it != fd_map_.end(), "mapped fd not found (function called twice?)"); int fd = it->second; fd_map_.erase(it); return fd; } bool Client::HasMappedFD(const std::string& name) { return fd_map_.find(name) != fd_map_.end(); } void Client::SendLogsToSupervisor() { // This LogSink will register itself and send all logs to the executor until // the object is destroyed. logsink_ = absl::make_unique<LogSink>(GetMappedFD(LogSink::kLogFDName)); } NetworkProxyClient* Client::GetNetworkProxyClient() { if (proxy_client_ == nullptr) { proxy_client_ = absl::make_unique<NetworkProxyClient>( GetMappedFD(NetworkProxyClient::kFDName)); } return proxy_client_.get(); } absl::Status Client::InstallNetworkProxyHandler() { if (fd_map_.find(NetworkProxyClient::kFDName) == fd_map_.end()) { return absl::FailedPreconditionError( "InstallNetworkProxyHandler() must be called at most once after the " "sandbox is installed. Also, the NetworkProxyServer needs to be " "enabled."); } return NetworkProxyHandler::InstallNetworkProxyHandler( GetNetworkProxyClient()); } } // namespace sandbox2