From 62656bd8417090620d63ca8515bf373ccf22169f Mon Sep 17 00:00:00 2001 From: "duncan.macgregor" Date: Tue, 24 Dec 2024 22:07:43 +0000 Subject: [PATCH 1/2] Parameterize benchmarks to check interpreted and compiled class performance. --- .../benchmarks/BuiltinBenchmark.java | 4 + .../javascript/benchmarks/MathBenchmark.java | 4 + .../benchmarks/ObjectBenchmark.java | 4 + .../benchmarks/PropertyBenchmark.java | 4 + .../benchmarks/SunSpiderBenchmark.java | 211 ++++++++++++++++++ .../javascript/benchmarks/V8Benchmark.java | 33 ++- 6 files changed, 253 insertions(+), 7 deletions(-) diff --git a/benchmarks/src/jmh/java/org/mozilla/javascript/benchmarks/BuiltinBenchmark.java b/benchmarks/src/jmh/java/org/mozilla/javascript/benchmarks/BuiltinBenchmark.java index 9071d4f7bb..665a89bc00 100644 --- a/benchmarks/src/jmh/java/org/mozilla/javascript/benchmarks/BuiltinBenchmark.java +++ b/benchmarks/src/jmh/java/org/mozilla/javascript/benchmarks/BuiltinBenchmark.java @@ -19,9 +19,13 @@ public class BuiltinBenchmark { @State(Scope.Thread) public static class AbstractClassState { + @Param({"false", "true"}) + public boolean interpreted; + public void init() throws IllegalAccessException, InvocationTargetException, InstantiationException { cx = Context.enter(); + cx.setInterpretedMode(interpreted); cx.setLanguageVersion(Context.VERSION_ES6); scope = cx.initStandardObjects(); diff --git a/benchmarks/src/jmh/java/org/mozilla/javascript/benchmarks/MathBenchmark.java b/benchmarks/src/jmh/java/org/mozilla/javascript/benchmarks/MathBenchmark.java index e298919eff..60e34430a9 100644 --- a/benchmarks/src/jmh/java/org/mozilla/javascript/benchmarks/MathBenchmark.java +++ b/benchmarks/src/jmh/java/org/mozilla/javascript/benchmarks/MathBenchmark.java @@ -33,9 +33,13 @@ public static class MathState { Function bitwiseRsh; Function bitwiseSignedRsh; + @Param({"false", "true"}) + public boolean interpreted; + @Setup(Level.Trial) public void setup() throws IOException { cx = Context.enter(); + cx.setInterpretedMode(interpreted); cx.setLanguageVersion(Context.VERSION_ES6); scope = cx.initStandardObjects(); diff --git a/benchmarks/src/jmh/java/org/mozilla/javascript/benchmarks/ObjectBenchmark.java b/benchmarks/src/jmh/java/org/mozilla/javascript/benchmarks/ObjectBenchmark.java index 269469a2f1..eda405d858 100644 --- a/benchmarks/src/jmh/java/org/mozilla/javascript/benchmarks/ObjectBenchmark.java +++ b/benchmarks/src/jmh/java/org/mozilla/javascript/benchmarks/ObjectBenchmark.java @@ -33,10 +33,14 @@ public static class FieldTestState { Scriptable strings; Scriptable ints; + @Param({"false", "true"}) + public boolean interpreted; + @Setup(Level.Trial) @SuppressWarnings("unused") public void create() throws IOException { cx = Context.enter(); + cx.setInterpretedMode(interpreted); cx.setLanguageVersion(Context.VERSION_ES6); scope = new Global(cx); diff --git a/benchmarks/src/jmh/java/org/mozilla/javascript/benchmarks/PropertyBenchmark.java b/benchmarks/src/jmh/java/org/mozilla/javascript/benchmarks/PropertyBenchmark.java index 402f003934..b93ac4e46b 100644 --- a/benchmarks/src/jmh/java/org/mozilla/javascript/benchmarks/PropertyBenchmark.java +++ b/benchmarks/src/jmh/java/org/mozilla/javascript/benchmarks/PropertyBenchmark.java @@ -24,9 +24,13 @@ public static class PropertyState { Object object; + @Param({"false", "true"}) + public boolean interpreted; + @Setup(Level.Trial) public void setup() throws IOException { cx = Context.enter(); + cx.setInterpretedMode(interpreted); cx.setLanguageVersion(Context.VERSION_ES6); scope = cx.initStandardObjects(); diff --git a/benchmarks/src/jmh/java/org/mozilla/javascript/benchmarks/SunSpiderBenchmark.java b/benchmarks/src/jmh/java/org/mozilla/javascript/benchmarks/SunSpiderBenchmark.java index 12555f6c0a..8e6d202144 100644 --- a/benchmarks/src/jmh/java/org/mozilla/javascript/benchmarks/SunSpiderBenchmark.java +++ b/benchmarks/src/jmh/java/org/mozilla/javascript/benchmarks/SunSpiderBenchmark.java @@ -22,9 +22,12 @@ abstract static class AbstractState { this.fileName = TEST_BASE + fileName; } + protected abstract boolean isInterpreted(); + @Setup(Level.Trial) public void setUp() { cx = Context.enter(); + cx.setInterpretedMode(isInterpreted()); cx.setLanguageVersion(Context.VERSION_ES6); scope = cx.initStandardObjects(); @@ -47,10 +50,18 @@ Object run() { @State(Scope.Thread) public static class ThreeDCubeState extends AbstractState { + @Param({"false", "true"}) + public boolean interpreted; + public ThreeDCubeState() { super("3d-cube.js"); } + @Override + protected boolean isInterpreted() { + return interpreted; + } + @Benchmark public Object threeDCube(ThreeDCubeState state) { return state.run(); @@ -59,10 +70,18 @@ public Object threeDCube(ThreeDCubeState state) { @State(Scope.Thread) public static class ThreeDMorphState extends AbstractState { + @Param({"false", "true"}) + public boolean interpreted; + public ThreeDMorphState() { super("3d-morph.js"); } + @Override + protected boolean isInterpreted() { + return interpreted; + } + @Benchmark public Object threeDMorph(ThreeDMorphState state) { return state.run(); @@ -71,10 +90,18 @@ public Object threeDMorph(ThreeDMorphState state) { @State(Scope.Thread) public static class ThreeDRayState extends AbstractState { + @Param({"false", "true"}) + public boolean interpreted; + public ThreeDRayState() { super("3d-raytrace.js"); } + @Override + protected boolean isInterpreted() { + return interpreted; + } + @Benchmark public Object threeDRayTrace(ThreeDRayState state) { return state.run(); @@ -83,10 +110,18 @@ public Object threeDRayTrace(ThreeDRayState state) { @State(Scope.Thread) public static class AccessBinaryTreesState extends AbstractState { + @Param({"false", "true"}) + public boolean interpreted; + public AccessBinaryTreesState() { super("access-binary-trees.js"); } + @Override + protected boolean isInterpreted() { + return interpreted; + } + @Benchmark public Object accessBinaryTrees(AccessBinaryTreesState state) { return state.run(); @@ -95,10 +130,18 @@ public Object accessBinaryTrees(AccessBinaryTreesState state) { @State(Scope.Thread) public static class AccessFannkuchState extends AbstractState { + @Param({"false", "true"}) + public boolean interpreted; + public AccessFannkuchState() { super("access-fannkuch.js"); } + @Override + protected boolean isInterpreted() { + return interpreted; + } + @Benchmark public Object accessFannkuch(AccessFannkuchState state) { return state.run(); @@ -107,10 +150,18 @@ public Object accessFannkuch(AccessFannkuchState state) { @State(Scope.Thread) public static class AccessNBodyState extends AbstractState { + @Param({"false", "true"}) + public boolean interpreted; + public AccessNBodyState() { super("access-nbody.js"); } + @Override + protected boolean isInterpreted() { + return interpreted; + } + @Benchmark public Object accessNBody(AccessNBodyState state) { return state.run(); @@ -119,10 +170,18 @@ public Object accessNBody(AccessNBodyState state) { @State(Scope.Thread) public static class AccessFannAccessNsieveState extends AbstractState { + @Param({"false", "true"}) + public boolean interpreted; + public AccessFannAccessNsieveState() { super("access-nsieve.js"); } + @Override + protected boolean isInterpreted() { + return interpreted; + } + @Benchmark public Object accessNsieve(AccessFannAccessNsieveState state) { return state.run(); @@ -131,10 +190,18 @@ public Object accessNsieve(AccessFannAccessNsieveState state) { @State(Scope.Thread) public static class Bitops3BitState extends AbstractState { + @Param({"false", "true"}) + public boolean interpreted; + public Bitops3BitState() { super("bitops-3bit-bits-in-byte.js"); } + @Override + protected boolean isInterpreted() { + return interpreted; + } + @Benchmark public Object bitops3BitBitsInByte(Bitops3BitState state) { return state.run(); @@ -143,10 +210,18 @@ public Object bitops3BitBitsInByte(Bitops3BitState state) { @State(Scope.Thread) public static class BitopsBitsState extends AbstractState { + @Param({"false", "true"}) + public boolean interpreted; + public BitopsBitsState() { super("bitops-bits-in-byte.js"); } + @Override + protected boolean isInterpreted() { + return interpreted; + } + @Benchmark public Object bitopsBitsInByte(BitopsBitsState state) { return state.run(); @@ -155,10 +230,18 @@ public Object bitopsBitsInByte(BitopsBitsState state) { @State(Scope.Thread) public static class BitopsAndState extends AbstractState { + @Param({"false", "true"}) + public boolean interpreted; + public BitopsAndState() { super("bitops-bitwise-and.js"); } + @Override + protected boolean isInterpreted() { + return interpreted; + } + @Benchmark public Object bitopsBitwiseAnd(BitopsAndState state) { return state.run(); @@ -167,10 +250,18 @@ public Object bitopsBitwiseAnd(BitopsAndState state) { @State(Scope.Thread) public static class BitopsNsieveState extends AbstractState { + @Param({"false", "true"}) + public boolean interpreted; + public BitopsNsieveState() { super("bitops-nsieve-bits.js"); } + @Override + protected boolean isInterpreted() { + return interpreted; + } + @Benchmark public Object bitopsNsieveBits(BitopsNsieveState state) { return state.run(); @@ -179,10 +270,18 @@ public Object bitopsNsieveBits(BitopsNsieveState state) { @State(Scope.Thread) public static class RecursiveState extends AbstractState { + @Param({"false", "true"}) + public boolean interpreted; + public RecursiveState() { super("controlflow-recursive.js"); } + @Override + protected boolean isInterpreted() { + return interpreted; + } + @Benchmark public Object controlflowRecursive(RecursiveState state) { return state.run(); @@ -191,10 +290,18 @@ public Object controlflowRecursive(RecursiveState state) { @State(Scope.Thread) public static class CryptoAesState extends AbstractState { + @Param({"false", "true"}) + public boolean interpreted; + public CryptoAesState() { super("crypto-aes.js"); } + @Override + protected boolean isInterpreted() { + return interpreted; + } + @Benchmark public Object cryptoAes(CryptoAesState state) { return state.run(); @@ -203,10 +310,18 @@ public Object cryptoAes(CryptoAesState state) { @State(Scope.Thread) public static class CryptoMd5State extends AbstractState { + @Param({"false", "true"}) + public boolean interpreted; + public CryptoMd5State() { super("crypto-md5.js"); } + @Override + protected boolean isInterpreted() { + return interpreted; + } + @Benchmark public Object cryptoMd5(CryptoMd5State state) { return state.run(); @@ -215,10 +330,18 @@ public Object cryptoMd5(CryptoMd5State state) { @State(Scope.Thread) public static class CryptoShaState extends AbstractState { + @Param({"false", "true"}) + public boolean interpreted; + public CryptoShaState() { super("crypto-sha1.js"); } + @Override + protected boolean isInterpreted() { + return interpreted; + } + @Benchmark public Object cryptoSha1(CryptoShaState state) { return state.run(); @@ -227,10 +350,18 @@ public Object cryptoSha1(CryptoShaState state) { @State(Scope.Thread) public static class DateFormatToFteState extends AbstractState { + @Param({"false", "true"}) + public boolean interpreted; + public DateFormatToFteState() { super("date-format-tofte.js"); } + @Override + protected boolean isInterpreted() { + return interpreted; + } + @Benchmark public Object dateFormatToFte(DateFormatToFteState state) { return state.run(); @@ -239,10 +370,18 @@ public Object dateFormatToFte(DateFormatToFteState state) { @State(Scope.Thread) public static class DateFormatXparbState extends AbstractState { + @Param({"false", "true"}) + public boolean interpreted; + public DateFormatXparbState() { super("date-format-xparb.js"); } + @Override + protected boolean isInterpreted() { + return interpreted; + } + @Benchmark public Object dateFormatXparb(DateFormatXparbState state) { return state.run(); @@ -251,10 +390,18 @@ public Object dateFormatXparb(DateFormatXparbState state) { @State(Scope.Thread) public static class MathCordicState extends AbstractState { + @Param({"false", "true"}) + public boolean interpreted; + public MathCordicState() { super("math-cordic.js"); } + @Override + protected boolean isInterpreted() { + return interpreted; + } + @Benchmark public Object mathCordic(MathCordicState state) { return state.run(); @@ -263,10 +410,18 @@ public Object mathCordic(MathCordicState state) { @State(Scope.Thread) public static class MathPartialState extends AbstractState { + @Param({"false", "true"}) + public boolean interpreted; + public MathPartialState() { super("math-partial-sums.js"); } + @Override + protected boolean isInterpreted() { + return interpreted; + } + @Benchmark public Object mathPartialSums(MathPartialState state) { return state.run(); @@ -275,10 +430,18 @@ public Object mathPartialSums(MathPartialState state) { @State(Scope.Thread) public static class MathSpectralNormState extends AbstractState { + @Param({"false", "true"}) + public boolean interpreted; + public MathSpectralNormState() { super("math-spectral-norm.js"); } + @Override + protected boolean isInterpreted() { + return interpreted; + } + @Benchmark public Object mathSpectralNorm(MathSpectralNormState state) { return state.run(); @@ -287,10 +450,18 @@ public Object mathSpectralNorm(MathSpectralNormState state) { @State(Scope.Thread) public static class RegexpState extends AbstractState { + @Param({"false", "true"}) + public boolean interpreted; + public RegexpState() { super("regexp-dna.js"); } + @Override + protected boolean isInterpreted() { + return interpreted; + } + @Benchmark public Object regexpDna(RegexpState state) { return state.run(); @@ -299,10 +470,18 @@ public Object regexpDna(RegexpState state) { @State(Scope.Thread) public static class StringBase64State extends AbstractState { + @Param({"false", "true"}) + public boolean interpreted; + public StringBase64State() { super("string-base64.js"); } + @Override + protected boolean isInterpreted() { + return interpreted; + } + @Benchmark public Object stringBase64(StringBase64State state) { return state.run(); @@ -311,10 +490,18 @@ public Object stringBase64(StringBase64State state) { @State(Scope.Thread) public static class StringFastaState extends AbstractState { + @Param({"false", "true"}) + public boolean interpreted; + public StringFastaState() { super("string-fasta.js"); } + @Override + protected boolean isInterpreted() { + return interpreted; + } + @Benchmark public Object stringFasta(StringFastaState state) { return state.run(); @@ -323,10 +510,18 @@ public Object stringFasta(StringFastaState state) { @State(Scope.Thread) public static class StringTagcloudState extends AbstractState { + @Param({"false", "true"}) + public boolean interpreted; + public StringTagcloudState() { super("string-tagcloud.js"); } + @Override + protected boolean isInterpreted() { + return interpreted; + } + @Benchmark public Object stringTagcloud(StringTagcloudState state) { return state.run(); @@ -335,10 +530,18 @@ public Object stringTagcloud(StringTagcloudState state) { @State(Scope.Thread) public static class StringUnpackState extends AbstractState { + @Param({"false", "true"}) + public boolean interpreted; + public StringUnpackState() { super("string-unpack-code.js"); } + @Override + protected boolean isInterpreted() { + return interpreted; + } + @Benchmark public Object stringUnpackCode(StringUnpackState state) { return state.run(); @@ -347,10 +550,18 @@ public Object stringUnpackCode(StringUnpackState state) { @State(Scope.Thread) public static class StringValidateState extends AbstractState { + @Param({"false", "true"}) + public boolean interpreted; + public StringValidateState() { super("string-validate-input.js"); } + @Override + protected boolean isInterpreted() { + return interpreted; + } + @Benchmark public Object stringValidateInput(StringValidateState state) { return state.run(); diff --git a/benchmarks/src/jmh/java/org/mozilla/javascript/benchmarks/V8Benchmark.java b/benchmarks/src/jmh/java/org/mozilla/javascript/benchmarks/V8Benchmark.java index 9c38ffe717..cc9e953c13 100644 --- a/benchmarks/src/jmh/java/org/mozilla/javascript/benchmarks/V8Benchmark.java +++ b/benchmarks/src/jmh/java/org/mozilla/javascript/benchmarks/V8Benchmark.java @@ -38,8 +38,9 @@ void evaluateSource(Context cx, Scriptable scope, String fileName) { } } - void initialize() { + void initialize(boolean interpreted) { cx = Context.enter(); + cx.setInterpretedMode(interpreted); cx.setLanguageVersion(Context.VERSION_ES6); scope = cx.initStandardObjects(); evaluateSource(cx, scope, "testsrc/benchmarks/framework.js"); @@ -64,9 +65,12 @@ void runCleanup() { public static class SplayState extends AbstractState { Callable splay; + @Param({"false", "true"}) + public boolean interpreted; + @Setup(Level.Trial) public void setUp() { - initialize(); + initialize(interpreted); evaluateSource(cx, scope, "testsrc/benchmarks/v8-benchmarks-v6/splay.js"); runSetup(); splay = getRunFunc("Splay"); @@ -89,9 +93,12 @@ public static class CryptoState extends AbstractState { Callable encrypt; Callable decrypt; + @Param({"false", "true"}) + public boolean interpreted; + @Setup(Level.Trial) public void setUp() { - initialize(); + initialize(interpreted); evaluateSource(cx, scope, "testsrc/benchmarks/v8-benchmarks-v6/crypto.js"); runSetup(); encrypt = getRunFunc("Encrypt"); @@ -121,9 +128,12 @@ public Object cryptoDecrypt(CryptoState state) { public static class DeltaBlueState extends AbstractState { Callable db; + @Param({"false", "true"}) + public boolean interpreted; + @Setup(Level.Trial) public void setUp() { - initialize(); + initialize(interpreted); evaluateSource(cx, scope, "testsrc/benchmarks/v8-benchmarks-v6/deltablue.js"); runSetup(); db = getRunFunc("DeltaBlue"); @@ -145,9 +155,12 @@ public Object deltaBlue(DeltaBlueState state) { public static class RayTraceState extends AbstractState { Callable rt; + @Param({"false", "true"}) + public boolean interpreted; + @Setup(Level.Trial) public void setUp() { - initialize(); + initialize(interpreted); evaluateSource(cx, scope, "testsrc/benchmarks/v8-benchmarks-v6/raytrace.js"); runSetup(); rt = getRunFunc("RayTrace"); @@ -195,9 +208,12 @@ public Object regExp(RegExpState state) { public static class RichardsState extends AbstractState { Callable r; + @Param({"false", "true"}) + public boolean interpreted; + @Setup(Level.Trial) public void setUp() { - initialize(); + initialize(interpreted); evaluateSource(cx, scope, "testsrc/benchmarks/v8-benchmarks-v6/richards.js"); runSetup(); r = getRunFunc("Richards"); @@ -220,9 +236,12 @@ public static class EarleyBoyerState extends AbstractState { Callable earley; Callable boyer; + @Param({"false", "true"}) + public boolean interpreted; + @Setup(Level.Trial) public void setUp() { - initialize(); + initialize(interpreted); evaluateSource(cx, scope, "testsrc/benchmarks/v8-benchmarks-v6/earley-boyer.js"); runSetup(); earley = getRunFunc("Earley"); From 21043c134c270d0cf3adf6cf127babd89085048d Mon Sep 17 00:00:00 2001 From: "duncan.macgregor" Date: Tue, 24 Dec 2024 22:09:04 +0000 Subject: [PATCH 2/2] Extract function calling out of the interpreter loop for performance. --- .../org/mozilla/javascript/Interpreter.java | 548 +++++++++--------- 1 file changed, 283 insertions(+), 265 deletions(-) diff --git a/rhino/src/main/java/org/mozilla/javascript/Interpreter.java b/rhino/src/main/java/org/mozilla/javascript/Interpreter.java index d777ae172d..4b107b342e 100644 --- a/rhino/src/main/java/org/mozilla/javascript/Interpreter.java +++ b/rhino/src/main/java/org/mozilla/javascript/Interpreter.java @@ -1208,20 +1208,52 @@ public static Object restartContinuation( return interpretLoop(cx, null, cjump); } + // arbitrary number to add to instructionCount when calling + // other functions + private static final int INVOCATION_COST = 100; + // arbitrary exception cost for instruction counting + private static final int EXCEPTION_COST = 100; + + private static final Object undefined = Undefined.instance; + + private static class NewState {} + + private static final class StateContinue extends NewState { + private final CallFrame frame; + + private StateContinue(CallFrame frame) { + this.frame = frame; + } + } + + private static final class NewThrowable extends NewState { + private final Object throwable; + + private NewThrowable(Object throwable) { + this.throwable = throwable; + } + } + + private static final class ContinueLoop extends NewState { + private final CallFrame frame; + private final int stackTop; + private final int indexReg; + + private ContinueLoop(CallFrame frame, int stackTop, int indexReg) { + this.frame = frame; + this.stackTop = stackTop; + this.indexReg = indexReg; + } + } + private static Object interpretLoop(Context cx, CallFrame frame, Object throwable) { // throwable holds exception object to rethrow or catch // It is also used for continuation restart in which case // it holds ContinuationJump final Object DBL_MRK = DOUBLE_MARK; - final Object undefined = Undefined.instance; final boolean instructionCounting = cx.instructionThreshold != 0; - // arbitrary number to add to instructionCount when calling - // other functions - final int INVOCATION_COST = 100; - // arbitrary exception cost for instruction counting - final int EXCEPTION_COST = 100; String stringReg = null; BigInteger bigIntReg = null; @@ -1930,268 +1962,32 @@ private static Object interpretLoop(Context cx, CallFrame frame, Object throwabl case Icode_TAIL_CALL: case Token.REF_CALL: { - if (instructionCounting) { - cx.instructionCount += INVOCATION_COST; - } - // stack change: function thisObj arg0 .. argN -> result - // indexReg: number of arguments - stackTop -= 1 + indexReg; - - // CALL generation ensures that fun and funThisObj - // are already Scriptable and Callable objects respectively - Callable fun = (Callable) stack[stackTop]; - Scriptable funThisObj = (Scriptable) stack[stackTop + 1]; - Scriptable funHomeObj = - (fun instanceof BaseFunction) - ? ((BaseFunction) fun).getHomeObject() - : null; - if (op == Icode_CALL_ON_SUPER) { - // funThisObj would have been the "super" object, which we - // used to lookup the function. Now that that's done, we - // discard it and invoke the function with the current - // "this". - funThisObj = frame.thisObj; - } - - if (op == Token.REF_CALL) { - Object[] outArgs = - getArgsArray(stack, sDbl, stackTop + 2, indexReg); - stack[stackTop] = - ScriptRuntime.callRef( - fun, funThisObj, - outArgs, cx); + var callState = + doCallByteCode( + cx, + frame, + instructionCounting, + op, + stackTop, + indexReg); + if (callState instanceof ContinueLoop) { + var contLoop = (ContinueLoop) callState; + frame = contLoop.frame; + stack = frame.stack; + sDbl = frame.sDbl; + stackTop = contLoop.stackTop; + indexReg = contLoop.indexReg; continue Loop; - } - Scriptable calleeScope = frame.scope; - if (frame.useActivation) { - calleeScope = - ScriptableObject.getTopLevelScope(frame.scope); - } - // Iteratively reduce known function types: arrows, lambdas, - // bound functions, call/apply, and no-such-method-handler in - // order to make a best-effort to keep them in this interpreter - // loop so continuations keep working. The loop initializer and - // condition are formulated so that they short-circuit the loop - // if the function is already an interpreted function, which - // should be the majority of cases. - for (; ; ) { - if (fun instanceof ArrowFunction) { - ArrowFunction afun = (ArrowFunction) fun; - fun = afun.getTargetFunction(); - funThisObj = afun.getCallThis(cx); - funHomeObj = afun.getBoundHomeObject(); - } else if (fun instanceof LambdaConstructor) { - break; - } else if (fun instanceof LambdaFunction) { - fun = ((LambdaFunction) fun).getTarget(); - } else if (fun instanceof BoundFunction) { - BoundFunction bfun = (BoundFunction) fun; - fun = bfun.getTargetFunction(); - funThisObj = bfun.getCallThis(cx, calleeScope); - Object[] boundArgs = bfun.getBoundArgs(); - int blen = boundArgs.length; - if (blen > 0) { - stack = - frame.ensureStackLength( - blen + stackTop + 2 + indexReg); - sDbl = frame.sDbl; - System.arraycopy( - stack, - stackTop + 2, - stack, - stackTop + 2 + blen, - indexReg); - System.arraycopy( - sDbl, - stackTop + 2, - sDbl, - stackTop + 2 + blen, - indexReg); - System.arraycopy( - boundArgs, 0, stack, stackTop + 2, blen); - indexReg += blen; - } - } else if (fun instanceof IdFunctionObject) { - IdFunctionObject ifun = (IdFunctionObject) fun; - // Bug 405654 -- make the best effort to keep - // Function.apply and Function.call within this - // interpreter loop invocation - if (BaseFunction.isApplyOrCall(ifun)) { - // funThisObj becomes fun - fun = ScriptRuntime.getCallable(funThisObj); - // first arg becomes thisObj - funThisObj = - getApplyThis( - cx, - stack, - sDbl, - stackTop + 2, - indexReg, - fun, - frame); - if (BaseFunction.isApply(ifun)) { - // Apply: second argument after new "this" - // should be array-like - // and we'll spread its elements on the stack - Object[] callArgs = - indexReg < 2 - ? ScriptRuntime.emptyArgs - : ScriptRuntime - .getApplyArguments( - cx, - stack[ - stackTop - + 3]); - int alen = callArgs.length; - stack = - frame.ensureStackLength( - alen + stackTop + 2); - sDbl = frame.sDbl; - System.arraycopy( - callArgs, 0, stack, stackTop + 2, alen); - indexReg = alen; - } else { - // Call: shift args left, starting from 2nd - if (indexReg > 0) { - if (indexReg > 1) { - System.arraycopy( - stack, - stackTop + 3, - stack, - stackTop + 2, - indexReg - 1); - System.arraycopy( - sDbl, - stackTop + 3, - sDbl, - stackTop + 2, - indexReg - 1); - } - indexReg--; - } - } - } else { - // Some other IdFunctionObject we don't know how to - // reduce. - break; - } - } else if (fun instanceof NoSuchMethodShim) { - NoSuchMethodShim nsmfun = (NoSuchMethodShim) fun; - // Bug 447697 -- make best effort to keep - // __noSuchMethod__ within this interpreter loop - // invocation. - stack = frame.ensureStackLength(stackTop + 4); - sDbl = frame.sDbl; - Object[] elements = - getArgsArray( - stack, sDbl, stackTop + 2, indexReg); - fun = nsmfun.noSuchMethodMethod; - stack[stackTop + 2] = nsmfun.methodName; - stack[stackTop + 3] = - cx.newArray(calleeScope, elements); - indexReg = 2; - } else if (fun == null) { - throw ScriptRuntime.notFunctionError(null, null); - } else { - // Current function is something that we can't reduce - // further. - break; - } - } - - if (fun instanceof InterpretedFunction) { - InterpretedFunction ifun = (InterpretedFunction) fun; - if (frame.fnOrScript.securityDomain - == ifun.securityDomain) { - CallFrame callParentFrame = frame; - if (op == Icode_TAIL_CALL) { - // In principle tail call can re-use the current - // frame and its stack arrays but it is hard to - // do properly. Any exceptions that can legally - // happen during frame re-initialization including - // StackOverflowException during innocent looking - // System.arraycopy may leave the current frame - // data corrupted leading to undefined behaviour - // in the catch code bellow that unwinds JS stack - // on exceptions. Then there is issue about frame - // release - // end exceptions there. - // To avoid frame allocation a released frame - // can be cached for re-use which would also benefit - // non-tail calls but it is not clear that this - // caching - // would gain in performance due to potentially - // bad interaction with GC. - callParentFrame = frame.parentFrame; - // Release the current frame. See Bug #344501 to see - // why - // it is being done here. - exitFrame(cx, frame, null); - } - CallFrame calleeFrame = - initFrame( - cx, - calleeScope, - funThisObj, - funHomeObj, - stack, - sDbl, - stackTop + 2, - indexReg, - ifun, - callParentFrame); - if (op != Icode_TAIL_CALL) { - frame.savedStackTop = stackTop; - frame.savedCallOp = op; - } - frame = calleeFrame; - continue StateLoop; - } - } - - if (fun instanceof NativeContinuation) { - // Jump to the captured continuation - ContinuationJump cjump; - cjump = - new ContinuationJump( - (NativeContinuation) fun, frame); - - // continuation result is the first argument if any - // of continuation call - if (indexReg == 0) { - cjump.result = undefined; - } else { - cjump.result = stack[stackTop + 2]; - cjump.resultDbl = sDbl[stackTop + 2]; - } - - // Start the real unwind job - throwable = cjump; + } else if (callState instanceof StateContinue) { + frame = ((StateContinue) callState).frame; + continue StateLoop; + } else if (callState instanceof NewThrowable) { + throwable = ((NewThrowable) callState).throwable; break withoutExceptions; + } else { + Kit.codeBug(); + break; } - - if (fun instanceof IdFunctionObject) { - IdFunctionObject ifun = (IdFunctionObject) fun; - if (NativeContinuation.isContinuationConstructor(ifun)) { - frame.stack[stackTop] = - captureContinuation( - cx, frame.parentFrame, false); - continue Loop; - } - } - - cx.lastInterpreterFrame = frame; - frame.savedCallOp = op; - frame.savedStackTop = stackTop; - stack[stackTop] = - fun.call( - cx, - calleeScope, - funThisObj, - getArgsArray( - stack, sDbl, stackTop + 2, indexReg)); - - continue Loop; } case Token.NEW: { @@ -2988,6 +2784,228 @@ private static Object interpretLoop(Context cx, CallFrame frame, Object throwabl : ScriptRuntime.wrapNumber(interpreterResultDbl); } + private static final NewState doCallByteCode( + Context cx, + CallFrame frame, + boolean instructionCounting, + int op, + int stackTop, + int indexReg) { + Object[] stack = frame.stack; + double[] sDbl = frame.sDbl; + + if (instructionCounting) { + cx.instructionCount += INVOCATION_COST; + } + // stack change: function thisObj arg0 .. argN -> result + // indexReg: number of arguments + stackTop -= 1 + indexReg; + + // CALL generation ensures that fun and funThisObj + // are already Scriptable and Callable objects respectively + Callable fun = (Callable) stack[stackTop]; + Scriptable funThisObj = (Scriptable) stack[stackTop + 1]; + Scriptable funHomeObj = + (fun instanceof BaseFunction) ? ((BaseFunction) fun).getHomeObject() : null; + if (op == Icode_CALL_ON_SUPER) { + // funThisObj would have been the "super" object, which we + // used to lookup the function. Now that that's done, we + // discard it and invoke the function with the current + // "this". + funThisObj = frame.thisObj; + } + + if (op == Token.REF_CALL) { + Object[] outArgs = getArgsArray(stack, sDbl, stackTop + 2, indexReg); + stack[stackTop] = + ScriptRuntime.callRef( + fun, funThisObj, + outArgs, cx); + return new ContinueLoop(frame, stackTop, indexReg); + } + Scriptable calleeScope = frame.scope; + if (frame.useActivation) { + calleeScope = ScriptableObject.getTopLevelScope(frame.scope); + } + // Iteratively reduce known function types: arrows, lambdas, + // bound functions, call/apply, and no-such-method-handler in + // order to make a best-effort to keep them in this interpreter + // loop so continuations keep working. The loop initializer and + // condition are formulated so that they short-circuit the loop + // if the function is already an interpreted function, which + // should be the majority of cases. + for (; ; ) { + if (fun instanceof ArrowFunction) { + ArrowFunction afun = (ArrowFunction) fun; + fun = afun.getTargetFunction(); + funThisObj = afun.getCallThis(cx); + funHomeObj = afun.getBoundHomeObject(); + } else if (fun instanceof LambdaConstructor) { + break; + } else if (fun instanceof LambdaFunction) { + fun = ((LambdaFunction) fun).getTarget(); + } else if (fun instanceof BoundFunction) { + BoundFunction bfun = (BoundFunction) fun; + fun = bfun.getTargetFunction(); + funThisObj = bfun.getCallThis(cx, calleeScope); + Object[] boundArgs = bfun.getBoundArgs(); + int blen = boundArgs.length; + if (blen > 0) { + stack = frame.ensureStackLength(blen + stackTop + 2 + indexReg); + sDbl = frame.sDbl; + System.arraycopy(stack, stackTop + 2, stack, stackTop + 2 + blen, indexReg); + System.arraycopy(sDbl, stackTop + 2, sDbl, stackTop + 2 + blen, indexReg); + System.arraycopy(boundArgs, 0, stack, stackTop + 2, blen); + indexReg += blen; + } + } else if (fun instanceof IdFunctionObject) { + IdFunctionObject ifun = (IdFunctionObject) fun; + // Bug 405654 -- make the best effort to keep + // Function.apply and Function.call within this + // interpreter loop invocation + if (BaseFunction.isApplyOrCall(ifun)) { + // funThisObj becomes fun + fun = ScriptRuntime.getCallable(funThisObj); + // first arg becomes thisObj + funThisObj = getApplyThis(cx, stack, sDbl, stackTop + 2, indexReg, fun, frame); + if (BaseFunction.isApply(ifun)) { + // Apply: second argument after new "this" + // should be array-like + // and we'll spread its elements on the stack + Object[] callArgs = + indexReg < 2 + ? ScriptRuntime.emptyArgs + : ScriptRuntime.getApplyArguments(cx, stack[stackTop + 3]); + int alen = callArgs.length; + stack = frame.ensureStackLength(alen + stackTop + 2); + sDbl = frame.sDbl; + System.arraycopy(callArgs, 0, stack, stackTop + 2, alen); + indexReg = alen; + } else { + // Call: shift args left, starting from 2nd + if (indexReg > 0) { + if (indexReg > 1) { + System.arraycopy( + stack, stackTop + 3, stack, stackTop + 2, indexReg - 1); + System.arraycopy( + sDbl, stackTop + 3, sDbl, stackTop + 2, indexReg - 1); + } + indexReg--; + } + } + } else { + // Some other IdFunctionObject we don't know how to + // reduce. + break; + } + } else if (fun instanceof NoSuchMethodShim) { + NoSuchMethodShim nsmfun = (NoSuchMethodShim) fun; + // Bug 447697 -- make best effort to keep + // __noSuchMethod__ within this interpreter loop + // invocation. + stack = frame.ensureStackLength(stackTop + 4); + sDbl = frame.sDbl; + Object[] elements = getArgsArray(stack, sDbl, stackTop + 2, indexReg); + fun = nsmfun.noSuchMethodMethod; + stack[stackTop + 2] = nsmfun.methodName; + stack[stackTop + 3] = cx.newArray(calleeScope, elements); + indexReg = 2; + } else if (fun == null) { + throw ScriptRuntime.notFunctionError(null, null); + } else { + // Current function is something that we can't reduce + // further. + break; + } + } + + if (fun instanceof InterpretedFunction) { + InterpretedFunction ifun = (InterpretedFunction) fun; + if (frame.fnOrScript.securityDomain == ifun.securityDomain) { + CallFrame callParentFrame = frame; + if (op == Icode_TAIL_CALL) { + // In principle tail call can re-use the current + // frame and its stack arrays but it is hard to + // do properly. Any exceptions that can legally + // happen during frame re-initialization including + // StackOverflowException during innocent looking + // System.arraycopy may leave the current frame + // data corrupted leading to undefined behaviour + // in the catch code bellow that unwinds JS stack + // on exceptions. Then there is issue about frame + // release + // end exceptions there. + // To avoid frame allocation a released frame + // can be cached for re-use which would also benefit + // non-tail calls but it is not clear that this + // caching + // would gain in performance due to potentially + // bad interaction with GC. + callParentFrame = frame.parentFrame; + // Release the current frame. See Bug #344501 to see + // why + // it is being done here. + exitFrame(cx, frame, null); + } + CallFrame calleeFrame = + initFrame( + cx, + calleeScope, + funThisObj, + funHomeObj, + stack, + sDbl, + stackTop + 2, + indexReg, + ifun, + callParentFrame); + if (op != Icode_TAIL_CALL) { + frame.savedStackTop = stackTop; + frame.savedCallOp = op; + } + return new StateContinue(calleeFrame); + } + } + + if (fun instanceof NativeContinuation) { + // Jump to the captured continuation + ContinuationJump cjump; + cjump = new ContinuationJump((NativeContinuation) fun, frame); + + // continuation result is the first argument if any + // of continuation call + if (indexReg == 0) { + cjump.result = undefined; + } else { + cjump.result = stack[stackTop + 2]; + cjump.resultDbl = sDbl[stackTop + 2]; + } + + // Start the real unwind job + return new NewThrowable(cjump); + } + + if (fun instanceof IdFunctionObject) { + IdFunctionObject ifun = (IdFunctionObject) fun; + if (NativeContinuation.isContinuationConstructor(ifun)) { + frame.stack[stackTop] = captureContinuation(cx, frame.parentFrame, false); + return new ContinueLoop(frame, stackTop, indexReg); + } + } + + cx.lastInterpreterFrame = frame; + frame.savedCallOp = op; + frame.savedStackTop = stackTop; + stack[stackTop] = + fun.call( + cx, + calleeScope, + funThisObj, + getArgsArray(stack, sDbl, stackTop + 2, indexReg)); + + return new ContinueLoop(frame, stackTop, indexReg); + } + private static Scriptable getCurrentFrameHomeObject(CallFrame frame) { if (frame.scope instanceof NativeCall) { return ((NativeCall) frame.scope).getHomeObject();