| // Copyright 2018 the V8 project authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| // Flags: --allow-natives-syntax --opt --no-always-opt |
| // Flags: --no-stress-background-compile --trace-opt --trace-deopt |
| |
| let id = 0; |
| |
| function runTest(f, message, mkICTraining, deoptArg, speculationCheck) { |
| function test(f, message, ictraining, deoptArg, speculationCheck) { |
| // Train the call ic to the maps. |
| let t = ictraining; |
| |
| // We put the training data into local variables |
| // to ensure their maps are kepts alive. If the |
| // maps die, gc *may* deoptimize {f}, which makes |
| // the test flaky. |
| let t1 = t(); |
| let t2 = t(); |
| let t3 = t(); |
| |
| %PrepareFunctionForOptimization(f); |
| for (let a of t1) { |
| f(a.arr, () => a.el); |
| } |
| for (let a of t2) { |
| f(a.arr, () => a.el); |
| } |
| %OptimizeFunctionOnNextCall(f); |
| message += " trained with" + JSON.stringify(t()); |
| if (deoptArg == undefined) { |
| // Make sure the optimized function can handle |
| // all trained maps without deopt. |
| for (let a of t3) { |
| message += " for args " + JSON.stringify(a) + " should have been optimized"; |
| f(a.arr, () => a.el); |
| assertOptimized(f, undefined, message); |
| } |
| } else { |
| // Trigger deopt, causing no-speculation bit to be set. |
| let a1 = deoptArg; |
| let a2 = deoptArg; |
| let a3 = deoptArg; |
| message += " for args " + JSON.stringify(a1); |
| message_unoptimized = message + " should have been unoptimized" |
| message_optimized = message + " should have been optimized" |
| f(a1.darr, () => a1.del); |
| assertUnoptimized(f, undefined, message_unoptimized); |
| if (speculationCheck) { |
| %PrepareFunctionForOptimization(f); |
| %OptimizeFunctionOnNextCall(f); |
| f(a2.darr, () => a2.del); |
| assertUnoptimized(f, undefined, message_unoptimized); |
| } |
| %PrepareFunctionForOptimization(f); |
| %OptimizeFunctionOnNextCall(f); |
| // No speculation should protect against further deopts. |
| f(a3.darr, () => a3.del); |
| assertOptimized(f, undefined, message_optimized); |
| } |
| } |
| |
| // Get function as a string. |
| var testString = test.toString(); |
| // Remove the function header.. |
| testString = testString.replace(new RegExp("[^\n]*"), "let f = " + f.toString() + ";"); |
| // ..and trailing '}'. |
| testString = testString.replace(new RegExp("[^\n]*$"), ""); |
| // Substitute parameters. |
| testString = testString.replace(new RegExp("ictraining", 'g'), mkICTraining.toString()); |
| testString = testString.replace(new RegExp("deoptArg", 'g'), |
| deoptArg ? JSON.stringify(deoptArg).replace(/"/g,'') : "undefined"); |
| testString = testString.replace(new RegExp("speculationCheck", 'g'), |
| speculationCheck ? JSON.stringify(deoptArg).replace(/"/g,'') : "undefined"); |
| |
| // Make field names unique to avoid learning of types. |
| id = id + 1; |
| testString = testString.replace(/[.]el/g, '.el' + id); |
| testString = testString.replace(/el:/g, 'el' + id + ':'); |
| testString = testString.replace(/[.]arr/g, '.arr' + id); |
| testString = testString.replace(/arr:/g, 'arr' + id + ':'); |
| testString = testString.replace(/[.]del/g, '.del' + id); |
| testString = testString.replace(/[.]darr/g, '.darr' + id); |
| |
| var modTest = new Function("message", testString); |
| modTest(message); |
| } |
| |
| let checks = { |
| smiReceiver: |
| { mkTrainingArguments : () => [{arr:[1], el:3}], |
| deoptingArguments : [{darr:[0.1], del:1}, {darr:[{}], del:1}] |
| }, |
| objectReceiver: |
| { mkTrainingArguments : () => [{arr:[{}], el:0.1}], |
| deoptingArguments : [] |
| }, |
| multipleSmiReceivers: |
| { mkTrainingArguments : () => { let b = [1]; b.x=3; return [{arr:[1], el:3}, {arr:b, el:3}] }, |
| deoptingArguments : [{darr:[0.1], del:1}, {darr:[{}], del:1}] |
| }, |
| multipleSmiReceiversPackedUnpacked: |
| { mkTrainingArguments : () => { let b = [1]; b[100] = 3; return [{arr:[1], el:3}, {arr:b, el:3}] }, |
| deoptingArguments : [{darr:[0.1], del:1}, {darr:[{}], del:1}] |
| }, |
| multipleDoubleReceivers: |
| { mkTrainingArguments : () => { let b = [0.1]; b.x=0.3; return [{arr:[0.1], el:0.3}, {arr:b, el:0.3}] }, |
| deoptingArguments : [{darr:[{}], del:true}, {darr:[1], del: 1}] |
| }, |
| multipleDoubleReceiversPackedUnpacked: |
| { mkTrainingArguments : () => { let b = [0.1]; b[100] = 0.3; return [{arr:[0.1], el:0.3}, {arr:b, el:0.3}] }, |
| deoptingArguments : [{darr:[{}], del:true}, {darr:[1], del: 1}] |
| }, |
| multipleMixedReceivers: |
| { mkTrainingArguments : () => { let b = [0.1]; b.x=0.3; return [{arr:[1], el:1}, {arr:[{}], el:true}, {arr:b, el:0.3}] }, |
| deoptingArguments : [] |
| }, |
| multipleMixedReceiversPackedUnpacked: |
| { mkTrainingArguments : () => { let b = [0.1]; b[100] = 0.3; return [{arr:[1], el:1}, {arr:[{}], el:true}, {arr:b, el:0.3}] }, |
| deoptingArguments : [] |
| }, |
| }; |
| |
| let no_speculation_checks = { |
| smiReceiver: |
| { mkTrainingArguments : () => [{arr:[1], el:3}], |
| deoptingArguments : [{darr:[0.1], del:true}] |
| }, |
| multipleSmiReceivers: |
| { mkTrainingArguments : () => { let b = [1]; b.x=3; return [{arr:[1], el:3}, {arr:[1], el:3}] }, |
| deoptingArguments : [{darr:[0.1], del:true}] |
| }, |
| multipleSmiReceiversPackedUnpacked: |
| { mkTrainingArguments : () => { let b = [1]; b[100] = 3; return [{arr:[1], el:3}, {arr:b, el:3}] }, |
| deoptingArguments : [{darr:[0.1], del:true}] |
| }, |
| multipleDoubleReceivers: |
| { mkTrainingArguments : () => { let b = [0.1]; b.x=0.3; return [{arr:[0.1], el:0.3}, {arr:b, el:0.3}] }, |
| deoptingArguments : [{darr:[1], del:true}] |
| }, |
| multipleDoubleReceiversPackedUnpacked: |
| { mkTrainingArguments : () => { let b = [0.1]; b[100] = 0.3; return [{arr:[0.1], el:0.3}, {arr:b, el:0.3}] }, |
| deoptingArguments : [{darr:[1], del:true}] |
| }, |
| }; |
| |
| const functions = { |
| push_reliable: (a,g) => { let b = g(); return a.push(2, b); }, |
| push_unreliable: (a,g) => { return a.push(2, g()); }, |
| pop_reliable: (a,g) => { let b = g(); return a.pop(2, b); }, |
| pop_unreliable: (a,g) => { return a.pop(2, g()); }, |
| shift_reliable: (a,g) => { let b = g(); return a.shift(2, b); }, |
| shift_unreliable: (a,g) => { return a.shift(2, g()); } |
| } |
| |
| const push_functions = { |
| push_reliable: (a,g) => { let b = g(); return a.push(2, b); }, |
| push_unreliable: (a,g) => { return a.push(2, g()); }, |
| } |
| |
| Object.keys(checks).forEach( |
| key => { |
| let check = checks[key]; |
| |
| for (fnc in functions) { |
| runTest(functions[fnc], "test-" + fnc + "-" + key, check.mkTrainingArguments); |
| // Test each deopting arg separately. |
| for (let deoptArg of check.deoptingArguments) { |
| runTest(functions[fnc], "testDeopt-" + fnc + "-" + key, check.mkTrainingArguments, deoptArg); |
| } |
| } |
| } |
| ); |
| |
| Object.keys(no_speculation_checks).forEach( |
| key => { |
| let check = no_speculation_checks[key]; |
| |
| for (fnc in push_functions) { |
| runTest(functions[fnc], "test-spec-check-" + fnc + "-" + key, check.mkTrainingArguments); |
| // Test each deopting arg separately. |
| for (let deoptArg of check.deoptingArguments) { |
| runTest(functions[fnc], "testDeopt-spec-check-" + fnc + "-" + key, check.mkTrainingArguments, deoptArg, true); |
| } |
| } |
| } |
| ); |