RESOLVED FIXED 190115
GC can collect JS wrappers of nodes in the mutation records waiting to be delivered
https://bugs.webkit.org/show_bug.cgi?id=190115
Summary GC can collect JS wrappers of nodes in the mutation records waiting to be del...
Ryosuke Niwa
Reported 2018-09-29 23:02:42 PDT
MutationObserver needs to keep JS wrappers of the elements in the pending mutation records alive.
Attachments
WIP (15.35 KB, patch)
2018-09-29 23:05 PDT, Ryosuke Niwa
no flags
Fixes the bug (17.16 KB, patch)
2018-10-01 16:06 PDT, Ryosuke Niwa
ggaren: review+
Radar WebKit Bug Importer
Comment 1 2018-09-29 23:03:29 PDT
Ryosuke Niwa
Comment 2 2018-09-29 23:05:08 PDT
Ryosuke Niwa
Comment 3 2018-09-29 23:06:01 PDT
This is basically the final patch (without a change log). I need to wait for the bug 190113 to be fixed first since this patch depends on it (WIP includes both the patch fo for this bug as well as the one for the bug 190113).
EWS Watchlist
Comment 4 2018-09-29 23:06:39 PDT
Attachment 351211 [details] did not pass style-queue: ERROR: Source/WebCore/dom/GCReachableRef.h:87: Should be indented on a separate line, with the colon or comma first on that line. [whitespace/indent] [4] ERROR: Source/WebCore/dom/GCReachableRef.h:90: Should be indented on a separate line, with the colon or comma first on that line. [whitespace/indent] [4] ERROR: Source/WebCore/dom/GCReachableRef.h:152: More than one command on the same line [whitespace/newline] [4] Total errors found: 3 in 7 files If any of these errors are false positives, please file a bug against check-webkit-style.
Ryosuke Niwa
Comment 5 2018-10-01 16:06:28 PDT
Created attachment 351318 [details] Fixes the bug
Ryosuke Niwa
Comment 6 2018-10-01 22:39:21 PDT
Ping reviewers.
Geoffrey Garen
Comment 7 2018-10-02 10:22:38 PDT
Comment on attachment 351318 [details] Fixes the bug View in context: https://bugs.webkit.org/attachment.cgi?id=351318&action=review > Source/WebCore/dom/MutationObserver.h:113 > Vector<Ref<MutationRecord>> m_records; > + HashSet<GCReachableRef<Node>> m_pendingTargets; It's kind of hard to follow the logic to keep m_pendingTargets in sync with m_records. Can we change MutationRecord to use GCReachableRef internally instead?
Ryosuke Niwa
Comment 8 2018-10-02 12:13:06 PDT
(In reply to Geoffrey Garen from comment #7) > Comment on attachment 351318 [details] > Fixes the bug > > View in context: > https://bugs.webkit.org/attachment.cgi?id=351318&action=review > > > Source/WebCore/dom/MutationObserver.h:113 > > Vector<Ref<MutationRecord>> m_records; > > + HashSet<GCReachableRef<Node>> m_pendingTargets; > > It's kind of hard to follow the logic to keep m_pendingTargets in sync with > m_records. Can we change MutationRecord to use GCReachableRef internally > instead? Can't do that because MutationRecord is exposed to JS. That would definitely create a ref / GC cycle.
Ryosuke Niwa
Comment 9 2018-10-02 12:20:41 PDT
Another alternative I've considered as adding a GCReachableRef to MutationRecord and clear it before we deliver. That has a downside of having to create a separate mechanism for when takeRecords() is called because we need to keep elements alive until those elements are accessed.
Geoffrey Garen
Comment 10 2018-10-02 15:15:22 PDT
Comment on attachment 351318 [details] Fixes the bug View in context: https://bugs.webkit.org/attachment.cgi?id=351318&action=review > Source/WebCore/dom/MutationObserver.cpp:118 > + // Don't clear m_pendingTargets here because we can correct JS wrappers > + // between the time takeRecords are called and nodes in records are accesssed. correct => collect are => is What will access the nodes referenced by takeRecords(), and what guarantees that those node wrappers will remain valid after MutationObserver::disconnect() or MutationObserver::deliver()? > Source/WebCore/dom/MutationObserver.cpp:188 > + m_pendingTargets.add(*mutation->target()); Is there value in putting everything in a hash table, or should we just use a vector? What's an example of a case where a mutation observer will enqueue the same target more than twice? > Source/WebCore/dom/MutationObserver.cpp:241 > + m_pendingTargets.clear(); // takeRecords() may have been called. You can get these return cases right automatically by swapping m_pendingTargets into an auto local variable (just like we do for m_records). > Source/WebCore/dom/MutationObserverRegistration.h:51 > + std::unique_ptr<HashSet<GCReachableRef<Node>>> clearTransientRegistrations(); Let's call this takeTransientRegistrations().
Geoffrey Garen
Comment 11 2018-10-02 15:29:54 PDT
Note: In a separate patch, we need MutationRecord to visitAdditionalChildren and addOpaqueRoot for the root of each node in the record (target, addedNodes, removedNodes, previousSibling, nextSibling).
Geoffrey Garen
Comment 12 2018-10-02 15:37:55 PDT
Comment on attachment 351318 [details] Fixes the bug View in context: https://bugs.webkit.org/attachment.cgi?id=351318&action=review r=me >> Source/WebCore/dom/MutationObserver.cpp:118 >> + // between the time takeRecords are called and nodes in records are accesssed. > > correct => collect > are => is > > What will access the nodes referenced by takeRecords(), and what guarantees that those node wrappers will remain valid after MutationObserver::disconnect() or MutationObserver::deliver()? I think a nice follow-up would be to change takeRecords() to be a custom binding, have it return a pair of the vector of mutation records and the hash table of pending targets, and have the caller retain the hash table of pending targets until conversion to a JS array is complete.
Ryosuke Niwa
Comment 13 2018-10-02 17:23:03 PDT
(In reply to Geoffrey Garen from comment #10) > Comment on attachment 351318 [details] > Fixes the bug > > View in context: > https://bugs.webkit.org/attachment.cgi?id=351318&action=review > > > Source/WebCore/dom/MutationObserver.cpp:118 > > + // Don't clear m_pendingTargets here because we can correct JS wrappers > > + // between the time takeRecords are called and nodes in records are accesssed. > > correct => collect > are => is Fixed. > What will access the nodes referenced by takeRecords(), and what guarantees > that those node wrappers will remain valid after > MutationObserver::disconnect() or MutationObserver::deliver()? There isn't as we discussed in person. Will fix in a follow up. > > Source/WebCore/dom/MutationObserver.cpp:188 > > + m_pendingTargets.add(*mutation->target()); > > Is there value in putting everything in a hash table, or should we just use > a vector? What's an example of a case where a mutation observer will enqueue > the same target more than twice? Yes, in the case multiple nodes are inserted into and removed from a single container node, the target node is that container node. > > Source/WebCore/dom/MutationObserver.cpp:241 > > + m_pendingTargets.clear(); // takeRecords() may have been called. > > You can get these return cases right automatically by swapping > m_pendingTargets into an auto local variable (just like we do for m_records). Sure, will do. > > Source/WebCore/dom/MutationObserverRegistration.h:51 > > + std::unique_ptr<HashSet<GCReachableRef<Node>>> clearTransientRegistrations(); > > Let's call this takeTransientRegistrations(). Good idea. Will do.
Ryosuke Niwa
Comment 14 2018-10-02 17:29:48 PDT
Matt Lewis
Comment 16 2018-10-03 11:02:53 PDT
Reverted r236781 for reason: The test added with this commit is timing out consistently. Committed r236799: <https://trac.webkit.org/changeset/236799>
Ryosuke Niwa
Comment 17 2018-10-03 11:08:09 PDT
We need to re-land this patch without a test then.
Ryosuke Niwa
Comment 18 2018-10-03 11:16:42 PDT
Ryan Haddad
Comment 19 2018-10-03 11:20:21 PDT
(In reply to Ryosuke Niwa from comment #17) > We need to re-land this patch without a test then. If this change is important enough to land without a test, it really seems like we should have a regression test for it.
Ryosuke Niwa
Comment 20 2018-10-03 11:22:57 PDT
(In reply to Ryan Haddad from comment #19) > (In reply to Ryosuke Niwa from comment #17) > > We need to re-land this patch without a test then. > If this change is important enough to land without a test, it really seems > like we should have a regression test for it. I don't see how we'd add a test given Matt rolled out the patch on the basis the the test would time out. I've already optimized the test to run as fast as possible. If this version of the test times out, I don't know what kind of a test would be possibly landed.
Alexey Proskuryakov
Comment 21 2018-10-03 13:45:02 PDT
Typically, tests for garbage collection effects run very fast, what is special about this one?
Ryosuke Niwa
Comment 22 2018-10-03 20:38:06 PDT
(In reply to Alexey Proskuryakov from comment #21) > Typically, tests for garbage collection effects run very fast, what is > special about this one? That's not my experience writing GC tests recently... I don't know what is different about this test but I had a difficulty reproducing the GC issue in the first place, and I spent an entire day optimizing the test itself. The plan now is to fix a related bug that MutationRecord doesn't keep elements / nodes and write a test for that since that would be the superset, and would (hopefully) require a single GC.
Ryosuke Niwa
Comment 23 2018-11-06 11:56:01 PST
New tests were added to cover some aspects of this bug in https://trac.webkit.org/changeset/236850
Note You need to log in before you can comment on or make changes to this bug.