RESOLVED FIXED 69569
Remove colSpan / rowSpan caching from RenderTableCell
https://bugs.webkit.org/show_bug.cgi?id=69569
Summary Remove colSpan / rowSpan caching from RenderTableCell
Julien Chaffraix
Reported 2011-10-06 15:13:52 PDT
Currently those 2 fields are used during layout and to check that colSpan or rowSpan have changed in updateElement. However most of this use does not mandate having 2 integers stored per cells. Especially since HTMLTableElement already stores them.
Attachments
Proposed change: Removed more caching of colSpan / rowSpan. (8.45 KB, patch)
2011-10-06 15:37 PDT, Julien Chaffraix
no flags
Change 2: Removed the bogus span changed logic. (8.30 KB, patch)
2011-10-15 10:11 PDT, Julien Chaffraix
no flags
Patch for landing (8.93 KB, patch)
2011-10-17 09:52 PDT, Julien Chaffraix
jchaffraix: commit-queue+
Julien Chaffraix
Comment 1 2011-10-06 15:37:56 PDT
Created attachment 110047 [details] Proposed change: Removed more caching of colSpan / rowSpan.
Darin Adler
Comment 2 2011-10-06 17:18:57 PDT
Comment on attachment 110047 [details] Proposed change: Removed more caching of colSpan / rowSpan. View in context: https://bugs.webkit.org/attachment.cgi?id=110047&action=review > Source/WebCore/html/HTMLTableCellElement.cpp:97 > + int oldRowSpan = max(1, attr->value().toInt()); Is the value in Attribute* really the old value? Are you sure?
Darin Adler
Comment 3 2011-10-06 17:19:22 PDT
Comment on attachment 110047 [details] Proposed change: Removed more caching of colSpan / rowSpan. How did you measure the performance impact of this change?
Julien Chaffraix
Comment 4 2011-10-06 18:23:09 PDT
> > Source/WebCore/html/HTMLTableCellElement.cpp:97 > > + int oldRowSpan = max(1, attr->value().toInt()); > > Is the value in Attribute* really the old value? Are you sure? I double checked and you are right. I was confused by Element::setAttribute who calls the attribute |old| but then sets the new value on it. The check is always false (I am waiting on the cr-linux EWS if it does not complain I will open a bug about colSpan / rowSpan mutation needing some coverage). > How did you measure the performance impact of this change? Not that it matters but I measured using PageCycler to see the impact on real pages. I did not want to go with a micro-benchmark if that's the underlying question.
WebKit Review Bot
Comment 5 2011-10-06 21:11:50 PDT
Comment on attachment 110047 [details] Proposed change: Removed more caching of colSpan / rowSpan. Attachment 110047 [details] did not pass chromium-ews (chromium-xvfb): Output: http://queues.webkit.org/results/9995104 New failing tests: fast/dom/HTMLTableElement/colSpan.html tables/mozilla/bugs/bug22246-2a.html tables/mozilla/bugs/bug22246-3a.html tables/mozilla/bugs/bug22246-3.html tables/mozilla/bugs/bug22246-2.html
Julien Chaffraix
Comment 6 2011-10-07 13:41:10 PDT
Comment on attachment 110047 [details] Proposed change: Removed more caching of colSpan / rowSpan. As pointed out by Darin, this patch is horribly wrong. I will see if some of the changes can be extracted - like removing the virtual dispatch - but I doubt we can easily save the memory without an unneeded overhaul of the way we handle attribute change.
Julien Chaffraix
Comment 7 2011-10-13 10:31:50 PDT
I looked into the performance degradation linked to this patch when getting rowSpan or colSpan - using a micro-benchmark - and couldn't get a significant measure (it looked around 1.5% degradation but that fell within the statistical noise). The patch saves 4% memory (8 bytes out of 208 bytes for a RenderTableCell on my machine) and could save 4% more if we are manage to pack 2 bits. However for the change to work, we would need to remove the old vs new value check. Looking at Chrome's benchmark pages (Alexia US among them), rowSpan / colSpan don't seem to be changed or queried through JS and are set once inside HTML. IMHO it would be fine to go ahead and save the memory but I would welcome comments from other developers.
Darin Adler
Comment 8 2011-10-14 13:10:17 PDT
Seems OK to me to remove the caching. Seems we should have some way in the attribute change mechanism to consider both the old and new values of an attribute when responding to a change without having to pay memory to store the old value.
Julien Chaffraix
Comment 9 2011-10-15 09:32:38 PDT
(In reply to comment #8) > Seems OK to me to remove the caching. Seems we should have some way in the attribute change mechanism to consider both the old and new values of an attribute when responding to a change without having to pay memory to store the old value. That's definitely a good idea. I looked at some of the classes in html/ to see how often we cache the attribute: in some case, we store the attribute to avoid reparsing but most of the cases are just to detect attribute change (and also caching). Filed bug 70177 to cover adding a way of getting the old value.
Julien Chaffraix
Comment 10 2011-10-15 10:11:06 PDT
Created attachment 111138 [details] Change 2: Removed the bogus span changed logic.
Darin Adler
Comment 11 2011-10-15 13:08:21 PDT
Comment on attachment 111138 [details] Change 2: Removed the bogus span changed logic. View in context: https://bugs.webkit.org/attachment.cgi?id=111138&action=review > Source/WebCore/ChangeLog:14 > + on PageCycler Alexia-US. I think it’s Alexa, not Alexia. > Source/WebCore/html/HTMLTableCellElement.h:29 > +#include "HTMLNames.h" When I did this recently in some other classes, I avoided putting HTMLNames.h into the header by putting the assertions into the .cpp file and having only the NDEBUG version be inlined. > Source/WebCore/rendering/RenderTableCell.cpp:52 > + m_hasAssociatedTableCellElement = node && (node->hasTagName(tdTag) || node->hasTagName(thTag)); Seems this should be done with initialization, not assignment. > Source/WebCore/rendering/RenderTableCell.cpp:84 > + ASSERT(node() && (node()->hasTagName(tdTag) || node()->hasTagName(thTag))); Normally we do not use && in ASSERT. Instead we do two separate assertions. That allows us to see which has failed. > Source/WebCore/rendering/RenderTableCell.cpp:87 > + if (parent() && section()) I’m not sure the null check of parent() is helpful, even though the old code had it. > Source/WebCore/rendering/RenderTableCell.h:153 > + // FIXME: It would be nice to pack those 2 bits into some of the previous fields. Should be "these", not "those".
Julien Chaffraix
Comment 12 2011-10-17 09:30:49 PDT
Comment on attachment 111138 [details] Change 2: Removed the bogus span changed logic. View in context: https://bugs.webkit.org/attachment.cgi?id=111138&action=review >> Source/WebCore/rendering/RenderTableCell.cpp:87 >> + if (parent() && section()) > > I’m not sure the null check of parent() is helpful, even though the old code had it. I am pretty sure it is needed. Calling section() assumes the following hierarchy: RenderTableSection -> RenderTableRow -> RenderTableCell. However nothing guarantees that it is respected and section() does not check for parent() so you could end up dereferencing NULL.
Julien Chaffraix
Comment 13 2011-10-17 09:52:53 PDT
Created attachment 111277 [details] Patch for landing
Julien Chaffraix
Comment 14 2011-10-17 18:18:26 PDT
Note You need to log in before you can comment on or make changes to this bug.