
Add child process launch parameter --trace-buffer-handle to facilitate writing tracing data before the child process is sandboxed. This is [1/3] CL of enabling tracing prior to sandboxing. Bug: 380411640 Change-Id: Ifadc5435c61cb2662ad3ee5b575ff477fccdb788 Reviewed-on: https://chromium-review.googlesource.com/c/chromium/src/+/6159045 Reviewed-by: Eric Seckler <eseckler@chromium.org> Reviewed-by: Alexander Timin <altimin@chromium.org> Reviewed-by: Etienne Pierre-Doray <etiennep@chromium.org> Commit-Queue: Kramer Ge <fangzhoug@chromium.org> Cr-Commit-Position: refs/heads/main@{#1410401}
300 lines
11 KiB
C++
300 lines
11 KiB
C++
// 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_
|