2014-04-20 20:14:48 -06:00
|
|
|
/* Copyright (c) 2008-2014, Avian Contributors
|
2013-12-03 14:08:21 -06:00
|
|
|
|
|
|
|
Permission to use, copy, modify, and/or distribute this software
|
|
|
|
for any purpose with or without fee is hereby granted, provided
|
|
|
|
that the above copyright notice and this permission notice appear
|
|
|
|
in all copies.
|
|
|
|
|
|
|
|
There is NO WARRANTY for this software. See license.txt for
|
|
|
|
details. */
|
|
|
|
|
|
|
|
package java.lang.reflect;
|
|
|
|
|
2014-10-07 16:50:02 +04:00
|
|
|
import avian.Classes;
|
|
|
|
|
2013-12-03 14:08:21 -06:00
|
|
|
import java.util.ArrayList;
|
|
|
|
import java.util.List;
|
2014-10-07 16:50:02 +04:00
|
|
|
import java.util.LinkedList;
|
|
|
|
import java.util.Map;
|
|
|
|
import java.util.HashMap;
|
2013-12-03 14:08:21 -06:00
|
|
|
|
2014-10-04 22:17:49 +04:00
|
|
|
public class SignatureParser {
|
2013-12-03 14:08:21 -06:00
|
|
|
private final ClassLoader loader;
|
|
|
|
private final char[] array;
|
2014-10-08 03:46:52 +04:00
|
|
|
private final String signature;
|
2013-12-03 14:08:21 -06:00
|
|
|
private int offset;
|
|
|
|
private final Type type;
|
2014-10-07 16:50:02 +04:00
|
|
|
private final Map<String, TypeVariable> typeVariables;
|
|
|
|
|
|
|
|
public static Type parse(ClassLoader loader, String signature, Class declaringClass) {
|
|
|
|
return new SignatureParser(loader, signature, collectTypeVariables(declaringClass)).type;
|
|
|
|
}
|
2013-12-03 14:08:21 -06:00
|
|
|
|
2014-10-07 16:50:02 +04:00
|
|
|
private static Type parse(ClassLoader loader, String signature, Map<String, TypeVariable> typeVariables) {
|
|
|
|
return new SignatureParser(loader, signature, typeVariables).type;
|
2013-12-03 14:08:21 -06:00
|
|
|
}
|
|
|
|
|
2014-10-07 16:50:02 +04:00
|
|
|
private SignatureParser(ClassLoader loader, String signature, Map<String, TypeVariable> typeVariables) {
|
2013-12-03 14:08:21 -06:00
|
|
|
this.loader = loader;
|
2014-10-08 03:46:52 +04:00
|
|
|
this.signature = signature;
|
2013-12-03 14:08:21 -06:00
|
|
|
array = signature.toCharArray();
|
2014-10-07 16:50:02 +04:00
|
|
|
this.typeVariables = typeVariables;
|
2013-12-03 14:08:21 -06:00
|
|
|
type = parseType();
|
|
|
|
if (offset != array.length) {
|
|
|
|
throw new IllegalArgumentException("Extra characters after " + offset
|
|
|
|
+ ": " + signature);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private Type parseType() {
|
|
|
|
char c = array[offset++];
|
|
|
|
if (c == 'B') {
|
|
|
|
return Byte.TYPE;
|
|
|
|
} else if (c == 'C') {
|
|
|
|
return Character.TYPE;
|
|
|
|
} else if (c == 'D') {
|
|
|
|
return Double.TYPE;
|
|
|
|
} else if (c == 'F') {
|
|
|
|
return Float.TYPE;
|
|
|
|
} else if (c == 'I') {
|
|
|
|
return Integer.TYPE;
|
|
|
|
} else if (c == 'J') {
|
|
|
|
return Long.TYPE;
|
|
|
|
} else if (c == 'S') {
|
|
|
|
return Short.TYPE;
|
|
|
|
} else if (c == 'Z') {
|
|
|
|
return Boolean.TYPE;
|
2014-10-07 16:50:02 +04:00
|
|
|
} else if (c == 'T') {
|
2014-10-08 03:46:52 +04:00
|
|
|
int end = signature.indexOf(';', offset);
|
|
|
|
if (end < 0) {
|
|
|
|
throw new RuntimeException("No semicolon found while parsing signature");
|
2014-10-07 16:50:02 +04:00
|
|
|
}
|
2014-10-08 03:46:52 +04:00
|
|
|
Type res = typeVariables.get(new String(array, offset, end - offset));
|
|
|
|
offset = end + 1;
|
|
|
|
return res;
|
2013-12-03 14:08:21 -06:00
|
|
|
} else if (c != 'L') {
|
2014-10-08 03:46:52 +04:00
|
|
|
throw new IllegalArgumentException("Unexpected character: " + c + ", signature: " + new String(array, 0, array.length) + ", i = " + offset);
|
2013-12-03 14:08:21 -06:00
|
|
|
}
|
|
|
|
StringBuilder builder = new StringBuilder();
|
|
|
|
Type ownerType = null;
|
|
|
|
for (;;) {
|
|
|
|
for (;;) {
|
|
|
|
c = array[offset++];
|
|
|
|
if (c == ';' || c == '<') {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
builder.append(c == '/' ? '.' : c);
|
|
|
|
}
|
|
|
|
String rawTypeName = builder.toString();
|
|
|
|
Class<?> rawType;
|
|
|
|
try {
|
|
|
|
rawType = loader.loadClass(rawTypeName);
|
|
|
|
} catch (ClassNotFoundException e) {
|
|
|
|
throw new RuntimeException("Could not find class " + rawTypeName);
|
|
|
|
}
|
2014-10-04 22:17:49 +04:00
|
|
|
|
|
|
|
int lastDollar = rawTypeName.lastIndexOf('$');
|
2014-10-07 16:50:02 +04:00
|
|
|
if (lastDollar != -1 && ownerType == null) {
|
2014-10-04 22:17:49 +04:00
|
|
|
String ownerName = rawTypeName.substring(0, lastDollar);
|
|
|
|
try {
|
|
|
|
ownerType = loader.loadClass(ownerName);
|
|
|
|
} catch (ClassNotFoundException e) {
|
|
|
|
throw new RuntimeException("Could not find class " + ownerName);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-12-03 14:08:21 -06:00
|
|
|
if (c == ';') {
|
|
|
|
return rawType;
|
|
|
|
}
|
|
|
|
List<Type> args = new ArrayList<Type>();
|
|
|
|
while (array[offset] != '>') {
|
|
|
|
args.add(parseType());
|
|
|
|
}
|
|
|
|
++offset;
|
|
|
|
c = array[offset++];
|
|
|
|
ParameterizedType type = makeType(args.toArray(new Type[args.size()]), ownerType, rawType);
|
|
|
|
if (c == ';') {
|
|
|
|
return type;
|
|
|
|
}
|
|
|
|
if (c != '.') {
|
|
|
|
throw new RuntimeException("TODO");
|
|
|
|
}
|
|
|
|
ownerType = type;
|
|
|
|
builder.append("$");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private static String typeName(Type type) {
|
|
|
|
if (type instanceof Class) {
|
|
|
|
Class<?> clazz = (Class<?>) type;
|
|
|
|
return clazz.getName();
|
|
|
|
}
|
|
|
|
return type.toString();
|
|
|
|
}
|
|
|
|
|
|
|
|
private static ParameterizedType makeType(final Type[] args, final Type owner, final Type raw) {
|
|
|
|
return new ParameterizedType() {
|
|
|
|
@Override
|
|
|
|
public Type getRawType() {
|
|
|
|
return raw;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public Type getOwnerType() {
|
|
|
|
return owner;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public Type[] getActualTypeArguments() {
|
|
|
|
return args;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public String toString() {
|
|
|
|
StringBuilder builder = new StringBuilder();
|
|
|
|
builder.append(typeName(raw));
|
|
|
|
builder.append('<');
|
|
|
|
String sep = "";
|
|
|
|
for (Type t : args) {
|
|
|
|
builder.append(sep).append(typeName(t));
|
|
|
|
sep = ", ";
|
|
|
|
}
|
|
|
|
builder.append('>');
|
|
|
|
return builder.toString();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
2014-10-07 16:50:02 +04:00
|
|
|
|
|
|
|
private static Map<String, TypeVariable> collectTypeVariables(Class clz) {
|
|
|
|
Map<String, TypeVariable> varsMap = new HashMap<String, TypeVariable>();
|
|
|
|
LinkedList<Class> classList = new LinkedList<Class>();
|
|
|
|
for (Class c = clz; c != null; c = c.getDeclaringClass()) {
|
|
|
|
classList.addFirst(c);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (Class cur : classList) {
|
2014-10-08 03:46:52 +04:00
|
|
|
final LinkedList<TypeVariableImpl> varsList = new LinkedList<TypeVariableImpl>();
|
2014-10-07 16:50:02 +04:00
|
|
|
if (cur.vmClass.addendum != null && cur.vmClass.addendum.signature != null) {
|
|
|
|
String signature = Classes.toString((byte[]) cur.vmClass.addendum.signature);
|
|
|
|
final char[] signChars = signature.toCharArray();
|
2014-10-08 03:46:52 +04:00
|
|
|
try {
|
|
|
|
int i = 0;
|
|
|
|
if (signChars[i] == '<') {
|
2014-10-07 16:50:02 +04:00
|
|
|
i++;
|
2014-10-08 03:46:52 +04:00
|
|
|
do {
|
|
|
|
final int colon = signature.indexOf(':', i);
|
|
|
|
if (colon < 0 || colon + 1 == signChars.length) {
|
|
|
|
throw new RuntimeException("Can't find ':' in the signature " + signature + " starting from " + i);
|
|
|
|
}
|
|
|
|
String typeVarName = new String(signChars, i, colon - i);
|
|
|
|
i = colon + 1;
|
|
|
|
|
|
|
|
int start = i;
|
|
|
|
int angles = 0;
|
|
|
|
while (angles > 0 || signChars[i] != ';') {
|
|
|
|
if (signChars[i] == '<') angles ++;
|
|
|
|
else if (signChars[i] == '>') angles --;
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
String typeName = new String(signChars, start, i - start + 1);
|
|
|
|
final Type baseType = SignatureParser.parse(cur.vmClass.loader, typeName, varsMap);
|
|
|
|
|
|
|
|
TypeVariableImpl tv = new TypeVariableImpl(typeVarName, baseType);
|
|
|
|
varsList.add(tv);
|
2014-10-07 16:50:02 +04:00
|
|
|
|
2014-10-08 03:46:52 +04:00
|
|
|
i++;
|
|
|
|
} while (signChars[i] != '>');
|
2014-10-07 16:50:02 +04:00
|
|
|
|
2014-10-08 03:46:52 +04:00
|
|
|
}
|
|
|
|
} catch (IndexOutOfBoundsException e) {
|
|
|
|
throw new RuntimeException("Signature of " + cur + " is broken (" + signature + ") and can't be parsed", e);
|
2014-10-07 16:50:02 +04:00
|
|
|
}
|
|
|
|
}
|
2014-10-08 03:46:52 +04:00
|
|
|
for (TypeVariableImpl tv : varsList) {
|
|
|
|
tv.setVars(varsList);
|
2014-10-07 16:50:02 +04:00
|
|
|
varsMap.put(tv.getName(), tv);
|
|
|
|
}
|
|
|
|
cur = cur.getDeclaringClass();
|
|
|
|
};
|
|
|
|
return varsMap;
|
|
|
|
}
|
|
|
|
|
|
|
|
private static class TypeVariableImpl implements TypeVariable {
|
|
|
|
private String name;
|
|
|
|
private Type baseType;
|
2014-10-08 03:46:52 +04:00
|
|
|
private TypeVariableImpl[] vars;
|
2014-10-07 16:50:02 +04:00
|
|
|
|
|
|
|
public Type[] getBounds() {
|
|
|
|
return new Type[] { baseType };
|
|
|
|
}
|
|
|
|
|
|
|
|
public GenericDeclaration getGenericDeclaration() {
|
|
|
|
return new GenericDeclaration() {
|
|
|
|
public TypeVariable<?>[] getTypeParameters() {
|
|
|
|
return vars;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
public String getName() {
|
|
|
|
return name;
|
|
|
|
}
|
|
|
|
|
|
|
|
TypeVariableImpl(String name, Type baseType) {
|
|
|
|
this.name = name;
|
|
|
|
this.baseType = baseType;
|
|
|
|
}
|
|
|
|
|
2014-10-08 03:46:52 +04:00
|
|
|
void setVars(List<TypeVariableImpl> vars) {
|
|
|
|
this.vars = new TypeVariableImpl[vars.size()];
|
2014-10-07 16:50:02 +04:00
|
|
|
vars.toArray(this.vars);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public String toString() {
|
|
|
|
return name;
|
|
|
|
}
|
|
|
|
}
|
2013-12-03 14:08:21 -06:00
|
|
|
}
|