WebKit Bugzilla
New
Browse
Log In
×
Sign in with GitHub
or
Remember my login
Create Account
·
Forgot Password
Forgotten password account recovery
RESOLVED FIXED
171416
RejectedPromiseTracker should produce better callstacks and avoid capturing callstacks unless there is a debugger/inspector
https://bugs.webkit.org/show_bug.cgi?id=171416
Summary
RejectedPromiseTracker should produce better callstacks and avoid capturing c...
Joseph Pecoraro
Reported
2017-04-27 20:10:59 PDT
RejectedPromiseTracker should avoid capturing callstacks unless there is a debugger/inspector The call stack captured by RejectedPromiseTracker is only used when logging to the console. Since most users will not open the debugger/inspector this wastes time to compute and memory to retain the stack trace.
Attachments
[PATCH] Proposed Fix
(15.54 KB, patch)
2017-04-28 16:52 PDT
,
Joseph Pecoraro
buildbot
: commit-queue-
Details
Formatted Diff
Diff
Archive of layout-test-results from ews102 for mac-elcapitan
(1.03 MB, application/zip)
2017-04-28 18:08 PDT
,
Build Bot
no flags
Details
Archive of layout-test-results from ews105 for mac-elcapitan-wk2
(1.32 MB, application/zip)
2017-04-28 18:09 PDT
,
Build Bot
no flags
Details
Archive of layout-test-results from ews113 for mac-elcapitan
(2.02 MB, application/zip)
2017-04-28 18:31 PDT
,
Build Bot
no flags
Details
[PATCH] Proposed Fix
(59.23 KB, patch)
2017-04-28 18:33 PDT
,
Joseph Pecoraro
no flags
Details
Formatted Diff
Diff
[PATCH] Proposed Fix
(59.24 KB, patch)
2017-04-28 18:48 PDT
,
Joseph Pecoraro
saam
: review+
Details
Formatted Diff
Diff
Show Obsolete
(5)
View All
Add attachment
proposed patch, testcase, etc.
Joseph Pecoraro
Comment 1
2017-04-28 13:22:53 PDT
I think we can just not do the work if the !globalObject->hasDebugger().
Joseph Pecoraro
Comment 2
2017-04-28 15:05:42 PDT
So this is interesting. I think we will want a call stack when a runtime error occurs. Even if web inspector is not open. This is the way we behave for uncaught exceptions and it is very useful for developers to open the tools, see the issue, and see the backtrace. The engine already generated a call stack so we should be able to use it. However, for promise rejections we have this weird situation where the JSC::Exception is gone and we just have the JSC::ErrorInstance: (lldb) btjs * thread #1: tid = 0x22b14ad, 0x0000000106ec3a92, queue = 'com.apple.main-thread, stop reason = step over frame #0: 0x0000000106ec3a92 WebCore`WebCore::createScriptCallStackFromReason(state=0x00007fff5f0ff770, reason=JSValue @ 0x00007fff5f0ff598) at RejectedPromiseTracker.cpp:112 frame #1: 0x0000000106ec3994 WebCore`WebCore::RejectedPromiseTracker::promiseRejected(this=0x00007fbb9c521550, state=0x00007fff5f0ff770, globalObject=0x000000011fbd6040, promise=0x000000011fbdbd60) at RejectedPromiseTracker.cpp:123 frame #2: 0x00000001064a3566 WebCore`WebCore::JSDOMWindowBase::promiseRejectionTracker(jsGlobalObject=0x000000011fbd6040, exec=0x00007fff5f0ff770, promise=0x000000011fbdbd60, operation=Reject) at JSDOMWindowBase.cpp:415 frame #3: 0x0000000111f5d018 JavaScriptCore`JSC::globalFuncHostPromiseRejectionTracker(exec=0x00007fff5f0ff770) at JSGlobalObjectFunctions.cpp:758 frame #4: 0x00005a1764a01028 0x1120b8725 frame #5: 0x00000001120b8725 rejectPromise#DJDkwU [LLInt](Cell[Window ID: 518]: 0x11fbd6040, Cell[Promise ID: 486]: 0x11fbdbd60, Cell[Error ID: 472]: 0x11fbe6100) frame #6: 0x00000001120b106e JavaScriptCore`vmEntryToJavaScript at LowLevelInterpreter64.asm:254 frame #7: 0x0000000111e8583e JavaScriptCore`JSC::JITCode::execute(this=0x000000011a0e8ac8, vm=0x000000011f800000, protoCallFrame=0x00007fff5f0ff9d8) at JITCode.cpp:81 ... (lldb) p *reason.asCell() (JSC::JSCell) $12 = { m_structureID = 472 m_indexingTypeAndMisc = '\0' m_type = ErrorInstanceType m_flags = '\0' m_cellState = DefinitelyWhite } Basically when the page threw a runtime exception in a then handler, a built-in try/catch catches it and propagates the error value: @globalPrivate function promiseReactionJob(state, reaction, argument) { ... try { result = handler(argument); } catch (error) { return promiseCapability.@reject.@call(@undefined, error); } ... } @reject goes to @rejectPromise: @globalPrivate function createResolvingFunctions(promise) { ... var reject = function (reason) { if (alreadyResolved) return @undefined; alreadyResolved = true; return @rejectPromise(promise, reason); }; return { @resolve: resolve, @reject: reject }; } Up to the RejectionTracker: @globalPrivate function rejectPromise(promise, reason) { ... promise.@promiseResult = reason; ... if (!promise.@promiseIsHandled) @hostPromiseRejectionTracker(promise, @promiseRejectionReject); ... } --- So the RejectedPromiseTracker just has the Error instance, which has an e.stack string property but we lost the nicer call stack with StackFrames which was used to generate the e.stack.
Joseph Pecoraro
Comment 3
2017-04-28 15:45:00 PDT
I can use vm.lastException() with some verification (the exception's error value is the reason value). This actually gives the correct stack from when the exception happened inside the promise reaction, and not a muffled stack that was since the try/catch in built-in code. So this is a win-win: It use the pre-existing Exception stack and has more correct information when logging to the Inspector console. I'm going to write some inspector tests for this now.
Radar WebKit Bug Importer
Comment 4
2017-04-28 16:41:57 PDT
<
rdar://problem/31899425
>
Joseph Pecoraro
Comment 5
2017-04-28 16:52:44 PDT
Created
attachment 308616
[details]
[PATCH] Proposed Fix
Build Bot
Comment 6
2017-04-28 18:08:10 PDT
Comment on
attachment 308616
[details]
[PATCH] Proposed Fix
Attachment 308616
[details]
did not pass mac-ews (mac): Output:
http://webkit-queues.webkit.org/results/3629907
New failing tests: js/promises-tests/promises-tests-2-2-6.html imported/w3c/web-platform-tests/streams/piping/close-propagation-forward.html http/tests/security/video-cross-origin-caching.html imported/w3c/web-platform-tests/streams/piping/error-propagation-forward.html streams/reference-implementation/readable-stream-templated.html js/dom/unhandled-promise-rejection-console-report.html inspector/debugger/break-on-exception-throw-in-promise.html js/dom/unhandled-promise-rejection-bindings-type-error.html streams/reference-implementation/writable-stream-abort.html imported/w3c/web-platform-tests/streams/piping/general.html imported/w3c/web-platform-tests/WebCryptoAPI/derive_bits_keys/test_ecdh_bits.html imported/w3c/web-platform-tests/html/semantics/embedded-content/media-elements/paused_true_during_pause.html inspector/debugger/break-on-uncaught-exception-throw-in-promise.html imported/w3c/web-platform-tests/WebCryptoAPI/derive_bits_keys/test_ecdh_keys.html streams/reference-implementation/writable-stream.html imported/w3c/web-platform-tests/streams/piping/error-propagation-backward.html imported/w3c/web-platform-tests/streams/piping/flow-control.html imported/w3c/web-platform-tests/html/semantics/embedded-content/media-elements/event_play_noautoplay.html streams/reference-implementation/pipe-to-options.html imported/w3c/web-platform-tests/html/semantics/embedded-content/media-elements/event_pause_noautoplay.html
Build Bot
Comment 7
2017-04-28 18:08:12 PDT
Created
attachment 308635
[details]
Archive of layout-test-results from ews102 for mac-elcapitan The attached test failures were seen while running run-webkit-tests on the mac-ews. Bot: ews102 Port: mac-elcapitan Platform: Mac OS X 10.11.6
Build Bot
Comment 8
2017-04-28 18:09:33 PDT
Comment on
attachment 308616
[details]
[PATCH] Proposed Fix
Attachment 308616
[details]
did not pass mac-wk2-ews (mac-wk2): Output:
http://webkit-queues.webkit.org/results/3629890
New failing tests: streams/reference-implementation/readable-stream-templated.html inspector/debugger/break-on-uncaught-exception-throw-in-promise.html http/tests/security/video-cross-origin-caching.html imported/w3c/web-platform-tests/streams/piping/close-propagation-forward.html webrtc/libwebrtc/release-while-creating-offer.html imported/w3c/web-platform-tests/streams/piping/error-propagation-forward.html imported/w3c/web-platform-tests/streams/piping/error-propagation-backward.html fast/mediastream/MediaStream-MediaElement-setObject-null.html js/promises-tests/promises-tests-2-2-6.html imported/w3c/web-platform-tests/html/semantics/embedded-content/media-elements/event_pause_noautoplay.html imported/w3c/web-platform-tests/streams/piping/general.html inspector/debugger/break-on-exception-throw-in-promise.html js/dom/unhandled-promise-rejection-bindings-type-error.html imported/w3c/web-platform-tests/WebCryptoAPI/derive_bits_keys/test_ecdh_bits.html imported/w3c/web-platform-tests/html/semantics/embedded-content/media-elements/paused_true_during_pause.html streams/reference-implementation/writable-stream.html webrtc/libwebrtc/release-while-setting-local-description.html streams/reference-implementation/pipe-to-options.html js/dom/unhandled-promise-rejection-console-report.html streams/reference-implementation/writable-stream-abort.html imported/w3c/web-platform-tests/WebCryptoAPI/derive_bits_keys/test_ecdh_keys.html imported/w3c/web-platform-tests/streams/piping/flow-control.html imported/w3c/web-platform-tests/html/semantics/embedded-content/media-elements/event_play_noautoplay.html
Build Bot
Comment 9
2017-04-28 18:09:34 PDT
Created
attachment 308637
[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
Build Bot
Comment 10
2017-04-28 18:31:05 PDT
Comment on
attachment 308616
[details]
[PATCH] Proposed Fix
Attachment 308616
[details]
did not pass mac-debug-ews (mac): Output:
http://webkit-queues.webkit.org/results/3629993
New failing tests: js/promises-tests/promises-tests-2-2-6.html imported/w3c/web-platform-tests/streams/piping/close-propagation-forward.html imported/w3c/web-platform-tests/streams/piping/error-propagation-forward.html streams/reference-implementation/readable-stream-templated.html js/dom/unhandled-promise-rejection-console-report.html inspector/debugger/break-on-exception-throw-in-promise.html js/dom/unhandled-promise-rejection-bindings-type-error.html streams/reference-implementation/writable-stream-abort.html imported/w3c/web-platform-tests/streams/piping/general.html imported/w3c/web-platform-tests/WebCryptoAPI/derive_bits_keys/test_ecdh_bits.html imported/w3c/web-platform-tests/html/semantics/embedded-content/media-elements/paused_true_during_pause.html inspector/debugger/break-on-uncaught-exception-throw-in-promise.html imported/w3c/web-platform-tests/html/semantics/embedded-content/media-elements/event_pause_noautoplay.html imported/w3c/web-platform-tests/WebCryptoAPI/derive_bits_keys/test_ecdh_keys.html streams/reference-implementation/writable-stream.html imported/w3c/web-platform-tests/streams/piping/error-propagation-backward.html imported/w3c/web-platform-tests/streams/piping/flow-control.html imported/w3c/web-platform-tests/html/semantics/embedded-content/media-elements/event_play_noautoplay.html streams/reference-implementation/pipe-to-options.html http/tests/security/video-cross-origin-caching.html
Build Bot
Comment 11
2017-04-28 18:31:07 PDT
Created
attachment 308640
[details]
Archive of layout-test-results from ews113 for mac-elcapitan The attached test failures were seen while running run-webkit-tests on the mac-debug-ews. Bot: ews113 Port: mac-elcapitan Platform: Mac OS X 10.11.6
Joseph Pecoraro
Comment 12
2017-04-28 18:33:21 PDT
Created
attachment 308641
[details]
[PATCH] Proposed Fix + Inspector folks since I include an inspector test to verify the stack.
Build Bot
Comment 13
2017-04-28 18:35:55 PDT
Attachment 308641
[details]
did not pass style-queue: ERROR: LayoutTests/ChangeLog:19: Line contains tab character. [whitespace/tab] [5] ERROR: LayoutTests/ChangeLog:20: Line contains tab character. [whitespace/tab] [5] Total errors found: 2 in 32 files If any of these errors are false positives, please file a bug against check-webkit-style.
Joseph Pecoraro
Comment 14
2017-04-28 18:48:10 PDT
Created
attachment 308645
[details]
[PATCH] Proposed Fix
Blaze Burg
Comment 15
2017-05-01 12:00:45 PDT
Comment on
attachment 308645
[details]
[PATCH] Proposed Fix View in context:
https://bugs.webkit.org/attachment.cgi?id=308645&action=review
New test looks good to me.
> LayoutTests/inspector/console/message-stack-trace.html:73 > + untilMessageMatches(condition).then((message) => {
Very nice.
> LayoutTests/inspector/console/resources/log-console-stack-trace.js:12 > + let code = callFrame.nativeCode ? "N" : (callFrame.programCode ? "P" : "F");
I don't see the [P] case used anywhere. Is that expected? Would we ever see it if we remove test harness call frames?
Joseph Pecoraro
Comment 16
2017-05-01 14:07:10 PDT
> I don't see the [P] case used anywhere. Is that expected? Would we ever see > it if we remove test harness call frames?
P is Program code. Here we are dumping stacks that resulted from a `setTimeout(func)` so we don't expect to see program code. If callstacks included async code then we would see the Program code that triggered the setTimeout. Another case would be doing a console.error/trace from global code.
Saam Barati
Comment 17
2017-05-01 15:02:40 PDT
Comment on
attachment 308645
[details]
[PATCH] Proposed Fix View in context:
https://bugs.webkit.org/attachment.cgi?id=308645&action=review
> Source/WebCore/ChangeLog:22 > + new stack since the try/catch. Instead, we want the stack from before
did you mean to say "since" here?
> Source/WebCore/ChangeLog:28 > + Switch to a pointer since the staack could be empty.
staack =>stack
> Source/WebCore/dom/RejectedPromiseTracker.cpp:114 > + if (exception->value() == reason) > + return createScriptCallStackFromException(&state, exception, ScriptCallStack::maxCallStackSizeToCapture);
Don't we only want to do this if the debugger is open?
Joseph Pecoraro
Comment 18
2017-05-01 15:16:13 PDT
> > Source/WebCore/dom/RejectedPromiseTracker.cpp:114 > > + if (exception->value() == reason) > > + return createScriptCallStackFromException(&state, exception, ScriptCallStack::maxCallStackSizeToCapture); > > Don't we only want to do this if the debugger is open?
I'm intentionally doing this for Exceptions in Unhandled Rejections to match our behavior for Uncaught Exceptions when a debugger is not open. In these cases we have an existing JSC::Exception object that already walked the stack, so this should be a smaller operation in comparison. It is also a huge time saver for developers to open Web Inspector and see the full stack for runtime exceptions and not have to try to reproduce with Web Inspector open. In cases where a Promise is explicitly rejected (someone calling `reject`) we don't capture the stack unless the debugger is open.
Saam Barati
Comment 19
2017-05-01 15:24:32 PDT
(In reply to Joseph Pecoraro from
comment #18
)
> > > Source/WebCore/dom/RejectedPromiseTracker.cpp:114 > > > + if (exception->value() == reason) > > > + return createScriptCallStackFromException(&state, exception, ScriptCallStack::maxCallStackSizeToCapture); > > > > Don't we only want to do this if the debugger is open? > > I'm intentionally doing this for Exceptions in Unhandled Rejections to match > our behavior for Uncaught Exceptions when a debugger is not open. In these > cases we have an existing JSC::Exception object that already walked the > stack, so this should be a smaller operation in comparison. It is also a > huge time saver for developers to open Web Inspector and see the full stack > for runtime exceptions and not have to try to reproduce with Web Inspector > open. > > In cases where a Promise is explicitly rejected (someone calling `reject`) > we don't capture the stack unless the debugger is open.
I think this might be worth it, but I looked at what this code does, and I'm not certain it will be less time than taking the stack trace itself. It looks like at the very least we binary search for line/column info at each frame.
Joseph Pecoraro
Comment 20
2017-05-01 15:25:50 PDT
<
https://trac.webkit.org/changeset/216035/webkit
>
Joseph Pecoraro
Comment 21
2017-05-01 15:33:08 PDT
> I think this might be worth it, but I looked at what this code does, and I'm > not certain it will be less time than taking the stack trace itself. It > looks like at the very least we binary search for line/column info at each > frame.
You are correct, I overlooked that part. I still feel it is justified since exceptions should be exceptional ("rare") cases. If this path gets hit for more than just exception cases we should reconsider.
Note
You need to
log in
before you can comment on or make changes to this bug.
Top of Page
Format For Printing
XML
Clone This Bug