# HG changeset patch # User attila # Date 1401963345 -7200 # Node ID b5c31bfe1496937c3ff7adfabc37d16bddb01a87 # Parent 82c9c92e37336bc37f2504465a4e782a8fea9991 8046014: MultiGlobalCompiledScript should cache :createProgramFunction handle Reviewed-by: lagergren, sundar diff -r 82c9c92e3733 -r b5c31bfe1496 nashorn/src/jdk/nashorn/internal/runtime/Context.java --- a/nashorn/src/jdk/nashorn/internal/runtime/Context.java Wed Jun 04 20:43:37 2014 +0200 +++ b/nashorn/src/jdk/nashorn/internal/runtime/Context.java Thu Jun 05 12:15:45 2014 +0200 @@ -37,10 +37,11 @@ import java.io.IOException; import java.io.PrintWriter; import java.lang.invoke.MethodHandle; +import java.lang.invoke.MethodHandles; +import java.lang.invoke.MethodType; import java.lang.ref.ReferenceQueue; import java.lang.ref.SoftReference; import java.lang.reflect.Field; -import java.lang.reflect.InvocationTargetException; import java.lang.reflect.Modifier; import java.net.MalformedURLException; import java.net.URL; @@ -120,6 +121,9 @@ private static final String LOAD_FX = "fx:"; private static final String LOAD_NASHORN = "nashorn:"; + private static MethodHandles.Lookup LOOKUP = MethodHandles.lookup(); + private static MethodType CREATE_PROGRAM_FUNCTION_TYPE = MethodType.methodType(ScriptFunction.class, ScriptObject.class); + /* Force DebuggerSupport to be loaded. */ static { DebuggerSupport.FORCELOAD = true; @@ -538,11 +542,12 @@ */ public MultiGlobalCompiledScript compileScript(final Source source) { final Class clazz = compile(source, this.errors, this._strict); + final MethodHandle createProgramFunctionHandle = getCreateProgramFunctionHandle(clazz); return new MultiGlobalCompiledScript() { @Override public ScriptFunction getFunction(final Global newGlobal) { - return getProgramFunction(clazz, newGlobal); + return invokeCreateProgramFunctionHandle(createProgramFunctionHandle, newGlobal); } }; } @@ -1009,15 +1014,24 @@ } private static ScriptFunction getProgramFunction(final Class script, final ScriptObject scope) { - if (script == null) { - return null; - } + return invokeCreateProgramFunctionHandle(getCreateProgramFunctionHandle(script), scope); + } + private static MethodHandle getCreateProgramFunctionHandle(final Class script) { try { - return (ScriptFunction)script.getMethod(CREATE_PROGRAM_FUNCTION.symbolName(), ScriptObject.class).invoke(null, scope); - } catch (IllegalAccessException | IllegalArgumentException | InvocationTargetException | NoSuchMethodException - | SecurityException e) { - throw new RuntimeException("Failed to create a program function for " + script.getName(), e); + return LOOKUP.findStatic(script, CREATE_PROGRAM_FUNCTION.symbolName(), CREATE_PROGRAM_FUNCTION_TYPE); + } catch (NoSuchMethodException | IllegalAccessException e) { + throw new AssertionError("Failed to retrieve a handle for the program function for " + script.getName(), e); + } + } + + private static ScriptFunction invokeCreateProgramFunctionHandle(final MethodHandle createProgramFunctionHandle, final ScriptObject scope) { + try { + return (ScriptFunction)createProgramFunctionHandle.invokeExact(scope); + } catch (final RuntimeException|Error e) { + throw e; + } catch (final Throwable t) { + throw new AssertionError("Failed to create a program function", t); } }