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 java.lang.invoke.CallSite;
014import java.lang.invoke.ConstantCallSite;
015import java.lang.invoke.MethodHandles.Lookup;
016import java.lang.invoke.MethodType;
017
018import static gololang.Messages.message;
019import static java.lang.invoke.MethodHandles.constant;
020import static org.eclipse.golo.runtime.Module.imports;
021
022public final class ClassReferenceSupport {
023
024  private ClassReferenceSupport() {
025    throw new UnsupportedOperationException("Don't instantiate invokedynamic bootstrap class");
026  }
027
028  public static CallSite bootstrap(Lookup caller, String name, MethodType type) throws ClassNotFoundException {
029    String className = name.replaceAll("#", "\\.");
030    Class<?> callerClass = caller.lookupClass();
031    ClassLoader classLoader = callerClass.getClassLoader();
032
033    Class<?> classRef = tryLoadingFromPrimitiveType(className);
034    if (classRef != null) {
035      return createCallSite(classRef);
036    }
037    classRef = tryLoadingFromName(className, classLoader, callerClass.getName());
038    if (classRef != null) {
039      return createCallSite(classRef);
040    }
041    classRef = tryLoadingFromImports(className, callerClass, classLoader);
042    if (classRef != null) {
043      return createCallSite(classRef);
044    }
045    throw new ClassNotFoundException(message("class_not_resolved", className));
046  }
047
048  private static Class<?> tryLoadingFromName(String name, ClassLoader classLoader, String callerName) {
049    try {
050      return Class.forName(name, true, classLoader);
051    } catch (ClassNotFoundException e) {
052      Warnings.unavailableClass(name, callerName);
053      return null;
054    }
055  }
056
057  private static Class<?> tryLoadingFromImports(String className, Class<?> callerClass, ClassLoader classLoader) {
058    for (String importedClassName : imports(callerClass)) {
059      Class<?> classRef = tryLoadingFromName(importedClassName + "." + className, classLoader, callerClass.getName());
060      if (classRef != null) {
061        return classRef;
062      } else {
063        if (importedClassName.endsWith(className)) {
064          classRef = tryLoadingFromName(importedClassName, classLoader, callerClass.getName());
065          if (classRef != null) {
066            return classRef;
067          }
068        }
069      }
070    }
071    return null;
072  }
073
074  private static Class<?> tryLoadingFromPrimitiveType(String name) {
075    switch (name) {
076      case "byte"    : return byte.class;
077      case "char"    : return char.class;
078      case "int"     : return int.class;
079      case "long"    : return long.class;
080      case "double"  : return double.class;
081      case "short"   : return short.class;
082      case "float"   : return float.class;
083      case "boolean" : return boolean.class;
084      default: return null;
085    }
086  }
087
088  private static CallSite createCallSite(Class<?> classRef) {
089    return new ConstantCallSite(constant(Class.class, classRef));
090  }
091
092}