RESOLVED FIXED 202711
Post increment/decrement should only call ToNumber once
https://bugs.webkit.org/show_bug.cgi?id=202711
Summary Post increment/decrement should only call ToNumber once
Robin Morisset
Reported 2019-10-08 16:05:34 PDT
Currently they call it twice, see the following test case: ``` var o = {}; var counter = 0; o.valueOf = () => {counter ++; return 42}; o++; if (counter != 1) throw "valueOf was executed " + counter + " times during a post-increment instead of once!"; ``` As far as I can tell this is not spec-compliant: 12.4.4 Postfix Increment Operator requires a single call to ToNumeric. Then there is a single chain of ToNumeric -> ToNumber -> ToPrimitive -> OrdinaryToPrimitive -> valueOf with no reason for the call to be duplicated anywhere. The problem appears to be from: ``` static RegisterID* emitPostIncOrDec(BytecodeGenerator& generator, RegisterID* dst, RegisterID* srcDst, Operator oper) { if (dst == srcDst) return generator.emitToNumber(generator.finalDestination(dst), srcDst); RefPtr<RegisterID> tmp = generator.emitToNumber(generator.tempDestination(dst), srcDst); emitIncOrDec(generator, srcDst, oper); return generator.move(dst, tmp.get()); } ``` which uses an emitToNumber, but then does an emitIncOrDec on the original value, which itself can lead to a slow path that does ToNumber anew.
Attachments
Patch (3.27 KB, patch)
2019-10-09 10:51 PDT, Robin Morisset
mark.lam: review+
Archive of layout-test-results from ews213 for win-future (13.65 MB, application/zip)
2019-10-10 00:33 PDT, EWS Watchlist
no flags
Patch (3.54 KB, patch)
2019-10-14 18:32 PDT, Robin Morisset
ews-watchlist: commit-queue-
Patch (3.54 KB, patch)
2019-10-15 11:12 PDT, Robin Morisset
no flags
Robin Morisset
Comment 1 2019-10-08 16:49:09 PDT
Things are a bit more subtle than I expected: wrapping the testcase above in a function and calling that function suppresses the bug! It comes back if "o++" is replaced by "var x = o++" so I suspect it is caused by the code: ``` RegisterID* PostfixNode::emitResolve(BytecodeGenerator& generator, RegisterID* dst) { if (dst == generator.ignoredResult()) return PrefixNode::emitResolve(generator, dst); ... RefPtr<RegisterID> oldValue = emitPostIncOrDec(generator, generator.finalDestination(dst), value.get(), m_operator); ... } ``` also in NodesCodegen.cpp. It visibly replaces "o++;" by "++o;" when the result is not stored anywhere. The mystery is why it only occurs when wrapped in a function, and not at the top-level.
Robin Morisset
Comment 2 2019-10-09 10:51:35 PDT
Mark Lam
Comment 3 2019-10-09 10:58:37 PDT
Comment on attachment 380545 [details] Patch r=me
EWS Watchlist
Comment 4 2019-10-10 00:33:17 PDT
Comment on attachment 380545 [details] Patch Attachment 380545 [details] did not pass win-ews (win): Output: https://webkit-queues.webkit.org/results/13113521 Number of test failures exceeded the failure limit.
EWS Watchlist
Comment 5 2019-10-10 00:33:19 PDT
Created attachment 380608 [details] Archive of layout-test-results from ews213 for win-future The attached test failures were seen while running run-webkit-tests on the win-ews. Bot: ews213 Port: win-future Platform: CYGWIN_NT-10.0-17763-3.0.5-338.x86_64-x86_64-64bit
Keith Miller
Comment 6 2019-10-10 17:47:00 PDT
Seems like this patch has broken jsc EWS... :(
Alexey Proskuryakov
Comment 7 2019-10-11 00:04:02 PDT
*** Bug 202710 has been marked as a duplicate of this bug. ***
Robin Morisset
Comment 8 2019-10-14 18:32:03 PDT
EWS Watchlist
Comment 9 2019-10-14 20:48:28 PDT
Comment on attachment 380943 [details] Patch Attachment 380943 [details] did not pass jsc-ews (mac): Output: https://webkit-queues.webkit.org/results/13131267 New failing tests: mozilla-tests.yaml/js1_5/Array/regress-101964.js.mozilla
Robin Morisset
Comment 10 2019-10-15 11:12:36 PDT
Created attachment 381002 [details] Patch Same patch as before, to check that the previous test failure was a flake.
Saam Barati
Comment 11 2019-10-15 11:21:39 PDT
Comment on attachment 381002 [details] Patch View in context: https://bugs.webkit.org/attachment.cgi?id=381002&action=review r=me > Source/JavaScriptCore/bytecompiler/NodesCodegen.cpp:1791 > + RefPtr<RegisterID> tmp = generator.emitToNumber(generator.newTemporary(), srcDst); > + RefPtr<RegisterID> result = generator.tempDestination(srcDst); > + generator.move(result.get(), tmp.get()); > + emitIncOrDec(generator, result.get(), oper); > + generator.move(srcDst, result.get()); why not something like this? ``` RefPtr<RegisterID> result = generator.emitToNumber(generator.tempDestination(dst), srcDst); emitIncOrDec(generator, result.get(), oper); generator.move(srcDst, result.get()); return generator.move(dst, result.get()); ```
Robin Morisset
Comment 12 2019-10-21 11:11:18 PDT
Comment on attachment 381002 [details] Patch The jsc-mips bot appears stuck, but since it does not show an actual test failure and all other bots are green, I'm trying to land this.
WebKit Commit Bot
Comment 13 2019-10-21 12:06:55 PDT
Comment on attachment 381002 [details] Patch Clearing flags on attachment: 381002 Committed r251371: <https://trac.webkit.org/changeset/251371>
WebKit Commit Bot
Comment 14 2019-10-21 12:06:57 PDT
All reviewed patches have been landed. Closing bug.
Radar WebKit Bug Importer
Comment 15 2019-10-21 12:12:37 PDT
Note You need to log in before you can comment on or make changes to this bug.