Bug 1751965 - Remove destroyed display items from reused display items list r=mstange

Differential Revision: https://phabricator.services.mozilla.com/D137235
This commit is contained in:
Miko Mynttinen 2022-01-28 16:30:55 +00:00
Родитель dff729bc6d
Коммит 2175f47f8f
5 изменённых файлов: 68 добавлений и 41 удалений

Просмотреть файл

@ -3336,16 +3336,16 @@ void nsLayoutUtils::PaintFrame(gfxContext* aRenderingContext, nsIFrame* aFrame,
} }
if (doFullRebuild) { if (doFullRebuild) {
list->DeleteAll(builder);
list->RestoreState();
if (useRetainedBuilder) { if (useRetainedBuilder) {
retainedBuilder->ClearFramesWithProps(); retainedBuilder->ClearFramesWithProps();
retainedBuilder->ClearReuseableDisplayItems();
mozilla::RDLUtils::AssertFrameSubtreeUnmodified( mozilla::RDLUtils::AssertFrameSubtreeUnmodified(
builder->RootReferenceFrame()); builder->RootReferenceFrame());
MOZ_ASSERT(retainedBuilder->List()->IsEmpty());
} }
list->DeleteAll(builder);
list->RestoreState();
builder->ClearRetainedWindowRegions(); builder->ClearRetainedWindowRegions();
builder->ClearWillChangeBudgets(); builder->ClearWillChangeBudgets();

Просмотреть файл

@ -1507,7 +1507,8 @@ MOZ_NEVER_INLINE_DEBUG void ReuseStackingContextItem(
aItem->UpdateBounds(aBuilder); aItem->UpdateBounds(aBuilder);
} }
DL_LOGD("Retaining display item %p", aItem); aBuilder->AddReusableDisplayItem(aItem);
DL_LOGD("Reusing display item %p", aItem);
} }
bool IsSupportedFrameType(const nsIFrame* aFrame) { bool IsSupportedFrameType(const nsIFrame* aFrame) {
@ -1558,10 +1559,8 @@ bool IsReuseableStackingContextItem(nsDisplayItem* aItem) {
* linked and collected in given |aOutItems| array. * linked and collected in given |aOutItems| array.
*/ */
void CollectStackingContextItems(nsDisplayListBuilder* aBuilder, void CollectStackingContextItems(nsDisplayListBuilder* aBuilder,
nsDisplayList* aList, nsDisplayList* aList, nsIFrame* aOuterFrame,
nsTArray<nsDisplayItem*>& aOutItems, int aDepth = 0, bool aParentReused = false) {
nsIFrame* aOuterFrame, int aDepth = 0,
bool aParentReused = false) {
nsDisplayList out; nsDisplayList out;
while (nsDisplayItem* item = aList->RemoveBottom()) { while (nsDisplayItem* item = aList->RemoveBottom()) {
@ -1593,8 +1592,8 @@ void CollectStackingContextItems(nsDisplayListBuilder* aBuilder,
const bool isStackingContextItem = IsReuseableStackingContextItem(item); const bool isStackingContextItem = IsReuseableStackingContextItem(item);
if (item->GetChildren()) { if (item->GetChildren()) {
CollectStackingContextItems(aBuilder, item->GetChildren(), aOutItems, CollectStackingContextItems(aBuilder, item->GetChildren(), item->Frame(),
item->Frame(), aDepth + 1, aDepth + 1,
aParentReused || isStackingContextItem); aParentReused || isStackingContextItem);
} }
@ -1604,7 +1603,6 @@ void CollectStackingContextItems(nsDisplayListBuilder* aBuilder,
out.AppendToTop(item); out.AppendToTop(item);
} else if (isStackingContextItem) { } else if (isStackingContextItem) {
// |item| is a stacking context item that can be reused. // |item| is a stacking context item that can be reused.
aOutItems.AppendElement(item);
ReuseStackingContextItem(aBuilder, item); ReuseStackingContextItem(aBuilder, item);
} else { } else {
// |item| is inside a container item that will be destroyed later. // |item| is inside a container item that will be destroyed later.
@ -1622,27 +1620,6 @@ void CollectStackingContextItems(nsDisplayListBuilder* aBuilder,
aList->RestoreState(); aList->RestoreState();
} }
/**
* Destroys the retained stacking context items that have not been reused and
* clears the array.
*/
void ClearPreviousItems(nsDisplayListBuilder* aBuilder,
nsTArray<nsDisplayItem*>& aItems) {
const size_t total = aItems.Length();
size_t reused = 0;
for (auto* item : aItems) {
if (item->IsReusedItem()) {
reused++;
item->SetReusable();
} else {
item->Destroy(aBuilder);
}
}
DL_LOGI("RDL - Reused %zu of %zu SC display items", reused, total);
aItems.Clear();
}
} // namespace RDL } // namespace RDL
bool RetainedDisplayListBuilder::TrySimpleUpdate( bool RetainedDisplayListBuilder::TrySimpleUpdate(
@ -1652,10 +1629,8 @@ bool RetainedDisplayListBuilder::TrySimpleUpdate(
return false; return false;
} }
MOZ_ASSERT(mPreviousItems.IsEmpty());
RDL::MarkAllAncestorFrames(aModifiedFrames, aOutFramesWithProps); RDL::MarkAllAncestorFrames(aModifiedFrames, aOutFramesWithProps);
RDL::CollectStackingContextItems(&mBuilder, &mList, mPreviousItems, RDL::CollectStackingContextItems(&mBuilder, &mList, RootReferenceFrame());
RootReferenceFrame());
return true; return true;
} }
@ -1746,7 +1721,7 @@ PartialUpdateResult RetainedDisplayListBuilder::AttemptPartialUpdate(
if (mBuilder.PartialBuildFailed()) { if (mBuilder.PartialBuildFailed()) {
DL_LOGI("RDL - Partial update failed!"); DL_LOGI("RDL - Partial update failed!");
mBuilder.LeavePresShell(RootReferenceFrame(), nullptr); mBuilder.LeavePresShell(RootReferenceFrame(), nullptr);
RDL::ClearPreviousItems(&mBuilder, mPreviousItems); mBuilder.ClearReuseableDisplayItems();
mList.DeleteAll(&mBuilder); mList.DeleteAll(&mBuilder);
modifiedDL.DeleteAll(&mBuilder); modifiedDL.DeleteAll(&mBuilder);
Metrics()->mPartialUpdateFailReason = PartialUpdateFailReason::Content; Metrics()->mPartialUpdateFailReason = PartialUpdateFailReason::Content;
@ -1775,7 +1750,7 @@ PartialUpdateResult RetainedDisplayListBuilder::AttemptPartialUpdate(
} else { } else {
MOZ_ASSERT(mList.IsEmpty()); MOZ_ASSERT(mList.IsEmpty());
mList = std::move(modifiedDL); mList = std::move(modifiedDL);
RDL::ClearPreviousItems(&mBuilder, mPreviousItems); mBuilder.ClearReuseableDisplayItems();
result = PartialUpdateResult::Updated; result = PartialUpdateResult::Updated;
} }

Просмотреть файл

@ -195,6 +195,9 @@ struct RetainedDisplayListBuilder {
* the frames in the modified frame lists. * the frames in the modified frame lists.
*/ */
void ClearFramesWithProps(); void ClearFramesWithProps();
void ClearReuseableDisplayItems() { mBuilder.ClearReuseableDisplayItems(); }
void AddSizeOfIncludingThis(nsWindowSizes&) const; void AddSizeOfIncludingThis(nsWindowSizes&) const;
NS_DECLARE_FRAME_PROPERTY_DELETABLE(Cached, RetainedDisplayListBuilder) NS_DECLARE_FRAME_PROPERTY_DELETABLE(Cached, RetainedDisplayListBuilder)
@ -272,9 +275,6 @@ struct RetainedDisplayListBuilder {
RetainedDisplayList mList; RetainedDisplayList mList;
WeakFrame mPreviousCaret; WeakFrame mPreviousCaret;
RetainedDisplayListMetrics mMetrics; RetainedDisplayListMetrics mMetrics;
// Stores reusable items collected during display list preprocessing.
nsTArray<nsDisplayItem*> mPreviousItems;
}; };
namespace RDLUtils { namespace RDLUtils {

Просмотреть файл

@ -2008,6 +2008,32 @@ void nsDisplayListBuilder::BuildCompositorHitTestInfoIfNeeded(
} }
} }
void nsDisplayListBuilder::AddReusableDisplayItem(nsDisplayItem* aItem) {
mReuseableItems.Insert(aItem);
}
void nsDisplayListBuilder::RemoveReusedDisplayItem(nsDisplayItem* aItem) {
MOZ_ASSERT(aItem->IsReusedItem());
mReuseableItems.Remove(aItem);
}
void nsDisplayListBuilder::ClearReuseableDisplayItems() {
const size_t total = mReuseableItems.Count();
size_t reused = 0;
for (auto* item : mReuseableItems) {
if (item->IsReusedItem()) {
reused++;
item->SetReusable();
} else {
item->Destroy(this);
}
}
DL_LOGI("RDL - Reused %zu of %zu SC display items", reused, total);
mReuseableItems.Clear();
}
void nsDisplayListBuilder::ReuseDisplayItem(nsDisplayItem* aItem) { void nsDisplayListBuilder::ReuseDisplayItem(nsDisplayItem* aItem) {
const auto* previous = mCurrentContainerASR; const auto* previous = mCurrentContainerASR;
const auto* asr = aItem->GetActiveScrolledRoot(); const auto* asr = aItem->GetActiveScrolledRoot();

Просмотреть файл

@ -1699,6 +1699,24 @@ class nsDisplayListBuilder {
return mIsReusingStackingContextItems; return mIsReusingStackingContextItems;
} }
/**
* Adds display item |aItem| to the reuseable display items set.
*/
void AddReusableDisplayItem(nsDisplayItem* aItem);
/**
* Removes display item |aItem| from the reuseable display items set.
* This is needed because display items are sometimes deleted during
* display list building.
* Called by |nsDisplayItem::Destroy()| when the item has been reused.
*/
void RemoveReusedDisplayItem(nsDisplayItem* aItem);
/**
* Clears the reuseable display items set.
*/
void ClearReuseableDisplayItems();
/** /**
* Marks the given display item |aItem| as reused, and updates the necessary * Marks the given display item |aItem| as reused, and updates the necessary
* display list builder state. * display list builder state.
@ -1900,6 +1918,9 @@ class nsDisplayListBuilder {
bool mIsForContent; bool mIsForContent;
bool mIsReusingStackingContextItems; bool mIsReusingStackingContextItems;
// Stores reusable items collected during display list preprocessing.
nsTHashSet<nsDisplayItem*> mReuseableItems;
}; };
class nsDisplayItem; class nsDisplayItem;
@ -2094,6 +2115,11 @@ class nsDisplayItem : public nsDisplayItemLink {
if (aBuilder->IsForPainting() && aBuilder->IsForContent()) { if (aBuilder->IsForPainting() && aBuilder->IsForContent()) {
DL_LOGV("Destroying display item %p (%s)", this, Name()); DL_LOGV("Destroying display item %p (%s)", this, Name());
} }
if (IsReusedItem()) {
aBuilder->RemoveReusedDisplayItem(this);
}
this->~nsDisplayItem(); this->~nsDisplayItem();
aBuilder->Destroy(type, this); aBuilder->Destroy(type, this);
} }