Bug 173077

Summary: Image should clear its ImageObserver* when CachedImage releases the last reference to its RefCounted<ImageObserver>
Product: WebKit Reporter: Said Abou-Hallawa <sabouhallawa>
Component: ImagesAssignee: Said Abou-Hallawa <sabouhallawa>
Status: RESOLVED FIXED    
Severity: Normal CC: buildbot, cdumez, commit-queue, darin, dbates, japhet, rniwa, ryanhaddad, simon.fraser, thorton, webkit-bug-importer
Priority: P2 Keywords: InRadar
Version: WebKit Nightly Build   
Hardware: Unspecified   
OS: Unspecified   
Attachments:
Description Flags
Patch
buildbot: commit-queue-
Archive of layout-test-results from ews105 for mac-elcapitan-wk2
none
Patch
none
Patch
none
Patch none

Said Abou-Hallawa
Reported 2017-06-07 15:07:17 PDT
Here is how the objects Image, ImageObserver and CachedImage are related to each other. -- CachedImage has a RefPtr<Image> and RefPtr<ImageObserver> (or actually it is RefPtr<CachedImageObserver>). CachedImage is the creator and the destroyer of these two pointers. -- ImageObserver has a Vector<CachedImage*>. CachedImage maintains correct state for this vector by adding and removing itself upon creation and deletion. -- Image has a raw pointer of ImageObserver*. This pointer is only cleared temporarily but it is never cleared permanently. Two things to mention here: -- CachedImage can clone itself to another CachedImage through CachedImage::setBodyDataFrom(). The RefPtr<Image> and RefPtr<ImageObserver> will be set in the cloned CachedImage, which means their ref counts are incremented. -- Image object can outlive the CachedImage. In the CachedImage destructor, CachedImage::clearImage() is called to clear the references from the ImageObserver to the destructing CachedImage. If CachedImage releases the last reference to its ImageObserver, this ImageObserver will eventually be deleted. A crash can happen if the Image accesses its raw pointer after the ImageObserver is deleted. So ideally CachedImage::clearImage(): should set the m_imageObserver of the Image to nullptr by calling m_image->setImageObserver(nullptr). But since the CachedImage can be cloned, ImageObserver will not be deleted until the last referencing CachedImage calls CachedImage::clearImage(). So calling m_image->setImageObserver(nullptr) unconditionally from CachedImage::clearImage() is wrong as well. This is the history of CachedImage::clearImage(): Before https://trac.webkit.org/changeset/209914 inline void CachedImage::clearImage() { // If our Image has an observer, it's always us so we need to clear the back pointer // before dropping our reference. if (m_image) m_image->setImageObserver(nullptr); m_image = nullptr; } After https://trac.webkit.org/changeset/209914 inline void CachedImage::clearImage() { if (m_imageObserver) { m_imageObserver->remove(*this); m_imageObserver = nullptr; } m_image = nullptr; } After https://trac.webkit.org/changeset/216273 inline void CachedImage::clearImage() { if (m_imageObserver) { m_imageObserver->remove(*this); m_imageObserver = nullptr; } if (m_image) { m_image->setImageObserver(nullptr); m_image = nullptr; } } But this caused a crash in http/tests/security/cross-origin-cached-images-with-memory-pressure.html so it was rolled out in https://trac.webkit.org/changeset/216293. So we still have a bug for not clearing the raw pointer Image::m_imageObserver if CachedImage releases the last reference to its CachedImage::m_imageObserver and Image accesses the deleted ImageObserver through the raw pointer Image::m_imageObserver.
Attachments
Patch (2.71 KB, patch)
2017-06-07 15:13 PDT, Said Abou-Hallawa
buildbot: commit-queue-
Archive of layout-test-results from ews105 for mac-elcapitan-wk2 (1.32 MB, application/zip)
2017-06-07 16:06 PDT, Build Bot
no flags
Patch (4.39 KB, patch)
2017-06-08 10:01 PDT, Said Abou-Hallawa
no flags
Patch (4.58 KB, patch)
2017-06-09 10:20 PDT, Said Abou-Hallawa
no flags
Patch (4.43 KB, patch)
2017-06-09 19:10 PDT, Said Abou-Hallawa
no flags
Said Abou-Hallawa
Comment 1 2017-06-07 15:13:30 PDT
Said Abou-Hallawa
Comment 2 2017-06-07 15:16:07 PDT
Build Bot
Comment 3 2017-06-07 16:06:09 PDT
Comment on attachment 312239 [details] Patch Attachment 312239 [details] did not pass mac-wk2-ews (mac-wk2): Output: http://webkit-queues.webkit.org/results/3890363 New failing tests: imported/w3c/web-platform-tests/fetch/api/cors/cors-origin.any.html
Build Bot
Comment 4 2017-06-07 16:06:10 PDT
Created attachment 312249 [details] Archive of layout-test-results from ews105 for mac-elcapitan-wk2 The attached test failures were seen while running run-webkit-tests on the mac-wk2-ews. Bot: ews105 Port: mac-elcapitan-wk2 Platform: Mac OS X 10.11.6
Said Abou-Hallawa
Comment 5 2017-06-08 10:01:49 PDT
Simon Fraser (smfr)
Comment 6 2017-06-08 15:29:16 PDT
Comment on attachment 312315 [details] Patch View in context: https://bugs.webkit.org/attachment.cgi?id=312315&action=review > Source/WebCore/loader/cache/CachedImage.cpp:375 > + if (!m_imageObserver->cachedImages().size()) { isEmpty()
Simon Fraser (smfr)
Comment 7 2017-06-08 15:29:29 PDT
Is this testable?
Said Abou-Hallawa
Comment 8 2017-06-09 10:20:10 PDT
WebKit Commit Bot
Comment 9 2017-06-09 11:38:43 PDT
Comment on attachment 312450 [details] Patch Clearing flags on attachment: 312450 Committed r218003: <http://trac.webkit.org/changeset/218003>
WebKit Commit Bot
Comment 10 2017-06-09 11:38:45 PDT
All reviewed patches have been landed. Closing bug.
Ryan Haddad
Comment 11 2017-06-09 15:09:28 PDT
(In reply to WebKit Commit Bot from comment #9) > Comment on attachment 312450 [details] > Patch > > Clearing flags on attachment: 312450 > > Committed r218003: <http://trac.webkit.org/changeset/218003> I'm seeing a few LayoutTests hit one of the asserts added with this change: http/tests/security/canvas-remote-read-remote-video-allowed-with-credentials.html imported/w3c/web-platform-tests/html/semantics/embedded-content/the-canvas-element/size.attributes.removed.html https://build.webkit.org/results/Apple%20El%20Capitan%20Debug%20WK2%20(Tests)/r218012%20(1627)/results.html
Ryan Haddad
Comment 12 2017-06-09 17:38:10 PDT
Reverted r218003 for reason: This change caused assertion failures in existing LayoutTests. Committed r218031: <http://trac.webkit.org/changeset/218031>
Said Abou-Hallawa
Comment 13 2017-06-09 18:08:57 PDT
(In reply to Ryan Haddad from comment #11) > (In reply to WebKit Commit Bot from comment #9) > > Comment on attachment 312450 [details] > > Patch > > > > Clearing flags on attachment: 312450 > > > > Committed r218003: <http://trac.webkit.org/changeset/218003> > > I'm seeing a few LayoutTests hit one of the asserts added with this change: > > http/tests/security/canvas-remote-read-remote-video-allowed-with-credentials. > html > imported/w3c/web-platform-tests/html/semantics/embedded-content/the-canvas- > element/size.attributes.removed.html > > https://build.webkit.org/results/ > Apple%20El%20Capitan%20Debug%20WK2%20(Tests)/r218012%20(1627)/results.html Yes. There is no guarantee that the Image is referenced only by CachedImage. One of the CachedImage constructors takes an argument of Image* which means the Image object is created and referenced outside the life cycle of CachedImage. When storing it in CachedImage::m_image, the refcount is incremented. So the Image object can outlive CachedImage. The only thing we can guarantee, is CachedImageObserver can’t outlive CachedImage. So the fix is to remove ASSERT(m_image->hasOneRef()) from CachedImage::clearImage().
Said Abou-Hallawa
Comment 14 2017-06-09 19:10:57 PDT
WebKit Commit Bot
Comment 15 2017-06-09 19:51:08 PDT
Comment on attachment 312530 [details] Patch Clearing flags on attachment: 312530 Committed r218038: <http://trac.webkit.org/changeset/218038>
WebKit Commit Bot
Comment 16 2017-06-09 19:51:10 PDT
All reviewed patches have been landed. Closing bug.
Note You need to log in before you can comment on or make changes to this bug.