Skip to content

Commit

Permalink
Bug 1488599 - Part 2: Fix will-change budget r=mattwoodrow
Browse files Browse the repository at this point in the history
Depends on D5245

Differential Revision: https://phabricator.services.mozilla.com/D5246

--HG--
extra : moz-landing-system : lando
  • Loading branch information
mikokm committed Sep 19, 2018
1 parent 03d0311 commit 5107e2d
Show file tree
Hide file tree
Showing 5 changed files with 61 additions and 40 deletions.
8 changes: 6 additions & 2 deletions layout/base/nsLayoutUtils.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -3700,11 +3700,15 @@ nsLayoutUtils::PaintFrame(gfxContext* aRenderingContext, nsIFrame* aFrame,

if (updateState == PartialUpdateResult::Failed) {
list.DeleteAll(&builder);

builder.ClearRetainedWindowRegions();
builder.ClearWillChangeBudget();

builder.EnterPresShell(aFrame);
builder.SetDirtyRect(visibleRect);
builder.ClearRetainedWindowRegions();
aFrame->BuildDisplayListForStackingContext(&builder, &list);
AddExtraBackgroundItems(builder, list, aFrame, canvasArea, visibleRegion, aBackstop);
AddExtraBackgroundItems(
builder, list, aFrame, canvasArea, visibleRegion, aBackstop);

builder.LeavePresShell(aFrame, &list);
updateState = PartialUpdateResult::Updated;
Expand Down
10 changes: 8 additions & 2 deletions layout/generic/nsFrame.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -986,6 +986,12 @@ nsIFrame::RemoveDisplayItemDataForDeletion()

RetainedDisplayListData* data = GetOrSetRetainedDisplayListData(rootFrame);

if (MayHaveWillChangeBudget()) {
// Keep the frame in list, so it can be removed from the will-change budget.
data->Flags(this) = RetainedDisplayListData::FrameFlags::HadWillChange;
return;
}

if (IsFrameModified() || HasOverrideDirtyRegion()) {
// Remove deleted frames from RetainedDisplayListData.
DebugOnly<bool> removed = data->Remove(this);
Expand Down Expand Up @@ -3486,7 +3492,7 @@ nsIFrame::BuildDisplayListForChild(nsDisplayListBuilder* aBuilder,
NS_FRAME_TOO_DEEP_IN_FRAME_TREE | NS_FRAME_IS_NONDISPLAY))
return;

aBuilder->ClearWillChangeBudget(child);
aBuilder->RemoveFromWillChangeBudget(child);

const bool shortcutPossible = aBuilder->IsPaintingToWindow() &&
aBuilder->BuildCompositorHitTestInfo();
Expand Down Expand Up @@ -3563,7 +3569,7 @@ nsIFrame::BuildDisplayListForChild(nsDisplayListBuilder* aBuilder,
isPlaceholder = true;
nsPlaceholderFrame* placeholder = static_cast<nsPlaceholderFrame*>(child);
child = placeholder->GetOutOfFlowFrame();
aBuilder->ClearWillChangeBudget(child);
aBuilder->RemoveFromWillChangeBudget(child);
NS_ASSERTION(child, "No out of flow frame?");
// If 'child' is a pushed float then it's owned by a block that's not an
// ancestor of the placeholder, and it will be painted by that block and
Expand Down
16 changes: 10 additions & 6 deletions layout/painting/RetainedDisplayListBuilder.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -660,6 +660,7 @@ RetainedDisplayListBuilder::MergeDisplayLists(

static void
TakeAndAddModifiedAndFramesWithPropsFromRootFrame(
nsDisplayListBuilder* aBuilder,
nsTArray<nsIFrame*>* aModifiedFrames,
nsTArray<nsIFrame*>* aFramesWithProps,
nsIFrame* aRootFrame)
Expand All @@ -683,6 +684,10 @@ TakeAndAddModifiedAndFramesWithPropsFromRootFrame(
if (flags & RetainedDisplayListData::FrameFlags::HasProps) {
aFramesWithProps->AppendElement(frame);
}

if (flags & RetainedDisplayListData::FrameFlags::HadWillChange) {
aBuilder->RemoveFromWillChangeBudget(frame);
}
}

data->Clear();
Expand Down Expand Up @@ -748,7 +753,7 @@ SubDocEnumCb(nsIDocument* aDocument, void* aData)
nsIFrame* rootFrame = GetRootFrameForPainting(data->builder, aDocument);
if (rootFrame) {
TakeAndAddModifiedAndFramesWithPropsFromRootFrame(
data->modifiedFrames, data->framesWithProps, rootFrame);
data->builder, data->modifiedFrames, data->framesWithProps, rootFrame);

nsIDocument* innerDoc = rootFrame->PresShell()->GetDocument();
if (innerDoc) {
Expand All @@ -763,14 +768,13 @@ GetModifiedAndFramesWithProps(nsDisplayListBuilder* aBuilder,
nsTArray<nsIFrame*>* aOutModifiedFrames,
nsTArray<nsIFrame*>* aOutFramesWithProps)
{
MOZ_ASSERT(aBuilder->RootReferenceFrame());
nsIFrame* rootFrame = aBuilder->RootReferenceFrame();
MOZ_ASSERT(rootFrame);

TakeAndAddModifiedAndFramesWithPropsFromRootFrame(
aOutModifiedFrames, aOutFramesWithProps, aBuilder->RootReferenceFrame());

nsIDocument* rootdoc =
aBuilder->RootReferenceFrame()->PresContext()->Document();
aBuilder, aOutModifiedFrames, aOutFramesWithProps, rootFrame);

nsIDocument* rootdoc = rootFrame->PresContext()->Document();
if (rootdoc) {
CbData data = { aBuilder, aOutModifiedFrames, aOutFramesWithProps };

Expand Down
48 changes: 23 additions & 25 deletions layout/painting/nsDisplayList.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -1060,6 +1060,7 @@ nsDisplayListBuilder::EndFrame()
NS_ASSERTION(!mInInvalidSubtree,
"Someone forgot to cleanup mInInvalidSubtree!");
mFrameToAnimatedGeometryRootMap.Clear();
mAGRBudgetSet.Clear();
mActiveScrolledRoots.Clear();
FreeClipChains();
FreeTemporaryItems();
Expand Down Expand Up @@ -2120,29 +2121,20 @@ nsDisplayListBuilder::AddToWillChangeBudget(nsIFrame* aFrame,
return true; // Already accounted
}

nsPresContext* key = aFrame->PresContext();
DocumentWillChangeBudget budget;
auto willChangeBudgetEntry = mWillChangeBudget.LookupForAdd(key);
if (willChangeBudgetEntry) {
// We have an existing entry.
budget = willChangeBudgetEntry.Data();
} else {
budget = DocumentWillChangeBudget();
willChangeBudgetEntry.OrInsert([&budget]() { return budget; });
}

nsRect area = aFrame->PresContext()->GetVisibleArea();
nsPresContext* presContext = aFrame->PresContext();
nsRect area = presContext->GetVisibleArea();
uint32_t budgetLimit = nsPresContext::AppUnitsToIntCSSPixels(area.width) *
nsPresContext::AppUnitsToIntCSSPixels(area.height);

uint32_t cost = GetLayerizationCost(aSize);

DocumentWillChangeBudget& budget = mWillChangeBudget.GetOrInsert(presContext);

bool onBudget =
(budget.mBudget + cost) / gWillChangeAreaMultiplier < budgetLimit;

if (onBudget) {
budget.mBudget += cost;
willChangeBudgetEntry.Data() = budget;
mWillChangeBudgetSet.Put(aFrame, cost);
mWillChangeBudgetSet.Put(aFrame, FrameWillChangeBudget(presContext, cost));
aFrame->SetMayHaveWillChangeBudget(true);
}

Expand Down Expand Up @@ -2179,23 +2171,29 @@ nsDisplayListBuilder::IsInWillChangeBudget(nsIFrame* aFrame,
}

void
nsDisplayListBuilder::ClearWillChangeBudget(nsIFrame* aFrame)
nsDisplayListBuilder::RemoveFromWillChangeBudget(nsIFrame* aFrame)
{
if (!aFrame->MayHaveWillChangeBudget()) {
FrameWillChangeBudget* frameBudget = mWillChangeBudgetSet.GetValue(aFrame);

if (!frameBudget) {
return;
}
aFrame->SetMayHaveWillChangeBudget(false);

uint32_t cost = 0;
if (!mWillChangeBudgetSet.Get(aFrame, &cost)) {
return;
DocumentWillChangeBudget* budget =
mWillChangeBudget.GetValue(frameBudget->mPresContext);

if (budget) {
budget->mBudget -= frameBudget->mUsage;
}

mWillChangeBudgetSet.Remove(aFrame);
}

DocumentWillChangeBudget& budget =
mWillChangeBudget.GetOrInsert(aFrame->PresContext());
MOZ_ASSERT(budget.mBudget >= cost);
budget.mBudget -= cost;
void
nsDisplayListBuilder::ClearWillChangeBudget()
{
mWillChangeBudgetSet.Clear();
mWillChangeBudget.Clear();
}

#ifdef MOZ_GFX_OPTIMIZE_MOBILE
Expand Down
19 changes: 14 additions & 5 deletions layout/painting/nsDisplayList.h
Original file line number Diff line number Diff line change
Expand Up @@ -1815,7 +1815,9 @@ class nsDisplayListBuilder
*/
bool IsInWillChangeBudget(nsIFrame* aFrame, const nsSize& aSize);

void ClearWillChangeBudget(nsIFrame* aFrame);
void RemoveFromWillChangeBudget(nsIFrame* aFrame);

void ClearWillChangeBudget();

void EnterSVGEffectsContents(nsDisplayList* aHoistedItemsStorage);
void ExitSVGEffectsContents();
Expand Down Expand Up @@ -2040,13 +2042,19 @@ class nsDisplayListBuilder

struct FrameWillChangeBudget
{
FrameWillChangeBudget(nsIFrame* aFrame, uint32_t aUsage)
: mFrame(aFrame)
FrameWillChangeBudget()
: mPresContext(nullptr)
, mUsage(0)
{
}

FrameWillChangeBudget(nsPresContext* aPresContext, uint32_t aUsage)
: mPresContext(aPresContext)
, mUsage(aUsage)
{
}

nsIFrame* mFrame;
nsPresContext* mPresContext;
uint32_t mUsage;
};

Expand Down Expand Up @@ -2084,7 +2092,8 @@ class nsDisplayListBuilder

// Any frame listed in this set is already counted in the budget
// and thus is in-budget.
nsDataHashtable<nsPtrHashKey<nsIFrame>, uint32_t> mWillChangeBudgetSet;
nsDataHashtable<nsPtrHashKey<nsIFrame>, FrameWillChangeBudget>
mWillChangeBudgetSet;

// Area of animated geometry root budget already allocated
uint32_t mUsedAGRBudget;
Expand Down

0 comments on commit 5107e2d

Please sign in to comment.