001/*
002 * Copyright (c) 2012-2021 Institut National des Sciences Appliquées de Lyon (INSA Lyon) and others
003 *
004 * This program and the accompanying materials are made available under the
005 * terms of the Eclipse Public License 2.0 which is available at
006 * http://www.eclipse.org/legal/epl-2.0.
007 *
008 * SPDX-License-Identifier: EPL-2.0
009 */
010
011package org.eclipse.golo.runtime;
012
013import gololang.DynamicObject;
014
015import java.lang.invoke.*;
016import java.lang.invoke.MethodHandles.Lookup;
017import java.util.Arrays;
018import java.util.HashSet;
019import java.util.WeakHashMap;
020
021import static java.lang.invoke.MethodHandles.*;
022import static java.lang.invoke.MethodType.methodType;
023
024
025public final class MethodInvocationSupport {
026
027  private MethodInvocationSupport() {
028    throw new UnsupportedOperationException("Don't instantiate invokedynamic bootstrap class");
029  }
030
031  /*
032   * This code is heavily inspired from the inline cache construction from
033   * Remi Forax's JSR292 cookbooks.
034   */
035
036  static final class InlineCache extends MutableCallSite {
037
038    static final int MEGAMORPHIC_THRESHOLD = 5;
039
040    final Lookup callerLookup;
041    final String name;
042    final boolean nullSafeGuarded;
043    final String[] argumentNames;
044
045    int depth = 0;
046    MethodHandle resetFallback;
047    WeakHashMap<Class<?>, MethodHandle> vtable;
048
049    InlineCache(Lookup callerLookup, String name, MethodType type, boolean nullSafeGuarded, String... argumentNames) {
050      super(type);
051      this.callerLookup = callerLookup;
052      this.name = name;
053      this.nullSafeGuarded = nullSafeGuarded;
054      this.argumentNames = argumentNames;
055    }
056
057    boolean isMegaMorphic() {
058      return depth > MEGAMORPHIC_THRESHOLD;
059    }
060
061    public MethodInvocation toMethodInvocation(Object[] args) {
062      return new MethodInvocation(name, type(), args, argumentNames);
063    }
064  }
065
066  private static final MethodHandle CLASS_GUARD;
067  private static final MethodHandle FALLBACK;
068  private static final MethodHandle RESET_FALLBACK;
069  private static final MethodHandle VTABLE_LOOKUP;
070
071  private static final MethodHandle OVERLOADED_GUARD_GENERIC;
072  private static final MethodHandle OVERLOADED_GUARD_1;
073  private static final MethodHandle OVERLOADED_GUARD_2;
074  private static final MethodHandle OVERLOADED_GUARD_3;
075  private static final MethodHandle OVERLOADED_GUARD_4;
076
077
078  private static final HashSet<String> DYNAMIC_OBJECT_RESERVED_METHOD_NAMES = new HashSet<String>() {
079    {
080      add("get");
081      add("define");
082      add("undefine");
083      add("mixin");
084      add("copy");
085      add("freeze");
086      add("properties");
087      add("invoker");
088      add("hasMethod");
089      add("fallback");
090      add("hasKind");
091      add("sameKind");
092      add("isFrozen");
093    }
094  };
095
096  static {
097    try {
098      Lookup lookup = MethodHandles.lookup();
099
100      CLASS_GUARD = lookup.findStatic(
101          MethodInvocationSupport.class,
102          "classGuard",
103          methodType(boolean.class, Class.class, Object.class));
104
105      FALLBACK = lookup.findStatic(
106          MethodInvocationSupport.class,
107          "fallback",
108          methodType(Object.class, InlineCache.class, Object[].class));
109
110      RESET_FALLBACK = lookup.findStatic(
111          MethodInvocationSupport.class,
112          "resetFallback",
113          methodType(Object.class, InlineCache.class, Object[].class));
114
115      VTABLE_LOOKUP = lookup.findStatic(
116          MethodInvocationSupport.class,
117          "vtableLookup",
118          methodType(MethodHandle.class, InlineCache.class, Object[].class));
119
120      OVERLOADED_GUARD_GENERIC = lookup.findStatic(
121          MethodInvocationSupport.class,
122          "overloadedGuard_generic",
123          methodType(boolean.class, Class[].class, Object[].class));
124
125      OVERLOADED_GUARD_1 = lookup.findStatic(
126          MethodInvocationSupport.class,
127          "overloadedGuard_1",
128          methodType(boolean.class, Class.class, Class.class, Object.class, Object.class));
129
130      OVERLOADED_GUARD_2 = lookup.findStatic(
131          MethodInvocationSupport.class,
132          "overloadedGuard_2",
133          methodType(boolean.class, Class.class, Class.class, Class.class, Object.class, Object.class, Object.class));
134
135      OVERLOADED_GUARD_3 = lookup.findStatic(
136          MethodInvocationSupport.class,
137          "overloadedGuard_3",
138          methodType(boolean.class, Class.class, Class.class, Class.class, Class.class, Object.class, Object.class, Object.class, Object.class));
139
140      OVERLOADED_GUARD_4 = lookup.findStatic(
141          MethodInvocationSupport.class,
142          "overloadedGuard_4",
143          methodType(boolean.class, Class.class, Class.class, Class.class, Class.class, Class.class, Object.class, Object.class, Object.class, Object.class, Object.class));
144
145    } catch (NoSuchMethodException | IllegalAccessException e) {
146      throw new Error("Could not bootstrap the required method handles", e);
147    }
148  }
149
150  public static CallSite bootstrap(Lookup caller, String name, MethodType type, Object... bsmArgs) {
151    boolean nullSafeGuarded = ((int) bsmArgs[0]) == 1;
152    String[] argumentNames = new String[bsmArgs.length - 1];
153    for (int i = 0; i < bsmArgs.length - 1; i++) {
154      argumentNames[i] = (String) bsmArgs[i + 1];
155    }
156    InlineCache callSite = new InlineCache(caller, name, type, nullSafeGuarded, argumentNames);
157    MethodHandle fallbackHandle = FALLBACK
158        .bindTo(callSite)
159        .asCollector(Object[].class, type.parameterCount())
160        .asType(type);
161    callSite.resetFallback = RESET_FALLBACK
162        .bindTo(callSite)
163        .asCollector(Object[].class, type.parameterCount())
164        .asType(type);
165    callSite.setTarget(fallbackHandle);
166    return callSite;
167  }
168
169  public static boolean classGuard(Class<?> expected, Object receiver) {
170    return receiver.getClass() == expected;
171  }
172
173  public static boolean overloadedGuard_generic(Class<?>[] types, Object[] arguments) {
174    if (arguments[0].getClass() != types[0]) {
175      return false;
176    }
177    for (int i = 1; i < types.length; i++) {
178      if ((arguments[i] != null) && (arguments[i].getClass() != types[i])) {
179        return false;
180      }
181    }
182    return true;
183  }
184
185  public static boolean overloadedGuard_1(Class<?> t1, Class<?> t2, Object receiver, Object arg) {
186    return receiver.getClass() == t1
187      && (arg == null || arg.getClass() == t2);
188  }
189
190  public static boolean overloadedGuard_2(Class<?> t1, Class<?> t2, Class<?> t3, Object receiver, Object arg1, Object arg2) {
191    return receiver.getClass() == t1
192      && (arg1 == null || arg1.getClass() == t2)
193      && (arg2 == null || arg2.getClass() == t3);
194  }
195
196  public static boolean overloadedGuard_3(Class<?> t1, Class<?> t2, Class<?> t3, Class<?> t4, Object receiver, Object arg1, Object arg2, Object arg3) {
197    return receiver.getClass() == t1
198      && (arg1 == null || arg1.getClass() == t2)
199      && (arg2 == null || arg2.getClass() == t3)
200      && (arg3 == null || arg3.getClass() == t4);
201  }
202
203  public static boolean overloadedGuard_4(Class<?> t1, Class<?> t2, Class<?> t3, Class<?> t4, Class<?> t5, Object receiver, Object arg1, Object arg2, Object arg3, Object arg4) {
204    return receiver.getClass() == t1
205      && (arg1 == null || arg1.getClass() == t2)
206      && (arg2 == null || arg2.getClass() == t3)
207      && (arg3 == null || arg3.getClass() == t4)
208      && (arg4 == null || arg4.getClass() == t5);
209  }
210
211  public static MethodHandle vtableLookup(InlineCache inlineCache, Object[] args) {
212    Class<?> receiverClass = args[0].getClass();
213    return inlineCache.vtable.computeIfAbsent(receiverClass, k -> lookupTarget(receiverClass, inlineCache, args));
214  }
215
216  private static MethodHandle lookupTarget(Class<?> receiverClass, InlineCache inlineCache, Object[] args) {
217    MethodInvocation invocation = inlineCache.toMethodInvocation(args);
218    if (receiverClass.isArray()) {
219      return new ArrayMethodFinder(invocation, inlineCache.callerLookup).find();
220    }
221    if (isCallOnDynamicObject(inlineCache, args[0])) {
222      DynamicObject dynamicObject = (DynamicObject) args[0];
223      return dynamicObject.invoker(inlineCache.name, inlineCache.type());
224    } else {
225      return findTarget(invocation, inlineCache);
226    }
227  }
228
229  public static Object resetFallback(InlineCache inlineCache, Object[] args) throws Throwable {
230    inlineCache.depth = 0;
231    return fallback(inlineCache, args);
232  }
233
234  public static Object fallback(InlineCache inlineCache, Object[] args) throws Throwable {
235
236    if (inlineCache.isMegaMorphic()) {
237      return installVTableDispatch(inlineCache, args);
238    }
239
240    if (args[0] == null) {
241      if (shouldReturnNull(inlineCache, args[0])) {
242        return null;
243      } else {
244        throw new NullPointerException("On method: "
245            + inlineCache.name + " " + inlineCache.type().dropParameterTypes(0, 1));
246      }
247    }
248
249    Class<?> receiverClass = args[0].getClass();
250    MethodHandle target = lookupTarget(receiverClass, inlineCache, args);
251
252    if (target == null) {
253      // TODO: extract method to look for a `fallback` method on the receiver
254      InlineCache fallbackCallSite = new InlineCache(
255          inlineCache.callerLookup,
256          "fallback",
257          methodType(Object.class, Object.class, Object.class, Object[].class),
258          false);
259      Object[] fallbackArgs = {
260        args[0],
261        inlineCache.name,
262        Arrays.copyOfRange(args, 1, args.length)
263      };
264      target = lookupTarget(receiverClass, fallbackCallSite, fallbackArgs);
265      if (target != null) {
266        return fallback(fallbackCallSite, fallbackArgs);
267      } else {
268        throw new NoSuchMethodError(receiverClass + "::" + inlineCache.name);
269      }
270    }
271
272    MethodHandle guard = CLASS_GUARD.bindTo(receiverClass);
273    MethodHandle fallback = inlineCache.getTarget();
274    MethodHandle root = guardWithTest(guard, target, fallback);
275    if (inlineCache.nullSafeGuarded) {
276      root = makeNullSafeGuarded(root);
277    }
278    inlineCache.setTarget(root);
279    inlineCache.depth += 1;
280    return target.invokeWithArguments(args);
281  }
282
283  private static MethodHandle makeNullSafeGuarded(MethodHandle root) {
284    MethodHandle catchThenNull = dropArguments(constant(Object.class, null), 0, NullPointerException.class);
285    root = catchException(root, NullPointerException.class, catchThenNull);
286    return root;
287  }
288
289  private static boolean shouldReturnNull(InlineCache inlineCache, Object arg) {
290    return (arg == null) && inlineCache.nullSafeGuarded;
291  }
292
293  private static Object installVTableDispatch(InlineCache inlineCache, Object[] args) throws Throwable {
294    if (inlineCache.vtable == null) {
295      inlineCache.vtable = new WeakHashMap<>();
296    }
297    MethodHandle lookup = VTABLE_LOOKUP
298        .bindTo(inlineCache)
299        .asCollector(Object[].class, args.length);
300    MethodHandle exactInvoker = exactInvoker(inlineCache.type());
301    MethodHandle vtableTarget = foldArguments(exactInvoker, lookup);
302    if (inlineCache.nullSafeGuarded) {
303      vtableTarget = makeNullSafeGuarded(vtableTarget);
304    }
305    inlineCache.setTarget(vtableTarget);
306    if (shouldReturnNull(inlineCache, args[0])) {
307      return null;
308    }
309    return vtableTarget.invokeWithArguments(args);
310  }
311
312  private static boolean isCallOnDynamicObject(InlineCache inlineCache, Object arg) {
313    return (arg instanceof DynamicObject)
314      && !DYNAMIC_OBJECT_RESERVED_METHOD_NAMES.contains(inlineCache.name)
315      && (!"toString".equals(inlineCache.name) || ((DynamicObject) arg).hasMethod("toString"));
316  }
317
318  private static MethodHandle guardOnOverloaded(MethodHandle target, MethodInvocation invocation, MethodHandle reset) {
319    Object[] args = invocation.arguments();
320    Class<?>[] types = new Class<?>[args.length];
321    for (int i = 0; i < types.length; i++) {
322      types[i] = (args[i] == null) ? Object.class : args[i].getClass();
323    }
324    MethodHandle guard;
325    switch (args.length) {
326      case 2:
327        guard = insertArguments(OVERLOADED_GUARD_1, 0, types[0], types[1]);
328        break;
329      case 3:
330        guard = insertArguments(OVERLOADED_GUARD_2, 0, types[0], types[1], types[2]);
331        break;
332      case 4:
333        guard = insertArguments(OVERLOADED_GUARD_3, 0, types[0], types[1], types[2], types[3]);
334        break;
335      case 5:
336        guard = insertArguments(OVERLOADED_GUARD_4, 0, types[0], types[1], types[2], types[3], types[4]);
337        break;
338      default:
339        guard = OVERLOADED_GUARD_GENERIC.bindTo(types).asCollector(Object[].class, types.length);
340    }
341    return guardWithTest(guard, target, reset);
342  }
343
344  private static MethodHandle findTarget(MethodInvocation invocation, InlineCache inlineCache) {
345    MethodHandle target;
346    Lookup lookup = inlineCache.callerLookup;
347
348    RegularMethodFinder regularMethodFinder = new RegularMethodFinder(invocation, lookup);
349    target = regularMethodFinder.find();
350    if (target != null) {
351      if (regularMethodFinder.isOverloaded()) {
352        return guardOnOverloaded(target, invocation, inlineCache.resetFallback);
353      }
354      return target;
355    }
356
357    target = new PropertyMethodFinder(invocation, lookup).find();
358    if (target != null) {
359      return target;
360    }
361
362    target = new AugmentationMethodFinder(invocation, lookup).find();
363    if (target != null) {
364      return target;
365    }
366    return null;
367  }
368}