blob: b3681f06b0cba1fccd3710e40eec713c5d266748 [file] [log] [blame]
// Copyright 2017 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "components/performance_manager/graph/page_node_impl.h"
#include <memory>
#include "base/check_op.h"
#include "base/functional/bind.h"
#include "base/memory/raw_ptr.h"
#include "base/time/default_tick_clock.h"
#include "components/performance_manager/graph/frame_node_impl.h"
#include "components/performance_manager/graph/graph_impl.h"
#include "components/performance_manager/graph/graph_impl_operations.h"
#include "components/performance_manager/graph/process_node_impl.h"
#include "components/performance_manager/public/graph/graph_operations.h"
#include "content/public/browser/browser_thread.h"
namespace performance_manager {
namespace {
using PageState = PageNode::PageState;
bool IsValidInitialPageState(PageState page_state) {
switch (page_state) {
case PageState::kActive:
case PageState::kPrerendering:
return true;
case PageState::kBackForwardCache:
return false;
}
NOTREACHED();
return false;
}
bool IsValidPageStateTransition(PageState old_state, PageState new_state) {
switch (old_state) {
case PageState::kActive:
return new_state == PageState::kBackForwardCache;
case PageState::kPrerendering:
case PageState::kBackForwardCache:
return new_state == PageState::kActive;
}
NOTREACHED();
return false;
}
} // namespace
PageNodeImpl::PageNodeImpl(const WebContentsProxy& contents_proxy,
const std::string& browser_context_id,
const GURL& visible_url,
PagePropertyFlags initial_properties,
base::TimeTicks visibility_change_time,
PageState page_state)
: contents_proxy_(contents_proxy),
visibility_change_time_(visibility_change_time),
main_frame_url_(visible_url),
browser_context_id_(browser_context_id),
is_visible_(initial_properties.Has(PagePropertyFlag::kIsVisible)),
is_audible_(initial_properties.Has(PagePropertyFlag::kIsAudible)),
has_picture_in_picture_(
initial_properties.Has(PagePropertyFlag::kHasPictureInPicture)),
page_state_(page_state) {
// Nodes are created on the UI thread, then accessed on the PM sequence.
// `weak_this_` can be returned from GetWeakPtrOnUIThread() and dereferenced
// on the PM sequence.
DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
DETACH_FROM_SEQUENCE(sequence_checker_);
weak_this_ = weak_factory_.GetWeakPtr();
DCHECK(IsValidInitialPageState(page_state));
if (is_audible_.value()) {
audible_change_time_ = base::TimeTicks::Now();
}
}
PageNodeImpl::~PageNodeImpl() {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
DCHECK_EQ(nullptr, opener_frame_node_);
DCHECK_EQ(nullptr, embedder_frame_node_);
DCHECK_EQ(EmbeddingType::kInvalid, embedding_type_);
DCHECK(!page_load_tracker_data_);
DCHECK(!site_data_);
DCHECK(!tab_connectedness_data_);
DCHECK(!frozen_frame_data_);
DCHECK(!page_aggregator_data_);
}
const std::string& PageNodeImpl::GetBrowserContextID() const {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
return browser_context_id_;
}
resource_attribution::PageContext PageNodeImpl::GetResourceContext() const {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
return resource_attribution::PageContext::FromPageNode(this);
}
PageNodeImpl::EmbeddingType PageNodeImpl::GetEmbeddingType() const {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
DCHECK(embedder_frame_node_ || embedding_type_ == EmbeddingType::kInvalid);
return embedding_type_;
}
PageType PageNodeImpl::GetType() const {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
return type_.value();
}
bool PageNodeImpl::IsFocused() const {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
return is_focused_.value();
}
bool PageNodeImpl::IsVisible() const {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
return is_visible_.value();
}
base::TimeDelta PageNodeImpl::GetTimeSinceLastVisibilityChange() const {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
return base::TimeTicks::Now() - visibility_change_time_;
}
bool PageNodeImpl::IsAudible() const {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
return is_audible_.value();
}
std::optional<base::TimeDelta> PageNodeImpl::GetTimeSinceLastAudibleChange()
const {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
if (audible_change_time_.has_value()) {
return base::TimeTicks::Now() - audible_change_time_.value();
}
return std::nullopt;
}
bool PageNodeImpl::HasPictureInPicture() const {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
return has_picture_in_picture_.value();
}
PageNode::LoadingState PageNodeImpl::GetLoadingState() const {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
return loading_state_.value();
}
ukm::SourceId PageNodeImpl::GetUkmSourceID() const {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
return ukm_source_id_.value();
}
PageNodeImpl::LifecycleState PageNodeImpl::GetLifecycleState() const {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
return lifecycle_state_.value();
}
bool PageNodeImpl::IsHoldingWebLock() const {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
return is_holding_weblock_.value();
}
bool PageNodeImpl::IsHoldingIndexedDBLock() const {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
return is_holding_indexeddb_lock_.value();
}
int64_t PageNodeImpl::GetNavigationID() const {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
return navigation_id_;
}
const std::string& PageNodeImpl::GetContentsMimeType() const {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
return contents_mime_type_;
}
std::optional<blink::mojom::PermissionStatus>
PageNodeImpl::GetNotificationPermissionStatus() const {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
return notification_permission_status_;
}
base::TimeDelta PageNodeImpl::GetTimeSinceLastNavigation() const {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
if (navigation_committed_time_.is_null()) {
return base::TimeDelta();
}
return base::TimeTicks::Now() - navigation_committed_time_;
}
const GURL& PageNodeImpl::GetMainFrameUrl() const {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
return main_frame_url_.value();
}
uint64_t PageNodeImpl::EstimateMainFramePrivateFootprintSize() const {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
uint64_t total = 0;
FrameNodeImpl* main_frame = main_frame_node();
if (main_frame) {
performance_manager::GraphImplOperations::VisitFrameAndChildrenPreOrder(
main_frame, [&total](FrameNodeImpl* frame_node) {
total += frame_node->GetPrivateFootprintKbEstimate();
return true;
});
}
return total;
}
bool PageNodeImpl::HadFormInteraction() const {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
return had_form_interaction_.value();
}
bool PageNodeImpl::HadUserEdits() const {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
return had_user_edits_.value();
}
const WebContentsProxy& PageNodeImpl::GetContentsProxy() const {
return contents_proxy();
}
PageState PageNodeImpl::GetPageState() const {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
return page_state_.value();
}
uint64_t PageNodeImpl::EstimateResidentSetSize() const {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
uint64_t total = 0;
performance_manager::GraphOperations::VisitFrameTreePreOrder(
this, [&total](const FrameNode* frame_node) {
total += frame_node->GetResidentSetKbEstimate();
return true;
});
return total;
}
uint64_t PageNodeImpl::EstimatePrivateFootprintSize() const {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
uint64_t total = 0;
performance_manager::GraphOperations::VisitFrameTreePreOrder(
this, [&total](const FrameNode* frame_node) {
total += frame_node->GetPrivateFootprintKbEstimate();
return true;
});
return total;
}
const WebContentsProxy& PageNodeImpl::contents_proxy() const {
return contents_proxy_;
}
base::WeakPtr<PageNodeImpl> PageNodeImpl::GetWeakPtrOnUIThread() {
DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
return weak_this_;
}
base::WeakPtr<PageNodeImpl> PageNodeImpl::GetWeakPtr() {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
return weak_factory_.GetWeakPtr();
}
void PageNodeImpl::AddFrame(base::PassKey<FrameNodeImpl>,
FrameNodeImpl* frame_node) {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
DCHECK(frame_node);
DCHECK_EQ(this, frame_node->page_node());
DCHECK(graph()->NodeInGraph(frame_node));
++frame_node_count_;
if (frame_node->parent_frame_node() == nullptr)
main_frame_nodes_.insert(frame_node);
}
void PageNodeImpl::RemoveFrame(base::PassKey<FrameNodeImpl>,
FrameNodeImpl* frame_node) {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
DCHECK(frame_node);
DCHECK_EQ(this, frame_node->page_node());
DCHECK(graph()->NodeInGraph(frame_node));
--frame_node_count_;
if (frame_node->parent_frame_node() == nullptr) {
size_t removed = main_frame_nodes_.erase(frame_node);
DCHECK_EQ(1u, removed);
}
}
void PageNodeImpl::SetLoadingState(LoadingState loading_state) {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
loading_state_.SetAndMaybeNotify(this, loading_state);
}
void PageNodeImpl::SetType(PageType type) {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
type_.SetAndMaybeNotify(this, type);
}
void PageNodeImpl::SetIsFocused(bool is_focused) {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
is_focused_.SetAndMaybeNotify(this, is_focused);
}
void PageNodeImpl::SetIsVisible(bool is_visible) {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
if (is_visible_.SetAndMaybeNotify(this, is_visible)) {
// The change time needs to be updated after observers are notified, as they
// use this to determine time passed since the *previous* visibility state
// change. They can infer the current state change time themselves via
// NowTicks.
visibility_change_time_ = base::TimeTicks::Now();
}
}
void PageNodeImpl::SetIsAudible(bool is_audible) {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
if (is_audible_.SetAndMaybeNotify(this, is_audible)) {
// The change time needs to be updated after observers are notified, as they
// use this to determine time passed since the *previous* state change. They
// can infer the current state change time themselves via NowTicks.
audible_change_time_ = base::TimeTicks::Now();
}
}
void PageNodeImpl::SetHasPictureInPicture(bool has_picture_in_picture) {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
has_picture_in_picture_.SetAndMaybeNotify(this, has_picture_in_picture);
}
void PageNodeImpl::SetUkmSourceId(ukm::SourceId ukm_source_id) {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
ukm_source_id_.SetAndMaybeNotify(this, ukm_source_id);
}
void PageNodeImpl::OnFaviconUpdated() {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
for (auto* observer : GetObservers())
observer->OnFaviconUpdated(this);
}
void PageNodeImpl::OnTitleUpdated() {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
for (auto* observer : GetObservers())
observer->OnTitleUpdated(this);
}
void PageNodeImpl::OnAboutToBeDiscarded(base::WeakPtr<PageNode> new_page_node) {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
if (!new_page_node) {
return;
}
for (auto* observer : GetObservers()) {
observer->OnAboutToBeDiscarded(this, new_page_node.get());
}
}
void PageNodeImpl::OnMainFrameNavigationCommitted(
bool same_document,
base::TimeTicks navigation_committed_time,
int64_t navigation_id,
const GURL& url,
const std::string& contents_mime_type,
std::optional<blink::mojom::PermissionStatus>
notification_permission_status) {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
// This should never be invoked with a null navigation, nor should it be
// called twice for the same navigation.
DCHECK_NE(0, navigation_id);
DCHECK_NE(navigation_id_, navigation_id);
navigation_committed_time_ = navigation_committed_time;
navigation_id_ = navigation_id;
contents_mime_type_ = contents_mime_type;
notification_permission_status_ = notification_permission_status;
main_frame_url_.SetAndMaybeNotify(this, url);
// No mainframe document change notification on same-document navigations.
if (same_document)
return;
for (auto* observer : GetObservers())
observer->OnMainFrameDocumentChanged(this);
}
void PageNodeImpl::OnNotificationPermissionStatusChange(
blink::mojom::PermissionStatus permission_status) {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
notification_permission_status_ = permission_status;
}
FrameNodeImpl* PageNodeImpl::opener_frame_node() const {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
return opener_frame_node_;
}
FrameNodeImpl* PageNodeImpl::embedder_frame_node() const {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
DCHECK(embedder_frame_node_ || embedding_type_ == EmbeddingType::kInvalid);
return embedder_frame_node_;
}
FrameNodeImpl* PageNodeImpl::main_frame_node() const {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
if (main_frame_nodes_.empty())
return nullptr;
// Return the current frame node if there is one. Iterating over this set is
// fine because it is almost always of length 1 or 2.
for (FrameNodeImpl* frame : main_frame_nodes_) {
if (frame->IsCurrent()) {
return frame;
}
}
// Otherwise, return any old main frame node.
return *main_frame_nodes_.begin();
}
const base::flat_set<raw_ptr<FrameNodeImpl, CtnExperimental>>&
PageNodeImpl::main_frame_nodes() const {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
return main_frame_nodes_;
}
void PageNodeImpl::SetOpenerFrameNode(FrameNodeImpl* opener) {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
DCHECK(opener);
DCHECK(graph()->NodeInGraph(opener));
DCHECK_NE(this, opener->page_node());
auto* previous_opener = opener_frame_node_.get();
if (previous_opener)
previous_opener->RemoveOpenedPage(PassKey(), this);
opener_frame_node_ = opener;
opener->AddOpenedPage(PassKey(), this);
for (auto* observer : GetObservers())
observer->OnOpenerFrameNodeChanged(this, previous_opener);
}
void PageNodeImpl::ClearOpenerFrameNode() {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
DCHECK_NE(nullptr, opener_frame_node_);
auto* previous_opener = opener_frame_node_.get();
opener_frame_node_->RemoveOpenedPage(PassKey(), this);
opener_frame_node_ = nullptr;
for (auto* observer : GetObservers())
observer->OnOpenerFrameNodeChanged(this, previous_opener);
}
void PageNodeImpl::SetEmbedderFrameNodeAndEmbeddingType(
FrameNodeImpl* embedder,
EmbeddingType embedding_type) {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
DCHECK(embedder);
DCHECK(graph()->NodeInGraph(embedder));
DCHECK_NE(this, embedder->page_node());
DCHECK_NE(EmbeddingType::kInvalid, embedding_type);
auto* previous_embedder = embedder_frame_node_.get();
auto previous_type = embedding_type_;
if (previous_embedder)
previous_embedder->RemoveEmbeddedPage(PassKey(), this);
embedder_frame_node_ = embedder;
embedding_type_ = embedding_type;
embedder->AddEmbeddedPage(PassKey(), this);
for (auto* observer : GetObservers())
observer->OnEmbedderFrameNodeChanged(this, previous_embedder,
previous_type);
}
void PageNodeImpl::ClearEmbedderFrameNodeAndEmbeddingType() {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
DCHECK_NE(nullptr, embedder_frame_node_);
DCHECK_NE(EmbeddingType::kInvalid, embedding_type_);
auto* previous_embedder = embedder_frame_node_.get();
auto previous_type = embedding_type_;
embedder_frame_node_->RemoveEmbeddedPage(PassKey(), this);
embedder_frame_node_ = nullptr;
embedding_type_ = EmbeddingType::kInvalid;
for (auto* observer : GetObservers())
observer->OnEmbedderFrameNodeChanged(this, previous_embedder,
previous_type);
}
void PageNodeImpl::set_has_nonempty_beforeunload(
bool has_nonempty_beforeunload) {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
has_nonempty_beforeunload_ = has_nonempty_beforeunload;
}
void PageNodeImpl::set_page_state(PageState page_state) {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
DCHECK(IsValidPageStateTransition(page_state_.value(), page_state));
page_state_.SetAndMaybeNotify(this, page_state);
}
void PageNodeImpl::OnJoiningGraph() {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
// Make sure all weak pointers, even `weak_this_` that was created on the UI
// thread in the constructor, can only be dereferenced on the graph sequence.
weak_factory_.BindToCurrentSequence(
base::subtle::BindWeakPtrFactoryPassKey());
}
void PageNodeImpl::OnBeforeLeavingGraph() {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
// Sever opener relationships.
if (opener_frame_node_)
ClearOpenerFrameNode();
// Sever embedder relationships.
if (embedder_frame_node_)
ClearEmbedderFrameNodeAndEmbeddingType();
DCHECK_EQ(0u, frame_node_count_);
}
void PageNodeImpl::RemoveNodeAttachedData() {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
page_load_tracker_data_.reset();
site_data_.reset();
tab_connectedness_data_.reset();
frozen_frame_data_.Reset();
page_aggregator_data_.Reset();
}
const FrameNode* PageNodeImpl::GetOpenerFrameNode() const {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
return opener_frame_node();
}
const FrameNode* PageNodeImpl::GetEmbedderFrameNode() const {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
return embedder_frame_node();
}
const FrameNode* PageNodeImpl::GetMainFrameNode() const {
return main_frame_node();
}
bool PageNodeImpl::VisitMainFrameNodes(const FrameNodeVisitor& visitor) const {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
for (FrameNodeImpl* frame_impl : main_frame_nodes_) {
const FrameNode* frame = frame_impl;
if (!visitor(frame)) {
return false;
}
}
return true;
}
const base::flat_set<raw_ptr<const FrameNode, CtnExperimental>>
PageNodeImpl::GetMainFrameNodes() const {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
base::flat_set<raw_ptr<const FrameNode, CtnExperimental>> main_frame_nodes(
main_frame_nodes_.begin(), main_frame_nodes_.end());
return main_frame_nodes;
}
void PageNodeImpl::SetLifecycleState(LifecycleState lifecycle_state) {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
lifecycle_state_.SetAndMaybeNotify(this, lifecycle_state);
}
void PageNodeImpl::SetIsHoldingWebLock(bool is_holding_weblock) {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
is_holding_weblock_.SetAndMaybeNotify(this, is_holding_weblock);
}
void PageNodeImpl::SetIsHoldingIndexedDBLock(bool is_holding_indexeddb_lock) {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
is_holding_indexeddb_lock_.SetAndMaybeNotify(this, is_holding_indexeddb_lock);
}
void PageNodeImpl::SetHadFormInteraction(bool had_form_interaction) {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
had_form_interaction_.SetAndMaybeNotify(this, had_form_interaction);
}
void PageNodeImpl::SetHadUserEdits(bool had_user_edits) {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
had_user_edits_.SetAndMaybeNotify(this, had_user_edits);
}
} // namespace performance_manager