1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299
|
// 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.
#ifndef CONTENT_BROWSER_BROWSER_CHILD_PROCESS_HOST_IMPL_H_
#define CONTENT_BROWSER_BROWSER_CHILD_PROCESS_HOST_IMPL_H_
#include <stdint.h>
#include <list>
#include <memory>
#include "base/memory/raw_ptr.h"
#include "base/memory/ref_counted.h"
#include "base/memory/unsafe_shared_memory_region.h"
#include "base/memory/weak_ptr.h"
#include "base/process/process.h"
#include "base/synchronization/waitable_event_watcher.h"
#include "base/task/single_thread_task_runner.h"
#include "build/build_config.h"
#include "components/metrics/histogram_child_process.h"
#include "content/browser/child_process_host_impl.h"
#include "content/browser/child_process_launcher.h"
#include "content/browser/tracing/tracing_service_controller.h"
#include "content/common/buildflags.h"
#include "content/common/child_process.mojom.h"
#include "content/public/browser/browser_child_process_host.h"
#include "content/public/browser/child_process_data.h"
#include "content/public/browser/child_process_host.h"
#include "content/public/browser/child_process_host_delegate.h"
#include "mojo/public/cpp/bindings/binder_map.h"
#include "mojo/public/cpp/system/invitation.h"
#include "services/resource_coordinator/public/mojom/memory_instrumentation/memory_instrumentation.mojom.h"
#if BUILDFLAG(IS_WIN)
#include "base/win/object_watcher.h"
#endif
#if BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS)
#include "content/browser/child_thread_type_switcher_linux.h"
#endif // BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS)
namespace base {
class CommandLine;
}
namespace tracing {
class SystemTracingService;
}
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 BrowserChildProcessHostImpl
: public BrowserChildProcessHost,
public ChildProcessHostDelegate,
public metrics::HistogramChildProcess,
#if BUILDFLAG(IS_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();
// BrowserChildProcessHost implementation:
bool Send(IPC::Message* message) override;
void Launch(std::unique_ptr<SandboxedProcessLauncherDelegate> delegate,
std::unique_ptr<base::CommandLine> cmd_line,
bool terminate_on_shutdown) override;
const ChildProcessData& GetData() override;
ChildProcessHost* GetHost() override;
ChildProcessTerminationInfo GetTerminationInfo(bool known_dead) override;
std::unique_ptr<base::PersistentMemoryAllocator> TakeMetricsAllocator()
override;
void SetName(const std::u16string& 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;
// HistogramChildProcess implementation:
void BindChildHistogramFetcherFactory(
mojo::PendingReceiver<metrics::mojom::ChildHistogramFetcherFactory>
factory) 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();
#if BUILDFLAG(CONTENT_ENABLE_LEGACY_IPC)
// Adds an IPC message filter.
void AddFilter(BrowserMessageFilter* filter);
#endif
// Same as Launch(), but the process is launched with preloaded files and file
// descriptors containing in `file_data`.
void LaunchWithFileData(
std::unique_ptr<SandboxedProcessLauncherDelegate> delegate,
std::unique_ptr<base::CommandLine> cmd_line,
std::unique_ptr<ChildProcessLauncherFileData> file_data,
bool terminate_on_shutdown);
// 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<SandboxedProcessLauncherDelegate> delegate,
std::unique_ptr<base::CommandLine> cmd_line,
std::unique_ptr<ChildProcessLauncherFileData> file_data,
bool terminate_on_shutdown);
#if !BUILDFLAG(IS_ANDROID)
void SetProcessPriority(base::Process::Priority priority);
#endif // !BUILDFLAG(IS_ANDROID)
#if BUILDFLAG(IS_ANDROID)
void EnableWarmUpConnection();
void DumpProcessStack();
#endif
BrowserChildProcessHostDelegate* delegate() const { return delegate_; }
mojo::OutgoingInvitation* GetInProcessMojoInvitation() {
in_process_ = true;
return &child_process_host_->GetMojoInvitation().value();
}
mojom::ChildProcess* child_process() const {
return static_cast<ChildProcessHostImpl*>(child_process_host_.get())
->child_process();
}
typedef std::list<raw_ptr<BrowserChildProcessHostImpl, CtnExperimental>>
BrowserChildProcessList;
private:
friend class BrowserChildProcessHostIterator;
friend class BrowserChildProcessObserver;
void OnProcessConnected();
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 BUILDFLAG(IS_ANDROID)
bool CanUseWarmUpConnection() override;
#endif
// memory_instrumentation::mojom::CoordinatorConnector implementation:
void RegisterCoordinatorClient(
mojo::PendingReceiver<memory_instrumentation::mojom::Coordinator>
receiver,
mojo::PendingRemote<memory_instrumentation::mojom::ClientProcess>
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<BrowserChildProcessHostImpl> process,
scoped_refptr<base::SingleThreadTaskRunner> task_runner,
const std::string& error);
static void TerminateProcessForBadMessage(
base::WeakPtr<BrowserChildProcessHostImpl> process,
const std::string& error);
#if BUILDFLAG(IS_WIN)
// ObjectWatcher::Delegate implementation.
void OnObjectSignaled(HANDLE object) override;
#endif
ChildProcessData data_;
std::string metrics_name_;
raw_ptr<BrowserChildProcessHostDelegate> delegate_;
std::unique_ptr<ChildProcessHost> child_process_host_;
mojo::Receiver<memory_instrumentation::mojom::CoordinatorConnector>
coordinator_connector_receiver_{this};
mojo::BinderMapWithContext<BrowserChildProcessHost*> binder_map_;
std::unique_ptr<ChildProcessLauncher> child_process_launcher_;
#if BUILDFLAG(IS_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<base::PersistentMemoryAllocator> metrics_allocator_;
// The histogram shared memory region used to transmit metrics. The memory
// region is allocated by the process host (this object) but ownership is
// shared with the child process launcher/helper which runs, and is destroyed,
// asynchronously. Depending on the feature configuration, either the host or
// the launcher is responsible for passing the memory region to the child.
// The destruction order of the host, launcher and child are indeterminate.
scoped_refptr<base::RefCountedData<base::UnsafeSharedMemoryRegion>>
metrics_shared_region_;
// The tracing config memory region. Ownership of the memory region object is
// shared with the child process launcher/helper which runs, and is destroyed,
// asynchronously.
scoped_refptr<base::RefCountedData<base::ReadOnlySharedMemoryRegion>>
tracing_config_memory_region_;
// The tracing output memory region to transmit traces. Ownership of the
// memory region object is shared with the child process launcher/helper which
// runs, and is destroyed, asynchronously.
scoped_refptr<base::RefCountedData<base::UnsafeSharedMemoryRegion>>
tracing_output_memory_region_;
// Indicates if the main browser process is used instead of a dedicated child
// process.
bool in_process_ = false;
// Indicates if legacy IPC is used to communicate with the child process. In
// this mode, the BrowserChildProcessHost waits for OnChannelConnected() to be
// called before sending the BrowserChildProcessLaunchedAndConnected
// notification.
bool has_legacy_ipc_channel_ = false;
// Indicates if the IPC channel is connected. Always true when not using
// legacy IPC.
bool is_channel_connected_ = true;
// Indicates if the BrowserChildProcessLaunchedAndConnected notification was
// sent for this instance.
bool launched_and_connected_ = false;
// Whether the child process exited abnormally (killed or crashed).
bool exited_abnormally_ = false;
#if BUILDFLAG(IS_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<TracingServiceController::ClientRegistration>
tracing_registration_;
#if BUILDFLAG(IS_POSIX) && !BUILDFLAG(IS_ANDROID)
// For child process to connect to the system tracing service.
std::unique_ptr<tracing::SystemTracingService> system_tracing_service_;
#endif
#if BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS)
ChildThreadTypeSwitcher child_thread_type_switcher_;
#endif // BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS)
base::WeakPtrFactory<BrowserChildProcessHostImpl> weak_factory_{this};
};
} // namespace content
#endif // CONTENT_BROWSER_BROWSER_CHILD_PROCESS_HOST_IMPL_H_
|