// 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. #ifndef CONTENT_BROWSER_BROWSER_CHILD_PROCESS_HOST_IMPL_H_ #define CONTENT_BROWSER_BROWSER_CHILD_PROCESS_HOST_IMPL_H_ #include #include #include #include "base/compiler_specific.h" #include "base/memory/ref_counted.h" #include "base/memory/weak_ptr.h" #include "base/memory/writable_shared_memory_region.h" #include "base/process/process.h" #include "base/single_thread_task_runner.h" #include "base/synchronization/waitable_event_watcher.h" #include "build/build_config.h" #include "content/browser/child_process_launcher.h" #include "content/browser/tracing/tracing_service_controller.h" #include "content/common/child_process.mojom.h" #include "content/common/child_process_host_impl.h" #include "content/public/browser/browser_child_process_host.h" #include "content/public/browser/child_process_data.h" #include "content/public/common/child_process_host.h" #include "content/public/common/child_process_host_delegate.h" #include "mojo/public/cpp/system/invitation.h" #include "services/resource_coordinator/public/mojom/memory_instrumentation/memory_instrumentation.mojom.h" #if defined(OS_WIN) #include "base/win/object_watcher.h" #endif namespace base { class CommandLine; } namespace content { class BrowserChildProcessHostIterator; class BrowserChildProcessObserver; class BrowserMessageFilter; // Plugins/workers and other child processes that live on the IO thread use this // class. RenderProcessHostImpl is the main exception that doesn't use this /// class because it lives on the UI thread. class CONTENT_EXPORT BrowserChildProcessHostImpl : public BrowserChildProcessHost, public ChildProcessHostDelegate, #if defined(OS_WIN) public base::win::ObjectWatcher::Delegate, #endif public ChildProcessLauncher::Client, public memory_instrumentation::mojom::CoordinatorConnector { public: // Constructs a process host with |ipc_mode| determining how IPC is done. BrowserChildProcessHostImpl(content::ProcessType process_type, BrowserChildProcessHostDelegate* delegate, ChildProcessHost::IpcMode ipc_mode); ~BrowserChildProcessHostImpl() override; // Terminates all child processes and deletes each BrowserChildProcessHost // instance. static void TerminateAll(); // Copies kEnableFeatures and kDisableFeatures to the command line. Generates // them from the FeatureList override state, to take into account overrides // from FieldTrials. static void CopyFeatureAndFieldTrialFlags(base::CommandLine* cmd_line); // Appends kTraceStartup and kTraceRecordMode flags to the command line, if // needed. static void CopyTraceStartupFlags(base::CommandLine* cmd_line); // BrowserChildProcessHost implementation: bool Send(IPC::Message* message) override; void Launch(std::unique_ptr delegate, std::unique_ptr cmd_line, bool terminate_on_shutdown) override; void LaunchWithPreloadedFiles( std::unique_ptr delegate, std::unique_ptr cmd_line, std::map files_to_preload, bool terminate_on_shutdown) override; const ChildProcessData& GetData() override; ChildProcessHost* GetHost() override; ChildProcessTerminationInfo GetTerminationInfo(bool known_dead) override; std::unique_ptr TakeMetricsAllocator() override; void SetName(const base::string16& name) override; void SetMetricsName(const std::string& metrics_name) override; void SetProcess(base::Process process) override; // ChildProcessHostDelegate implementation: void OnChannelInitialized(IPC::Channel* channel) override; void OnChildDisconnected() override; const base::Process& GetProcess() override; void BindHostReceiver(mojo::GenericPendingReceiver receiver) override; bool OnMessageReceived(const IPC::Message& message) override; void OnChannelConnected(int32_t peer_pid) override; void OnChannelError() override; void OnBadMessageReceived(const IPC::Message& message) override; // Terminates the process and logs a stack trace after a bad message was // received from the child process. void TerminateOnBadMessageReceived(const std::string& error); // Removes this host from the host list. Calls ChildProcessHost::ForceShutdown void ForceShutdown(); // Adds an IPC message filter. void AddFilter(BrowserMessageFilter* filter); // Unlike Launch(), AppendExtraCommandLineSwitches will not be called // in this function. If AppendExtraCommandLineSwitches has been called before // reaching launch, call this function instead so the command line switches // won't be appended twice void LaunchWithoutExtraCommandLineSwitches( std::unique_ptr delegate, std::unique_ptr cmd_line, std::map files_to_preload, bool terminate_on_shutdown); static void HistogramBadMessageTerminated(ProcessType process_type); #if defined(OS_ANDROID) void EnableWarmUpConnection(); void DumpProcessStack(); #endif BrowserChildProcessHostDelegate* delegate() const { return delegate_; } mojo::OutgoingInvitation* GetInProcessMojoInvitation() { return &child_process_host_->GetMojoInvitation().value(); } IPC::Channel* child_channel() const { return channel_; } mojom::ChildProcess* child_process() const { return static_cast(child_process_host_.get()) ->child_process(); } typedef std::list BrowserChildProcessList; private: friend class BrowserChildProcessHostIterator; friend class BrowserChildProcessObserver; static BrowserChildProcessList* GetIterator(); static void AddObserver(BrowserChildProcessObserver* observer); static void RemoveObserver(BrowserChildProcessObserver* observer); // Creates the |metrics_allocator_|. void CreateMetricsAllocator(); // Passes the |metrics_allocator_|, if any, to the managed process. This // requires the process to have been launched and the IPC channel to be // available. void ShareMetricsAllocatorToProcess(); // ChildProcessLauncher::Client implementation. void OnProcessLaunched() override; void OnProcessLaunchFailed(int error_code) override; #if defined(OS_ANDROID) bool CanUseWarmUpConnection() override; #endif // memory_instrumentation::mojom::CoordinatorConnector implementation: void RegisterCoordinatorClient( mojo::PendingReceiver receiver, mojo::PendingRemote client_process) override; // Returns true if the process has successfully launched. Must only be called // on the IO thread. bool IsProcessLaunched() const; static void OnMojoError( base::WeakPtr process, scoped_refptr task_runner, const std::string& error); static void TerminateProcessForBadMessage( base::WeakPtr process, const std::string& error); #if defined(OS_WIN) // ObjectWatcher::Delegate implementation. void OnObjectSignaled(HANDLE object) override; #endif ChildProcessData data_; std::string metrics_name_; BrowserChildProcessHostDelegate* delegate_; std::unique_ptr child_process_host_; mojo::Receiver coordinator_connector_receiver_{this}; std::unique_ptr child_process_; #if defined(OS_WIN) // Watches to see if the child process exits before the IPC channel has // been connected. Thereafter, its exit is determined by an error on the // IPC channel. base::win::ObjectWatcher early_exit_watcher_; #endif // The memory allocator, if any, in which the process will write its metrics. std::unique_ptr metrics_allocator_; // The shared memory region used by |metrics_allocator_| that should be // transferred to the child process. base::WritableSharedMemoryRegion metrics_shared_region_; IPC::Channel* channel_ = nullptr; bool is_channel_connected_; bool notify_child_disconnected_; #if defined(OS_ANDROID) // whether the child process can use pre-warmed up connection for better // performance. bool can_use_warm_up_connection_ = false; #endif // Keeps this process registered with the tracing subsystem. std::unique_ptr tracing_registration_; base::WeakPtrFactory weak_factory_{this}; }; } // namespace content #endif // CONTENT_BROWSER_BROWSER_CHILD_PROCESS_HOST_IMPL_H_