| // Copyright 2012 The Chromium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "chromeos/process_proxy/process_proxy_registry.h" |
| |
| #include <memory> |
| |
| #include "base/command_line.h" |
| #include "base/functional/bind.h" |
| #include "base/message_loop/message_pump_type.h" |
| #include "base/strings/string_number_conversions.h" |
| #include "base/task/lazy_thread_pool_task_runner.h" |
| #include "base/task/sequenced_task_runner.h" |
| |
| namespace chromeos { |
| |
| namespace { |
| |
| const char kWatcherThreadName[] = "ProcessWatcherThread"; |
| |
| const char kStdoutOutputType[] = "stdout"; |
| const char kExitOutputType[] = "exit"; |
| |
| const char* ProcessOutputTypeToString(ProcessOutputType type) { |
| switch (type) { |
| case PROCESS_OUTPUT_TYPE_OUT: |
| return kStdoutOutputType; |
| case PROCESS_OUTPUT_TYPE_EXIT: |
| return kExitOutputType; |
| default: |
| return NULL; |
| } |
| } |
| |
| // This instance must be leaked because the destructor would be run on the main |
| // thread, and not the task runner. |
| static base::LazyInstance<ProcessProxyRegistry>::Leaky |
| g_process_proxy_registry = LAZY_INSTANCE_INITIALIZER; |
| |
| } // namespace |
| |
| ProcessProxyRegistry::ProcessProxyInfo::ProcessProxyInfo() = default; |
| |
| ProcessProxyRegistry::ProcessProxyInfo::ProcessProxyInfo( |
| const ProcessProxyInfo& other) { |
| // This should be called with empty info only. |
| DCHECK(!other.proxy.get()); |
| } |
| |
| ProcessProxyRegistry::ProcessProxyInfo::~ProcessProxyInfo() = default; |
| |
| ProcessProxyRegistry::ProcessProxyRegistry() = default; |
| |
| ProcessProxyRegistry::~ProcessProxyRegistry() { |
| DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_); |
| |
| ShutDown(); |
| } |
| |
| void ProcessProxyRegistry::ShutDown() { |
| DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_); |
| |
| // Close all proxies we own. |
| while (!proxy_map_.empty()) |
| CloseProcess(proxy_map_.begin()->first); |
| |
| if (watcher_thread_) { |
| watcher_thread_->Stop(); |
| watcher_thread_.reset(); |
| } |
| } |
| |
| // static |
| ProcessProxyRegistry* ProcessProxyRegistry::Get() { |
| DCHECK(ProcessProxyRegistry::GetTaskRunner()->RunsTasksInCurrentSequence()); |
| return g_process_proxy_registry.Pointer(); |
| } |
| |
| // static |
| int ProcessProxyRegistry::ConvertToSystemPID(const std::string& id) { |
| // The `id` is <pid>-<guid>. `base::StringToInt()` will parse until the '-'. |
| int out; |
| base::StringToInt(id, &out); |
| return out; |
| } |
| |
| // static |
| scoped_refptr<base::SequencedTaskRunner> ProcessProxyRegistry::GetTaskRunner() { |
| static base::LazyThreadPoolSequencedTaskRunner task_runner = |
| LAZY_THREAD_POOL_SEQUENCED_TASK_RUNNER_INITIALIZER( |
| base::TaskTraits(base::MayBlock(), base::TaskPriority::BEST_EFFORT)); |
| return task_runner.Get(); |
| } |
| |
| bool ProcessProxyRegistry::OpenProcess(const base::CommandLine& cmdline, |
| const std::string& user_id_hash, |
| const OutputCallback& output_callback, |
| std::string* id) { |
| DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_); |
| |
| if (!EnsureWatcherThreadStarted()) |
| return false; |
| |
| // Create and open new proxy. |
| scoped_refptr<ProcessProxy> proxy(new ProcessProxy()); |
| if (!proxy->Open(cmdline, user_id_hash, id)) |
| return false; |
| |
| // Kick off watcher. |
| // We can use Unretained because proxy will stop calling callback after it is |
| // closed, which is done before this object goes away. |
| if (!proxy->StartWatchingOutput( |
| watcher_thread_->task_runner(), GetTaskRunner(), |
| base::BindRepeating(&ProcessProxyRegistry::OnProcessOutput, |
| base::Unretained(this), *id))) { |
| proxy->Close(); |
| return false; |
| } |
| |
| ProcessProxyInfo& info = proxy_map_[*id]; |
| info.proxy.swap(proxy); |
| info.callback = output_callback; |
| |
| return true; |
| } |
| |
| void ProcessProxyRegistry::SendInput(const std::string& id, |
| const std::string& data, |
| base::OnceCallback<void(bool)> callback) { |
| DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_); |
| |
| std::map<std::string, ProcessProxyInfo>::iterator it = proxy_map_.find(id); |
| if (it == proxy_map_.end()) |
| return std::move(callback).Run(false); |
| it->second.proxy->Write(data, std::move(callback)); |
| } |
| |
| bool ProcessProxyRegistry::CloseProcess(const std::string& id) { |
| DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_); |
| |
| std::map<std::string, ProcessProxyInfo>::iterator it = proxy_map_.find(id); |
| if (it == proxy_map_.end()) |
| return false; |
| |
| it->second.proxy->Close(); |
| proxy_map_.erase(it); |
| return true; |
| } |
| |
| bool ProcessProxyRegistry::OnTerminalResize(const std::string& id, |
| int width, |
| int height) { |
| DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_); |
| |
| std::map<std::string, ProcessProxyInfo>::iterator it = proxy_map_.find(id); |
| if (it == proxy_map_.end()) |
| return false; |
| |
| return it->second.proxy->OnTerminalResize(width, height); |
| } |
| |
| void ProcessProxyRegistry::AckOutput(const std::string& id) { |
| DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_); |
| |
| std::map<std::string, ProcessProxyInfo>::iterator it = proxy_map_.find(id); |
| if (it == proxy_map_.end()) |
| return; |
| |
| it->second.proxy->AckOutput(); |
| } |
| |
| void ProcessProxyRegistry::OnProcessOutput(const std::string& id, |
| ProcessOutputType type, |
| const std::string& data) { |
| DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_); |
| |
| const char* type_str = ProcessOutputTypeToString(type); |
| DCHECK(type_str); |
| |
| std::map<std::string, ProcessProxyInfo>::iterator it = proxy_map_.find(id); |
| if (it == proxy_map_.end()) |
| return; |
| it->second.callback.Run(id, std::string(type_str), data); |
| |
| // Contact with the slave end of the terminal has been lost. We have to close |
| // the process. |
| if (type == PROCESS_OUTPUT_TYPE_EXIT) |
| CloseProcess(id); |
| } |
| |
| bool ProcessProxyRegistry::EnsureWatcherThreadStarted() { |
| DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_); |
| if (watcher_thread_.get()) |
| return true; |
| |
| // TODO(tbarzic): Change process output watcher to watch for fd readability on |
| // FILE thread, and move output reading to worker thread instead of |
| // spinning a new thread. |
| watcher_thread_ = std::make_unique<base::Thread>(kWatcherThreadName); |
| return watcher_thread_->StartWithOptions( |
| base::Thread::Options(base::MessagePumpType::IO, 0)); |
| } |
| |
| const base::Process* ProcessProxyRegistry::GetProcessForTesting( |
| const std::string& id) { |
| DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_); |
| std::map<std::string, ProcessProxyInfo>::iterator it = proxy_map_.find(id); |
| if (it == proxy_map_.end()) |
| return nullptr; |
| |
| return it->second.proxy->GetProcessForTesting(); // IN-TEST |
| } |
| |
| } // namespace chromeos |