2015-03-13 12:52:59 -06:00
|
|
|
/* Copyright (c) 2008-2015, Avian Contributors
|
2010-09-10 15:05:29 -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 avian;
|
|
|
|
|
|
|
|
import static avian.Stream.read1;
|
|
|
|
import static avian.Stream.read2;
|
|
|
|
|
2014-02-26 13:39:05 -07:00
|
|
|
import java.net.URL;
|
|
|
|
import java.net.MalformedURLException;
|
|
|
|
import java.security.CodeSource;
|
|
|
|
import java.security.AllPermission;
|
|
|
|
import java.security.Permissions;
|
|
|
|
import java.security.ProtectionDomain;
|
|
|
|
import java.security.cert.Certificate;
|
2010-09-10 15:05:29 -06:00
|
|
|
import java.lang.reflect.Modifier;
|
|
|
|
import java.lang.reflect.Method;
|
|
|
|
import java.lang.reflect.Field;
|
2013-02-21 15:37:17 -07:00
|
|
|
import java.lang.reflect.Proxy;
|
|
|
|
import java.lang.annotation.Annotation;
|
2010-09-10 15:05:29 -06:00
|
|
|
import java.io.InputStream;
|
|
|
|
import java.io.ByteArrayInputStream;
|
|
|
|
import java.io.IOException;
|
|
|
|
|
|
|
|
public class Classes {
|
|
|
|
private static final int LinkFlag = 1 << 8;
|
|
|
|
|
|
|
|
public static native VMClass defineVMClass
|
|
|
|
(ClassLoader loader, byte[] b, int offset, int length);
|
|
|
|
|
|
|
|
public static native VMClass primitiveClass(char name);
|
|
|
|
|
|
|
|
public static native void initialize(VMClass vmClass);
|
2014-12-29 10:00:51 -07:00
|
|
|
|
2010-09-10 15:05:29 -06:00
|
|
|
public static native boolean isAssignableFrom(VMClass a, VMClass b);
|
|
|
|
|
|
|
|
public static native VMClass getVMClass(Object o);
|
|
|
|
|
2014-08-28 16:59:06 -06:00
|
|
|
public static native VMClass toVMClass(Class c);
|
|
|
|
|
2010-09-10 15:05:29 -06:00
|
|
|
private static native VMClass resolveVMClass(ClassLoader loader, byte[] spec)
|
|
|
|
throws ClassNotFoundException;
|
|
|
|
|
|
|
|
private static VMClass loadVMClass(ClassLoader loader,
|
|
|
|
byte[] nameBytes, int offset, int length)
|
|
|
|
{
|
|
|
|
byte[] spec = new byte[length + 1];
|
|
|
|
System.arraycopy(nameBytes, offset, spec, 0, length);
|
|
|
|
|
|
|
|
try {
|
|
|
|
VMClass c = resolveVMClass(loader, spec);
|
|
|
|
if (c == null) {
|
|
|
|
throw new NoClassDefFoundError();
|
|
|
|
}
|
|
|
|
return c;
|
|
|
|
} catch (ClassNotFoundException e) {
|
|
|
|
NoClassDefFoundError error = new NoClassDefFoundError
|
2011-03-26 23:24:48 -06:00
|
|
|
(new String(nameBytes, offset, length));
|
2010-09-10 15:05:29 -06:00
|
|
|
error.initCause(e);
|
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private static Object parseAnnotationValue(ClassLoader loader,
|
|
|
|
Object pool,
|
|
|
|
InputStream in)
|
|
|
|
throws IOException
|
|
|
|
{
|
|
|
|
switch (read1(in)) {
|
|
|
|
case 'Z':
|
|
|
|
return Boolean.valueOf(Singleton.getInt(pool, read2(in) - 1) != 0);
|
|
|
|
|
|
|
|
case 'B':
|
|
|
|
return Byte.valueOf((byte) Singleton.getInt(pool, read2(in) - 1));
|
|
|
|
|
|
|
|
case 'C':
|
|
|
|
return Character.valueOf((char) Singleton.getInt(pool, read2(in) - 1));
|
|
|
|
|
|
|
|
case 'S':
|
|
|
|
return Short.valueOf((short) Singleton.getInt(pool, read2(in) - 1));
|
|
|
|
|
|
|
|
case 'I':
|
|
|
|
return Integer.valueOf(Singleton.getInt(pool, read2(in) - 1));
|
|
|
|
|
|
|
|
case 'F':
|
|
|
|
return Float.valueOf
|
|
|
|
(Float.intBitsToFloat(Singleton.getInt(pool, read2(in) - 1)));
|
|
|
|
|
|
|
|
case 'J': {
|
|
|
|
return Long.valueOf(Singleton.getLong(pool, read2(in) - 1));
|
|
|
|
}
|
|
|
|
|
|
|
|
case 'D': {
|
|
|
|
return Double.valueOf
|
|
|
|
(Double.longBitsToDouble(Singleton.getLong(pool, read2(in) - 1)));
|
|
|
|
}
|
|
|
|
|
|
|
|
case 's': {
|
|
|
|
byte[] data = (byte[]) Singleton.getObject(pool, read2(in) - 1);
|
|
|
|
|
2011-03-26 23:24:48 -06:00
|
|
|
return new String(data, 0, data.length - 1);
|
2010-09-10 15:05:29 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
case 'e': {
|
|
|
|
byte[] typeName = (byte[]) Singleton.getObject(pool, read2(in) - 1);
|
|
|
|
byte[] name = (byte[]) Singleton.getObject(pool, read2(in) - 1);
|
|
|
|
|
|
|
|
return Enum.valueOf
|
2010-09-14 10:49:41 -06:00
|
|
|
(SystemClassLoader.getClass
|
|
|
|
(loadVMClass(loader, typeName, 1, typeName.length - 3)),
|
2011-03-26 23:24:48 -06:00
|
|
|
new String(name, 0, name.length - 1));
|
2010-09-10 15:05:29 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
case 'c':{
|
|
|
|
byte[] name = (byte[]) Singleton.getObject(pool, read2(in) - 1);
|
|
|
|
|
2010-09-25 15:51:30 -06:00
|
|
|
return SystemClassLoader.getClass
|
|
|
|
(loadVMClass(loader, name, 1, name.length - 3));
|
2010-09-10 15:05:29 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
case '@':
|
2013-11-26 14:15:16 -06:00
|
|
|
return getAnnotation(loader, parseAnnotation(loader, pool, in));
|
2010-09-10 15:05:29 -06:00
|
|
|
|
|
|
|
case '[': {
|
|
|
|
Object[] array = new Object[read2(in)];
|
|
|
|
for (int i = 0; i < array.length; ++i) {
|
|
|
|
array[i] = parseAnnotationValue(loader, pool, in);
|
|
|
|
}
|
|
|
|
return array;
|
2014-12-29 10:00:51 -07:00
|
|
|
}
|
2010-09-10 15:05:29 -06:00
|
|
|
|
|
|
|
default: throw new AssertionError();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private static Object[] parseAnnotation(ClassLoader loader,
|
|
|
|
Object pool,
|
|
|
|
InputStream in)
|
|
|
|
throws IOException
|
|
|
|
{
|
|
|
|
byte[] typeName = (byte[]) Singleton.getObject(pool, read2(in) - 1);
|
|
|
|
Object[] annotation = new Object[(read2(in) + 1) * 2];
|
2010-09-14 10:49:41 -06:00
|
|
|
annotation[1] = SystemClassLoader.getClass
|
2010-09-10 15:05:29 -06:00
|
|
|
(loadVMClass(loader, typeName, 1, typeName.length - 3));
|
|
|
|
|
|
|
|
for (int i = 2; i < annotation.length; i += 2) {
|
|
|
|
byte[] name = (byte[]) Singleton.getObject(pool, read2(in) - 1);
|
2011-03-26 23:24:48 -06:00
|
|
|
annotation[i] = new String(name, 0, name.length - 1);
|
2010-09-10 15:05:29 -06:00
|
|
|
annotation[i + 1] = parseAnnotationValue(loader, pool, in);
|
|
|
|
}
|
|
|
|
|
|
|
|
return annotation;
|
|
|
|
}
|
|
|
|
|
|
|
|
private static Object[] parseAnnotationTable(ClassLoader loader,
|
|
|
|
Object pool,
|
|
|
|
InputStream in)
|
|
|
|
throws IOException
|
|
|
|
{
|
|
|
|
Object[] table = new Object[read2(in)];
|
|
|
|
for (int i = 0; i < table.length; ++i) {
|
|
|
|
table[i] = parseAnnotation(loader, pool, in);
|
|
|
|
}
|
|
|
|
return table;
|
|
|
|
}
|
|
|
|
|
|
|
|
private static void parseAnnotationTable(ClassLoader loader,
|
|
|
|
Addendum addendum)
|
|
|
|
{
|
|
|
|
if (addendum != null && addendum.annotationTable instanceof byte[]) {
|
|
|
|
try {
|
|
|
|
addendum.annotationTable = parseAnnotationTable
|
|
|
|
(loader, addendum.pool, new ByteArrayInputStream
|
|
|
|
((byte[]) addendum.annotationTable));
|
|
|
|
} catch (IOException e) {
|
|
|
|
AssertionError error = new AssertionError();
|
|
|
|
error.initCause(e);
|
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private static int resolveSpec(ClassLoader loader, byte[] spec, int start) {
|
|
|
|
int result;
|
|
|
|
int end;
|
|
|
|
switch (spec[start]) {
|
|
|
|
case 'L':
|
|
|
|
++ start;
|
|
|
|
end = start;
|
|
|
|
while (spec[end] != ';') ++ end;
|
|
|
|
result = end + 1;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case '[':
|
|
|
|
end = start + 1;
|
|
|
|
while (spec[end] == '[') ++ end;
|
|
|
|
switch (spec[end]) {
|
|
|
|
case 'L':
|
|
|
|
++ end;
|
|
|
|
while (spec[end] != ';') ++ end;
|
|
|
|
++ end;
|
|
|
|
break;
|
2014-12-29 10:00:51 -07:00
|
|
|
|
2010-09-10 15:05:29 -06:00
|
|
|
default:
|
|
|
|
++ end;
|
|
|
|
}
|
|
|
|
result = end;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
return start + 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
loadVMClass(loader, spec, start, end - start);
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
fix Class.getDeclaredMethods
getDeclaredMethods was returning methods which were inherited from
interfaces but not (re)declared in the class itself, due to the VM's
internal use of VMClass.methodTable differing from its role in
reflection. For reflection, we must only include the declared
methods, not the inherited but un-redeclared ones.
Previously, we saved the original method table in
ClassAddendum.methodTable before creating a new one which contains
both declared and inherited methods. That wasted space, so this patch
replaces ClassAddendum.methodTable with
ClassAddendum.declaredMethodCount, which specifies how many of the
methods in VMClass.methodTable were declared in that class.
Alternatively, we could ensure that undeclared methods always have
their VMMethod.class_ field set to the declaring class instead of the
inheriting class. I tried this, but it led to subtle crashes in
interface method lookup. The rest of the VM relies not only on
VMClass.methodTable containing all inherited interface methods but
also that those methods point to the inheriting class, not the
declaring class. Changing those assumptions would be a much bigger
(and more dangerous in terms of regression potential) effort than I
care to take on right now. The solution I chose is a bit ugly, but
it's safe.
2014-03-03 09:56:26 -07:00
|
|
|
private static int declaredMethodCount(VMClass c) {
|
|
|
|
ClassAddendum a = c.addendum;
|
|
|
|
if (a != null) {
|
|
|
|
int count = a.declaredMethodCount;
|
|
|
|
if (count >= 0) {
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
VMMethod[] table = c.methodTable;
|
|
|
|
return table == null ? 0 : table.length;
|
|
|
|
}
|
|
|
|
|
2010-09-10 15:05:29 -06:00
|
|
|
public static void link(VMClass c, ClassLoader loader) {
|
|
|
|
acquireClassLock();
|
|
|
|
try {
|
|
|
|
if ((c.vmFlags & LinkFlag) == 0) {
|
|
|
|
if (c.super_ != null) {
|
|
|
|
link(c.super_, loader);
|
|
|
|
}
|
|
|
|
|
|
|
|
parseAnnotationTable(loader, c.addendum);
|
|
|
|
|
|
|
|
if (c.interfaceTable != null) {
|
|
|
|
int stride = ((c.flags & Modifier.INTERFACE) != 0 ? 1 : 2);
|
|
|
|
for (int i = 0; i < c.interfaceTable.length; i += stride) {
|
|
|
|
link((VMClass) c.interfaceTable[i], loader);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
fix Class.getDeclaredMethods
getDeclaredMethods was returning methods which were inherited from
interfaces but not (re)declared in the class itself, due to the VM's
internal use of VMClass.methodTable differing from its role in
reflection. For reflection, we must only include the declared
methods, not the inherited but un-redeclared ones.
Previously, we saved the original method table in
ClassAddendum.methodTable before creating a new one which contains
both declared and inherited methods. That wasted space, so this patch
replaces ClassAddendum.methodTable with
ClassAddendum.declaredMethodCount, which specifies how many of the
methods in VMClass.methodTable were declared in that class.
Alternatively, we could ensure that undeclared methods always have
their VMMethod.class_ field set to the declaring class instead of the
inheriting class. I tried this, but it led to subtle crashes in
interface method lookup. The rest of the VM relies not only on
VMClass.methodTable containing all inherited interface methods but
also that those methods point to the inheriting class, not the
declaring class. Changing those assumptions would be a much bigger
(and more dangerous in terms of regression potential) effort than I
care to take on right now. The solution I chose is a bit ugly, but
it's safe.
2014-03-03 09:56:26 -07:00
|
|
|
VMMethod[] methodTable = c.methodTable;
|
|
|
|
if (methodTable != null) {
|
|
|
|
for (int i = 0; i < methodTable.length; ++i) {
|
|
|
|
VMMethod m = methodTable[i];
|
2010-09-10 15:05:29 -06:00
|
|
|
|
|
|
|
for (int j = 1; j < m.spec.length;) {
|
|
|
|
j = resolveSpec(loader, m.spec, j);
|
|
|
|
}
|
|
|
|
|
|
|
|
parseAnnotationTable(loader, m.addendum);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (c.fieldTable != null) {
|
|
|
|
for (int i = 0; i < c.fieldTable.length; ++i) {
|
|
|
|
VMField f = c.fieldTable[i];
|
|
|
|
|
|
|
|
resolveSpec(loader, f.spec, 0);
|
|
|
|
|
|
|
|
parseAnnotationTable(loader, f.addendum);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
c.vmFlags |= LinkFlag;
|
|
|
|
}
|
|
|
|
} finally {
|
|
|
|
releaseClassLock();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public static void link(VMClass c) {
|
|
|
|
link(c, c.loader);
|
|
|
|
}
|
|
|
|
|
2013-02-21 15:37:17 -07:00
|
|
|
public static Class forName(String name, boolean initialize,
|
|
|
|
ClassLoader loader)
|
|
|
|
throws ClassNotFoundException
|
|
|
|
{
|
|
|
|
if (loader == null) {
|
|
|
|
loader = Class.class.getClassLoader();
|
|
|
|
}
|
|
|
|
Class c = loader.loadClass(name);
|
|
|
|
VMClass vmc = SystemClassLoader.vmClass(c);
|
2014-12-29 10:00:51 -07:00
|
|
|
link(vmc, loader);
|
2013-02-21 15:37:17 -07:00
|
|
|
if (initialize) {
|
2014-12-29 10:00:51 -07:00
|
|
|
initialize(vmc);
|
2013-02-21 15:37:17 -07:00
|
|
|
}
|
|
|
|
return c;
|
|
|
|
}
|
|
|
|
|
|
|
|
public static Class forCanonicalName(String name) {
|
|
|
|
return forCanonicalName(null, name);
|
|
|
|
}
|
|
|
|
|
|
|
|
public static Class forCanonicalName(ClassLoader loader, String name) {
|
|
|
|
try {
|
|
|
|
if (name.startsWith("[")) {
|
|
|
|
return forName(name, true, loader);
|
|
|
|
} else if (name.startsWith("L")) {
|
|
|
|
return forName(name.substring(1, name.length() - 1), true, loader);
|
|
|
|
} else {
|
|
|
|
if (name.length() == 1) {
|
|
|
|
return SystemClassLoader.getClass
|
2014-12-29 10:00:51 -07:00
|
|
|
(primitiveClass(name.charAt(0)));
|
2013-02-21 15:37:17 -07:00
|
|
|
} else {
|
|
|
|
throw new ClassNotFoundException(name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} catch (ClassNotFoundException e) {
|
|
|
|
throw new RuntimeException(e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private static int next(char c, String s, int start) {
|
|
|
|
for (int i = start; i < s.length(); ++i) {
|
|
|
|
if (s.charAt(i) == c) return i;
|
|
|
|
}
|
|
|
|
throw new RuntimeException();
|
|
|
|
}
|
|
|
|
|
|
|
|
public static Class[] getParameterTypes(VMMethod vmMethod) {
|
|
|
|
int count = vmMethod.parameterCount;
|
|
|
|
|
|
|
|
Class[] types = new Class[count];
|
|
|
|
int index = 0;
|
|
|
|
|
|
|
|
String spec = new String
|
2013-02-22 11:55:01 -07:00
|
|
|
(vmMethod.spec, 1, vmMethod.spec.length - 2);
|
2013-02-21 15:37:17 -07:00
|
|
|
|
|
|
|
try {
|
|
|
|
for (int i = 0; i < spec.length(); ++i) {
|
|
|
|
char c = spec.charAt(i);
|
|
|
|
if (c == ')') {
|
|
|
|
break;
|
|
|
|
} else if (c == 'L') {
|
|
|
|
int start = i + 1;
|
|
|
|
i = next(';', spec, start);
|
|
|
|
String name = spec.substring(start, i).replace('/', '.');
|
|
|
|
types[index++] = Class.forName(name, true, vmMethod.class_.loader);
|
|
|
|
} else if (c == '[') {
|
|
|
|
int start = i;
|
|
|
|
while (spec.charAt(i) == '[') ++i;
|
|
|
|
|
|
|
|
if (spec.charAt(i) == 'L') {
|
|
|
|
i = next(';', spec, i + 1);
|
|
|
|
String name = spec.substring(start, i).replace('/', '.');
|
|
|
|
types[index++] = Class.forName
|
|
|
|
(name, true, vmMethod.class_.loader);
|
|
|
|
} else {
|
|
|
|
String name = spec.substring(start, i + 1);
|
|
|
|
types[index++] = forCanonicalName(vmMethod.class_.loader, name);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
String name = spec.substring(i, i + 1);
|
|
|
|
types[index++] = forCanonicalName(vmMethod.class_.loader, name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} catch (ClassNotFoundException e) {
|
|
|
|
throw new RuntimeException(e);
|
|
|
|
}
|
|
|
|
|
|
|
|
return types;
|
|
|
|
}
|
2013-02-22 11:55:01 -07:00
|
|
|
|
2013-02-21 15:37:17 -07:00
|
|
|
public static int findField(VMClass vmClass, String name) {
|
|
|
|
if (vmClass.fieldTable != null) {
|
2014-12-29 10:00:51 -07:00
|
|
|
link(vmClass);
|
2013-02-21 15:37:17 -07:00
|
|
|
|
|
|
|
for (int i = 0; i < vmClass.fieldTable.length; ++i) {
|
|
|
|
if (toString(vmClass.fieldTable[i].name).equals(name)) {
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
public static String toString(byte[] array) {
|
|
|
|
return new String(array, 0, array.length - 1);
|
|
|
|
}
|
|
|
|
|
2014-08-28 16:59:06 -06:00
|
|
|
private static boolean match(VMClass a, VMClass b) {
|
2014-09-03 16:24:15 -06:00
|
|
|
// TODO: in theory we should be able to just do an == comparison
|
|
|
|
// here instead of recursively comparing array element types.
|
|
|
|
// However, the VM currently can create multiple array classes for
|
|
|
|
// the same element type. We should fix that so that there's only
|
|
|
|
// ever one of each per classloader, eliminating the need for a
|
|
|
|
// recursive comparison. See also the native implementation of
|
|
|
|
// isAssignableFrom.
|
2014-08-28 16:59:06 -06:00
|
|
|
if (a.arrayDimensions > 0) {
|
|
|
|
return match(a.arrayElementClass, b.arrayElementClass);
|
|
|
|
} else {
|
|
|
|
return a == b;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-02-22 11:55:01 -07:00
|
|
|
public static boolean match(Class[] a, Class[] b) {
|
2013-02-21 15:37:17 -07:00
|
|
|
if (a.length == b.length) {
|
|
|
|
for (int i = 0; i < a.length; ++i) {
|
2014-08-28 16:59:06 -06:00
|
|
|
if (! match(toVMClass(a[i]), toVMClass(b[i]))) {
|
2013-02-21 15:37:17 -07:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-02-22 11:55:01 -07:00
|
|
|
public static int findMethod(VMClass vmClass, String name,
|
2013-02-21 15:37:17 -07:00
|
|
|
Class[] parameterTypes)
|
|
|
|
{
|
fix Class.getDeclaredMethods
getDeclaredMethods was returning methods which were inherited from
interfaces but not (re)declared in the class itself, due to the VM's
internal use of VMClass.methodTable differing from its role in
reflection. For reflection, we must only include the declared
methods, not the inherited but un-redeclared ones.
Previously, we saved the original method table in
ClassAddendum.methodTable before creating a new one which contains
both declared and inherited methods. That wasted space, so this patch
replaces ClassAddendum.methodTable with
ClassAddendum.declaredMethodCount, which specifies how many of the
methods in VMClass.methodTable were declared in that class.
Alternatively, we could ensure that undeclared methods always have
their VMMethod.class_ field set to the declaring class instead of the
inheriting class. I tried this, but it led to subtle crashes in
interface method lookup. The rest of the VM relies not only on
VMClass.methodTable containing all inherited interface methods but
also that those methods point to the inheriting class, not the
declaring class. Changing those assumptions would be a much bigger
(and more dangerous in terms of regression potential) effort than I
care to take on right now. The solution I chose is a bit ugly, but
it's safe.
2014-03-03 09:56:26 -07:00
|
|
|
VMMethod[] methodTable = vmClass.methodTable;
|
|
|
|
if (methodTable != null) {
|
2014-12-29 10:00:51 -07:00
|
|
|
link(vmClass);
|
2013-02-21 15:37:17 -07:00
|
|
|
|
|
|
|
if (parameterTypes == null) {
|
|
|
|
parameterTypes = new Class[0];
|
|
|
|
}
|
|
|
|
|
fix Class.getDeclaredMethods
getDeclaredMethods was returning methods which were inherited from
interfaces but not (re)declared in the class itself, due to the VM's
internal use of VMClass.methodTable differing from its role in
reflection. For reflection, we must only include the declared
methods, not the inherited but un-redeclared ones.
Previously, we saved the original method table in
ClassAddendum.methodTable before creating a new one which contains
both declared and inherited methods. That wasted space, so this patch
replaces ClassAddendum.methodTable with
ClassAddendum.declaredMethodCount, which specifies how many of the
methods in VMClass.methodTable were declared in that class.
Alternatively, we could ensure that undeclared methods always have
their VMMethod.class_ field set to the declaring class instead of the
inheriting class. I tried this, but it led to subtle crashes in
interface method lookup. The rest of the VM relies not only on
VMClass.methodTable containing all inherited interface methods but
also that those methods point to the inheriting class, not the
declaring class. Changing those assumptions would be a much bigger
(and more dangerous in terms of regression potential) effort than I
care to take on right now. The solution I chose is a bit ugly, but
it's safe.
2014-03-03 09:56:26 -07:00
|
|
|
for (int i = 0; i < methodTable.length; ++i) {
|
|
|
|
VMMethod m = methodTable[i];
|
|
|
|
if (toString(m.name).equals(name)
|
|
|
|
&& match(parameterTypes, getParameterTypes(m)))
|
2013-02-21 15:37:17 -07:00
|
|
|
{
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2013-04-23 13:47:15 -06:00
|
|
|
public static int countMethods(VMClass vmClass, boolean publicOnly) {
|
|
|
|
int count = 0;
|
fix Class.getDeclaredMethods
getDeclaredMethods was returning methods which were inherited from
interfaces but not (re)declared in the class itself, due to the VM's
internal use of VMClass.methodTable differing from its role in
reflection. For reflection, we must only include the declared
methods, not the inherited but un-redeclared ones.
Previously, we saved the original method table in
ClassAddendum.methodTable before creating a new one which contains
both declared and inherited methods. That wasted space, so this patch
replaces ClassAddendum.methodTable with
ClassAddendum.declaredMethodCount, which specifies how many of the
methods in VMClass.methodTable were declared in that class.
Alternatively, we could ensure that undeclared methods always have
their VMMethod.class_ field set to the declaring class instead of the
inheriting class. I tried this, but it led to subtle crashes in
interface method lookup. The rest of the VM relies not only on
VMClass.methodTable containing all inherited interface methods but
also that those methods point to the inheriting class, not the
declaring class. Changing those assumptions would be a much bigger
(and more dangerous in terms of regression potential) effort than I
care to take on right now. The solution I chose is a bit ugly, but
it's safe.
2014-03-03 09:56:26 -07:00
|
|
|
VMMethod[] methodTable = vmClass.methodTable;
|
|
|
|
if (methodTable != null) {
|
|
|
|
for (int i = 0, j = declaredMethodCount(vmClass); i < j; ++i) {
|
|
|
|
VMMethod m = methodTable[i];
|
|
|
|
if (((! publicOnly) || ((m.flags & Modifier.PUBLIC)) != 0)
|
|
|
|
&& (! toString(m.name).startsWith("<")))
|
2013-04-23 13:47:15 -06:00
|
|
|
{
|
|
|
|
++ count;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
|
|
|
public static Method[] getMethods(VMClass vmClass, boolean publicOnly) {
|
|
|
|
Method[] array = new Method[countMethods(vmClass, publicOnly)];
|
fix Class.getDeclaredMethods
getDeclaredMethods was returning methods which were inherited from
interfaces but not (re)declared in the class itself, due to the VM's
internal use of VMClass.methodTable differing from its role in
reflection. For reflection, we must only include the declared
methods, not the inherited but un-redeclared ones.
Previously, we saved the original method table in
ClassAddendum.methodTable before creating a new one which contains
both declared and inherited methods. That wasted space, so this patch
replaces ClassAddendum.methodTable with
ClassAddendum.declaredMethodCount, which specifies how many of the
methods in VMClass.methodTable were declared in that class.
Alternatively, we could ensure that undeclared methods always have
their VMMethod.class_ field set to the declaring class instead of the
inheriting class. I tried this, but it led to subtle crashes in
interface method lookup. The rest of the VM relies not only on
VMClass.methodTable containing all inherited interface methods but
also that those methods point to the inheriting class, not the
declaring class. Changing those assumptions would be a much bigger
(and more dangerous in terms of regression potential) effort than I
care to take on right now. The solution I chose is a bit ugly, but
it's safe.
2014-03-03 09:56:26 -07:00
|
|
|
VMMethod[] methodTable = vmClass.methodTable;
|
|
|
|
if (methodTable != null) {
|
2014-12-29 10:00:51 -07:00
|
|
|
link(vmClass);
|
2013-04-23 13:47:15 -06:00
|
|
|
|
|
|
|
int ai = 0;
|
fix Class.getDeclaredMethods
getDeclaredMethods was returning methods which were inherited from
interfaces but not (re)declared in the class itself, due to the VM's
internal use of VMClass.methodTable differing from its role in
reflection. For reflection, we must only include the declared
methods, not the inherited but un-redeclared ones.
Previously, we saved the original method table in
ClassAddendum.methodTable before creating a new one which contains
both declared and inherited methods. That wasted space, so this patch
replaces ClassAddendum.methodTable with
ClassAddendum.declaredMethodCount, which specifies how many of the
methods in VMClass.methodTable were declared in that class.
Alternatively, we could ensure that undeclared methods always have
their VMMethod.class_ field set to the declaring class instead of the
inheriting class. I tried this, but it led to subtle crashes in
interface method lookup. The rest of the VM relies not only on
VMClass.methodTable containing all inherited interface methods but
also that those methods point to the inheriting class, not the
declaring class. Changing those assumptions would be a much bigger
(and more dangerous in terms of regression potential) effort than I
care to take on right now. The solution I chose is a bit ugly, but
it's safe.
2014-03-03 09:56:26 -07:00
|
|
|
for (int i = 0, j = declaredMethodCount(vmClass); i < j; ++i) {
|
|
|
|
VMMethod m = methodTable[i];
|
|
|
|
if (((! publicOnly) || ((m.flags & Modifier.PUBLIC) != 0))
|
|
|
|
&& ! toString(m.name).startsWith("<"))
|
2013-04-23 13:47:15 -06:00
|
|
|
{
|
|
|
|
array[ai++] = makeMethod(SystemClassLoader.getClass(vmClass), i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return array;
|
|
|
|
}
|
|
|
|
|
2013-12-06 15:45:46 -07:00
|
|
|
public static int countFields(VMClass vmClass, boolean publicOnly) {
|
|
|
|
int count = 0;
|
|
|
|
if (vmClass.fieldTable != null) {
|
|
|
|
for (int i = 0; i < vmClass.fieldTable.length; ++i) {
|
|
|
|
if ((! publicOnly)
|
|
|
|
|| ((vmClass.fieldTable[i].flags & Modifier.PUBLIC))
|
|
|
|
!= 0)
|
|
|
|
{
|
|
|
|
++ count;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
|
|
|
public static Field[] getFields(VMClass vmClass, boolean publicOnly) {
|
|
|
|
Field[] array = new Field[countFields(vmClass, publicOnly)];
|
|
|
|
if (vmClass.fieldTable != null) {
|
2014-12-29 10:00:51 -07:00
|
|
|
link(vmClass);
|
2013-12-06 15:45:46 -07:00
|
|
|
|
|
|
|
int ai = 0;
|
|
|
|
for (int i = 0; i < vmClass.fieldTable.length; ++i) {
|
|
|
|
if (((vmClass.fieldTable[i].flags & Modifier.PUBLIC) != 0)
|
|
|
|
|| (! publicOnly))
|
|
|
|
{
|
|
|
|
array[ai++] = makeField(SystemClassLoader.getClass(vmClass), i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return array;
|
|
|
|
}
|
|
|
|
|
2013-02-21 15:37:17 -07:00
|
|
|
public static Annotation getAnnotation(ClassLoader loader, Object[] a) {
|
|
|
|
if (a[0] == null) {
|
|
|
|
a[0] = Proxy.newProxyInstance
|
|
|
|
(loader, new Class[] { (Class) a[1] },
|
|
|
|
new AnnotationInvocationHandler(a));
|
|
|
|
}
|
|
|
|
return (Annotation) a[0];
|
|
|
|
}
|
|
|
|
|
2013-10-31 00:28:01 -05:00
|
|
|
public static Object getAnnotationDefaultValue(ClassLoader loader,
|
|
|
|
MethodAddendum addendum) {
|
|
|
|
if (addendum == null) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
byte[] annotationDefault = (byte[]) addendum.annotationDefault;
|
|
|
|
if (annotationDefault == null) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
return parseAnnotationValue(loader, addendum.pool,
|
|
|
|
new ByteArrayInputStream(annotationDefault));
|
|
|
|
} catch (IOException e) {
|
|
|
|
AssertionError error = new AssertionError();
|
|
|
|
error.initCause(e);
|
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-12-06 15:45:46 -07:00
|
|
|
private static int index(VMMethod m) {
|
|
|
|
VMMethod[] table = m.class_.methodTable;
|
|
|
|
for (int i = 0; i < table.length; ++i) {
|
|
|
|
if (m == table[i]) return i;
|
|
|
|
}
|
|
|
|
throw new AssertionError();
|
|
|
|
}
|
|
|
|
|
|
|
|
public static Method makeMethod(VMMethod m) {
|
|
|
|
return makeMethod(SystemClassLoader.getClass(m.class_), index(m));
|
|
|
|
}
|
2014-02-26 13:39:05 -07:00
|
|
|
|
|
|
|
public static ProtectionDomain getProtectionDomain(VMClass c) {
|
|
|
|
CodeSource source = null;
|
|
|
|
if (c.source != null) {
|
|
|
|
try {
|
|
|
|
source = new CodeSource
|
|
|
|
(new URL(new String(c.source, 0, c.source.length - 1)),
|
|
|
|
(Certificate[]) null);
|
|
|
|
} catch (MalformedURLException ignored) { }
|
|
|
|
}
|
|
|
|
|
|
|
|
Permissions p = new Permissions();
|
|
|
|
p.add(new AllPermission());
|
|
|
|
|
|
|
|
return new ProtectionDomain(source, p);
|
|
|
|
}
|
2014-12-29 10:00:51 -07:00
|
|
|
|
2013-02-21 15:37:17 -07:00
|
|
|
public static native Method makeMethod(Class c, int slot);
|
2014-12-29 10:00:51 -07:00
|
|
|
|
2013-12-06 15:45:46 -07:00
|
|
|
public static native Field makeField(Class c, int slot);
|
2013-02-21 15:37:17 -07:00
|
|
|
|
2010-09-10 15:05:29 -06:00
|
|
|
private static native void acquireClassLock();
|
|
|
|
|
|
|
|
private static native void releaseClassLock();
|
2014-08-21 13:09:42 -06:00
|
|
|
|
|
|
|
public static native String makeString(byte[] array, int offset, int length);
|
2010-09-10 15:05:29 -06:00
|
|
|
}
|