Closed Bug 542361 Opened 15 years ago Closed 15 years ago

Fix and remove the GetUsedX assertions

Categories

(Core :: Layout, defect)

defect
Not set
normal

Tracking

()

RESOLVED FIXED
mozilla1.9.3a1

People

(Reporter: ventnor.bugzilla, Assigned: ventnor.bugzilla)

References

(Depends on 1 open bug)

Details

Attachments

(1 file, 6 obsolete files)

Attached patch Patch (obsolete) (deleted) β€” β€” Splinter Review
      No description provided.
Attachment #423670 - Flags: review?(roc)
This looks like the right idea.

I think the GetUsed* methods should check for NS_FRAME_FIRST_REFLOW and not NS_FRAME_IN_REFLOW --- in that case, the frame hasn't been reflowed yet and we should return 0,0,0,0 no matter what the style says (because frames are always completely empty until they get reflowed).

We can get rid of nsLayoutUtils::sDisableGetUsedXAssertions and everything that uses it.

I wonder if we should use a frame state bit (we have a free one!) to indicate that there is at least one usedBorderProperty/usedPaddingProperty/usedMarginProperty, so we can skip getting or deleting the properties in the common case when we know there can't be one.

I'm curious about what dbaron and bz think about this approach.
(In reply to comment #1)
> I wonder if we should use a frame state bit (we have a free one!) to indicate
> that there is at least one
> usedBorderProperty/usedPaddingProperty/usedMarginProperty, so we can skip
> getting or deleting the properties in the common case when we know there can't
> be one.

I don't know if this is necessary considering these operations are O(1) on a hashtable and are handled reasonably well in the most common cases (AFAICT).
I think the DidSetStyleContext code should only store properties if properties aren't already there.  Otherwise it could cause problems if we get a sequence of style context changes.  We still want the value to reflect the padding/border/margin used the last time we did a reflow.

Also, code inside NS_ASSERTION doesn't get run in non-debug builds, so you need to fix nsIFrame::GetUsedMargin and GetUsedPadding.

One other issue is that in theory nsCSSOffsetState can be constructed in cases where we don't actually do a reflow.  I'm not sure if that matters in practice, but you might want to restrict deleting (and also setting?) the properties to cases where the nsCSSOffsetState is an nsHTMLReflowState.  It probably requires a brief code audit to see if that matters (and, if it doesn't, code comments warning that the code assumes that it doesn't).


But the general approach looks good.
Attached patch Patch 2 (obsolete) (deleted) β€” β€” Splinter Review
This addresses the review comments. I am able to differentiate between nsCSSOffsetState and nsHTMLReflowState because they call InitOffsets in different locations.
Attachment #423670 - Attachment is obsolete: true
Attachment #423705 - Flags: review?(roc)
Attachment #423670 - Flags: review?(roc)
dbaron, feel free to chime in with more comments if any.
+    if (!GetProperty(nsGkAtoms::usedMarginProperty) &&
+        aOldStyleContext->GetStyleMargin()->GetMargin(oldValue)) {
+      if (!GetStyleMargin()->GetMargin(newValue) ||
+          oldValue != newValue) {
+        // margin most definitely changed, store it
+        SetProperty(nsGkAtoms::usedMarginProperty,
+                    new nsMargin(oldValue),
+                    nsCSSOffsetState::DestroyMarginFunc);

I think you should do the GetProperty check just before you do SetProperty, because it's more expensive than the other checks you're doing here.

To answer my question about whether a state bit is worth having, I think you should make an opt build and create a testcase that does a tight loop of border changes on a lot of small DIVs in the page, and then profile that testcase with your patch, and see if we spend significant time in GetProperty/SetProperty/DeleteProperty.
Attached patch Patch 3 (obsolete) (deleted) β€” β€” Splinter Review
We decided IRL that using our last state bit for this case is not worth it.
Other comments are addressed.
Attachment #423705 - Attachment is obsolete: true
Attachment #423734 - Flags: review?(roc)
Attachment #423705 - Flags: review?(roc)
More precisely, we can't tell if it's worth it, so on the principle of avoiding premature optimization, we're not going to do it (yet).
+  if (GetStateBits() & NS_FRAME_FIRST_REFLOW)
+    return margin;

I think you should not take this early exit if NS_FRAME_IN_REFLOW is set. Calling GetUsedMargin during the first reflow should be able to return something nonzero. Ditto for the other properties.
Do we need the calls on aOldStyleContext to be Gets, or can they be Peeks?  If we can avoid style data when it's not actually needed, that might be nice...
Attached patch Patch 4 (obsolete) (deleted) β€” β€” Splinter Review
Attachment #423734 - Attachment is obsolete: true
Attachment #423737 - Flags: review?(roc)
Attachment #423734 - Flags: review?(roc)
(In reply to comment #10)
> Do we need the calls on aOldStyleContext to be Gets, or can they be Peeks?  If
> we can avoid style data when it's not actually needed, that might be nice...

I don't think there are calls on style contexts that do peeks, are there? In any case, we still need the old style data to do the comparison.
nsIFrame says
  // Attention: the old style context is the one we're forgetting,
  // and hence possibly completely bogus for GetStyle* purposes.
  // Use PeekStyleData instead.
  virtual void DidSetStyleContext(nsStyleContext* aOldStyleContext) = 0;

OK, so we should use PeekStyleData. I presume that if that returns null, then the frame hasn't been reflowed with that old style context yet, so we shouldn't store the old value.
Hmm. I guess there's a hole in this approach, namely that we get the wrong results in a sequence like this:

-- reflow the frame
-- DidSetStyleContext(contextA)
-- get some style structs from contextA
-- DidSetStyleContext(contextB)
-- we save contextA's border/padding/margin as the usedBorder/Padding/Margin, even though we haven't actually done a reflow with contextA

To avoid this, I guess we need to use a frame state bit after all :-). We need to detect whether a reflow has happened since the last DidSetStyleContext. If it hasn't, and we get a DidSetStyleContext again, then we need to skip saving the border/padding/margin to properties.
(In reply to comment #14)
> Hmm. I guess there's a hole in this approach, namely that we get the wrong
> results in a sequence like this:
> 
> -- reflow the frame
> -- DidSetStyleContext(contextA)
> -- get some style structs from contextA
> -- DidSetStyleContext(contextB)
> -- we save contextA's border/padding/margin as the usedBorder/Padding/Margin,
> even though we haven't actually done a reflow with contextA
> 
> To avoid this, I guess we need to use a frame state bit after all :-). We need
> to detect whether a reflow has happened since the last DidSetStyleContext. If
> it hasn't, and we get a DidSetStyleContext again, then we need to skip saving
> the border/padding/margin to properties.

This was precisely what dbaron was concerned about, which was why I added the GetProperty checks. We don't overwrite the property if it exists in any case.
(In reply to comment #13)
> nsIFrame says
>   // Attention: the old style context is the one we're forgetting,
>   // and hence possibly completely bogus for GetStyle* purposes.
>   // Use PeekStyleData instead.
>   virtual void DidSetStyleContext(nsStyleContext* aOldStyleContext) = 0;
> 
> OK, so we should use PeekStyleData. I presume that if that returns null, then
> the frame hasn't been reflowed with that old style context yet, so we shouldn't
> store the old value.

There are quite a few existing lines in this function that ignore this rule...
Should I fix those as part of this patch?
No,

I guess you're right in comment #15.
The comment about GetStyleData returning incorrect data are no longer applicable (or might stop being so once dbaron lands some stuff; I'm not sure what the state of that code is).  The reason to peek instead of get would be purely as an optimization.
Attached patch Patch 5 (obsolete) (deleted) β€” β€” Splinter Review
Attachment #423737 - Attachment is obsolete: true
Attachment #423742 - Flags: review?(roc)
Attachment #423737 - Flags: review?(roc)
Comment on attachment 423742 [details] [diff] [review]
Patch 5

Looks good to me.
Attachment #423742 - Flags: review?(roc)
Attachment #423742 - Flags: review?(bzbarsky)
Attachment #423742 - Flags: review+
I'm not a good review choice here unless this can wait at least 1-2 weeks...
Comment on attachment 423742 [details] [diff] [review]
Patch 5

OK, let's try dbaron.
Attachment #423742 - Flags: review?(bzbarsky) → review?(dbaron)
For what it's worth, all the PeekStyleData gets you in this case is basically the NS_FRAME_FIRST_REFLOW optimization, but done right (so it doesn't affect within-reflow).  In other cases, the Peek->Get should chain across each style context change.  (If it didn't chain across consecutive style context changes, i.e., if there were cases where when Peek* was non-null on the old context we didn't call Get* on the new context, then using Peek would be an incorrect optimization.)
Comment on attachment 423742 [details] [diff] [review]
Patch 5

>+        // margin most definitely changed, store it

I'd drop this comment.

>+    NS_ASSERTION(hasMargin, "We should have a margin here!");

>+    NS_ASSERTION(hasPadding, "We should have padding here!");

Could you add back the "(out of memory?)" to both of these?


You need to fix a bunch of comments in nsIFrame.h.  For GetUsedMargin, 
GetUsedPadding, and GetUsedBorder, you should replace:
  Can only be called after Reflow for the frame has at least *started*.
with:
  Like GetRect(), returns the dimensions as of the most recent reflow.

r=dbaron with that
Attachment #423742 - Flags: review?(dbaron) → review+
This is really great, finally we can paint, handle events, etc in dirty frame trees without feeling guilty!
Attached patch For checkin (obsolete) (deleted) β€” β€” Splinter Review
Fixes dbaron's comments.
Attachment #423742 - Attachment is obsolete: true
Keywords: checkin-needed
http://hg.mozilla.org/mozilla-central/rev/8c2e7ae5cceb
Status: NEW → RESOLVED
Closed: 15 years ago
Keywords: checkin-needed
Resolution: --- → FIXED
Target Milestone: --- → mozilla1.9.3a1
Backed out because of this failure on Linux:

http://tinderbox.mozilla.org/showlog.cgi?log=Firefox/1265296841.1265299980.17408.gz

>TEST-UNEXPECTED-FAIL | chrome://mochikit/content/chrome/content/xul/content/test/test_bug398289.html | persistent attribute in tab box broken, expected:
>
>got:
>
Status: RESOLVED → REOPENED
Resolution: FIXED → ---
I bet the problem is that we don't use nsHTMLReflowState for XUL Layout, so you need an equivalent to the change in nsHTMLReflowState::InitConstraints for XUL.  

(That still wouldn't fix percentage padding and margin being broken in XUL, but that's already broken, I *think*.  Would probably be good to fix, though, but probably not in this bug.)
(And it might also be good to add a reftest that would more directly catch the regression.)
Attached patch New patch (deleted) β€” β€” Splinter Review
The only change is in nsBox at the bottom of the patch.
Attachment #423844 - Attachment is obsolete: true
Attachment #427659 - Flags: review?(roc)
http://hg.mozilla.org/mozilla-central/rev/11c0a0745d39
Status: REOPENED → RESOLVED
Closed: 15 years ago15 years ago
Resolution: --- → FIXED
Depends on: 550325
Depends on: 586628
Depends on: 603043
You need to log in before you can comment on or make changes to this bug.

Attachment

General

Created:
Updated:
Size: