2008-02-19 18:06:52 +00:00
|
|
|
/* Copyright (c) 2008, Avian Contributors
|
|
|
|
|
|
|
|
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. */
|
|
|
|
|
2007-06-21 01:37:43 +00:00
|
|
|
package java.lang;
|
|
|
|
|
2009-09-19 00:01:54 +00:00
|
|
|
import avian.AnnotationInvocationHandler;
|
|
|
|
import avian.Pair;
|
|
|
|
|
2007-07-27 02:39:53 +00:00
|
|
|
import java.lang.reflect.Constructor;
|
2007-07-24 01:44:20 +00:00
|
|
|
import java.lang.reflect.Method;
|
|
|
|
import java.lang.reflect.Field;
|
2007-07-27 23:56:19 +00:00
|
|
|
import java.lang.reflect.Modifier;
|
2009-06-02 23:14:38 +00:00
|
|
|
import java.lang.reflect.Type;
|
|
|
|
import java.lang.reflect.TypeVariable;
|
2009-09-19 00:01:54 +00:00
|
|
|
import java.lang.reflect.Proxy;
|
|
|
|
import java.lang.reflect.InvocationHandler;
|
2007-08-30 23:31:32 +00:00
|
|
|
import java.lang.reflect.InvocationTargetException;
|
2009-06-02 23:14:38 +00:00
|
|
|
import java.lang.reflect.GenericDeclaration;
|
2009-09-19 00:01:54 +00:00
|
|
|
import java.lang.reflect.AnnotatedElement;
|
2009-06-02 23:14:38 +00:00
|
|
|
import java.lang.annotation.Annotation;
|
2007-09-12 01:13:05 +00:00
|
|
|
import java.io.InputStream;
|
2007-09-14 03:59:39 +00:00
|
|
|
import java.io.IOException;
|
|
|
|
import java.net.URL;
|
2009-08-05 00:31:14 +00:00
|
|
|
import java.util.Map;
|
|
|
|
import java.util.HashMap;
|
2009-06-02 23:14:38 +00:00
|
|
|
import java.security.ProtectionDomain;
|
2009-08-03 14:56:19 +00:00
|
|
|
import java.security.Permissions;
|
|
|
|
import java.security.AllPermission;
|
2007-07-24 01:44:20 +00:00
|
|
|
|
2009-09-19 00:01:54 +00:00
|
|
|
public final class Class <T>
|
|
|
|
implements Type, GenericDeclaration, AnnotatedElement
|
|
|
|
{
|
2009-08-05 15:57:51 +00:00
|
|
|
private static final int PrimitiveFlag = 1 << 5;
|
2009-09-19 00:01:54 +00:00
|
|
|
private static final int LinkFlag = 1 << 8;
|
2007-08-18 17:53:30 +00:00
|
|
|
|
2007-07-12 23:46:08 +00:00
|
|
|
private short flags;
|
2009-08-18 20:26:28 +00:00
|
|
|
private short vmFlags;
|
2007-07-12 23:46:08 +00:00
|
|
|
private short fixedSize;
|
2009-08-18 20:26:28 +00:00
|
|
|
private byte arrayElementSize;
|
|
|
|
private byte arrayDimensions;
|
2007-07-12 23:46:08 +00:00
|
|
|
private int[] objectMask;
|
|
|
|
private byte[] name;
|
2009-08-27 22:26:25 +00:00
|
|
|
private byte[] sourceFile;
|
2007-07-12 23:46:08 +00:00
|
|
|
private Class super_;
|
2007-07-24 01:44:20 +00:00
|
|
|
private Object[] interfaceTable;
|
|
|
|
private Method[] virtualTable;
|
|
|
|
private Field[] fieldTable;
|
|
|
|
private Method[] methodTable;
|
2009-09-19 00:01:54 +00:00
|
|
|
private Addendum addendum;
|
2007-11-02 21:08:14 +00:00
|
|
|
private Object staticTable;
|
2007-07-30 23:19:05 +00:00
|
|
|
private ClassLoader loader;
|
2007-06-21 01:37:43 +00:00
|
|
|
|
|
|
|
private Class() { }
|
2007-07-04 22:27:08 +00:00
|
|
|
|
2007-08-20 02:57:32 +00:00
|
|
|
public String toString() {
|
|
|
|
return getName();
|
|
|
|
}
|
|
|
|
|
2007-09-13 00:21:37 +00:00
|
|
|
private static byte[] replace(int a, int b, byte[] s, int offset,
|
|
|
|
int length)
|
|
|
|
{
|
|
|
|
byte[] array = new byte[length];
|
|
|
|
for (int i = 0; i < length; ++i) {
|
|
|
|
byte c = s[i];
|
|
|
|
array[i] = (byte) (c == a ? b : c);
|
|
|
|
}
|
|
|
|
return array;
|
|
|
|
}
|
|
|
|
|
2007-07-12 23:46:08 +00:00
|
|
|
public String getName() {
|
2009-08-05 15:57:51 +00:00
|
|
|
if (name == null) {
|
|
|
|
if ((vmFlags & PrimitiveFlag) != 0) {
|
|
|
|
if (this == primitiveClass('V')) {
|
2009-08-28 22:17:23 +00:00
|
|
|
name = "void\0".getBytes();
|
2009-08-05 15:57:51 +00:00
|
|
|
} else if (this == primitiveClass('Z')) {
|
2009-08-28 22:17:23 +00:00
|
|
|
name = "boolean\0".getBytes();
|
2009-08-05 15:57:51 +00:00
|
|
|
} else if (this == primitiveClass('B')) {
|
2009-08-28 22:17:23 +00:00
|
|
|
name = "byte\0".getBytes();
|
2009-08-05 15:57:51 +00:00
|
|
|
} else if (this == primitiveClass('C')) {
|
2009-08-28 22:17:23 +00:00
|
|
|
name = "char\0".getBytes();
|
2009-08-05 15:57:51 +00:00
|
|
|
} else if (this == primitiveClass('S')) {
|
2009-08-28 22:17:23 +00:00
|
|
|
name = "short\0".getBytes();
|
2009-08-05 15:57:51 +00:00
|
|
|
} else if (this == primitiveClass('I')) {
|
2009-08-28 22:17:23 +00:00
|
|
|
name = "int\0".getBytes();
|
2009-08-05 15:57:51 +00:00
|
|
|
} else if (this == primitiveClass('F')) {
|
2009-08-28 22:17:23 +00:00
|
|
|
name = "float\0".getBytes();
|
2009-08-05 15:57:51 +00:00
|
|
|
} else if (this == primitiveClass('J')) {
|
2009-08-28 22:17:23 +00:00
|
|
|
name = "long\0".getBytes();
|
2009-08-05 15:57:51 +00:00
|
|
|
} else if (this == primitiveClass('D')) {
|
2009-08-28 22:17:23 +00:00
|
|
|
name = "double\0".getBytes();
|
2009-08-05 15:57:51 +00:00
|
|
|
} else {
|
|
|
|
throw new AssertionError();
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
throw new AssertionError();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-09-13 00:21:37 +00:00
|
|
|
return new String
|
|
|
|
(replace('/', '.', name, 0, name.length - 1), 0, name.length - 1, false);
|
2007-07-12 23:46:08 +00:00
|
|
|
}
|
2007-07-24 01:44:20 +00:00
|
|
|
|
2009-08-13 14:57:06 +00:00
|
|
|
public String getCanonicalName() {
|
|
|
|
if ((vmFlags & PrimitiveFlag) != 0) {
|
|
|
|
return getName();
|
|
|
|
} else if (isArray()) {
|
|
|
|
return getComponentType().getCanonicalName() + "[]";
|
|
|
|
} else {
|
|
|
|
return getName().replace('$', '.');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public String getSimpleName() {
|
|
|
|
if ((vmFlags & PrimitiveFlag) != 0) {
|
|
|
|
return getName();
|
|
|
|
} else if (isArray()) {
|
|
|
|
return getComponentType().getSimpleName() + "[]";
|
|
|
|
} else {
|
|
|
|
String name = getCanonicalName();
|
|
|
|
int index = name.lastIndexOf('.');
|
|
|
|
if (index >= 0) {
|
|
|
|
return name.substring(index + 1);
|
|
|
|
} else {
|
|
|
|
return name;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-11-02 21:08:14 +00:00
|
|
|
public Object staticTable() {
|
2007-08-18 17:15:03 +00:00
|
|
|
return staticTable;
|
|
|
|
}
|
|
|
|
|
2007-08-30 23:31:32 +00:00
|
|
|
public T newInstance()
|
|
|
|
throws IllegalAccessException, InstantiationException
|
|
|
|
{
|
|
|
|
try {
|
|
|
|
return (T) getConstructor().newInstance();
|
|
|
|
} catch (NoSuchMethodException e) {
|
|
|
|
throw new RuntimeException(e);
|
|
|
|
} catch (InvocationTargetException e) {
|
|
|
|
throw new RuntimeException(e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-09-19 00:01:54 +00:00
|
|
|
private static Class loadClass(ClassLoader loader,
|
|
|
|
byte[] nameBytes, int offset, int length)
|
|
|
|
{
|
|
|
|
String name = new String(nameBytes, offset, length, false);
|
|
|
|
try {
|
|
|
|
return loader.loadClass(name);
|
|
|
|
} catch (ClassNotFoundException e) {
|
|
|
|
NoClassDefFoundError error = new NoClassDefFoundError(name);
|
|
|
|
error.initCause(e);
|
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private static Object resolveAnnotationValue(ClassLoader loader,
|
|
|
|
Object raw)
|
|
|
|
{
|
|
|
|
if (raw instanceof Pair) {
|
|
|
|
Pair<byte[],byte[]> p = (Pair<byte[],byte[]>) raw;
|
|
|
|
return Enum.valueOf
|
|
|
|
(loadClass(loader, p.first, 1, p.first.length - 3),
|
|
|
|
new String(p.second, 0, p.second.length - 1, false));
|
|
|
|
} else if (raw instanceof byte[]) {
|
|
|
|
byte[] typeName = (byte[]) raw;
|
|
|
|
return loadClass(loader, typeName, 0, typeName.length - 1);
|
|
|
|
} else if (raw.getClass().isArray()) {
|
|
|
|
Object[] array = (Object[]) raw;
|
|
|
|
if (array.length > 0 && array[0] == null) {
|
|
|
|
resolveAnnotation(loader, array);
|
|
|
|
} else {
|
|
|
|
for (int i = 0; i < array.length; ++i) {
|
|
|
|
array[i] = resolveAnnotationValue(loader, array[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return raw;
|
|
|
|
}
|
|
|
|
|
|
|
|
private static void resolveAnnotation(ClassLoader loader,
|
|
|
|
Object[] annotation)
|
|
|
|
{
|
|
|
|
byte[] typeName = (byte[]) annotation[1];
|
|
|
|
annotation[1] = loadClass(loader, typeName, 1, typeName.length - 3);
|
|
|
|
|
|
|
|
for (int i = 2; i < annotation.length; i += 2) {
|
|
|
|
byte[] name = (byte[]) annotation[i];
|
|
|
|
annotation[i] = new String(name, 0, name.length - 1, false).intern();
|
|
|
|
annotation[i + 1] = resolveAnnotationValue(loader, annotation[i + 1]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private static void resolveAnnotationTable(ClassLoader loader,
|
|
|
|
Object[] table)
|
|
|
|
{
|
|
|
|
for (int i = 0; i < table.length; ++i) {
|
|
|
|
resolveAnnotation(loader, (Object[]) table[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
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;
|
|
|
|
|
|
|
|
default:
|
|
|
|
++ end;
|
|
|
|
}
|
|
|
|
result = end;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
return start + 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
loadClass(loader, spec, start, end - start);
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
private static native void acquireClassLock();
|
|
|
|
|
|
|
|
private static native void releaseClassLock();
|
|
|
|
|
|
|
|
void link(ClassLoader loader) {
|
|
|
|
acquireClassLock();
|
|
|
|
try {
|
|
|
|
if ((vmFlags & LinkFlag) == 0) {
|
|
|
|
if (super_ != null) {
|
|
|
|
super_.link(loader);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (addendum != null && addendum.annotationTable != null) {
|
|
|
|
resolveAnnotationTable(loader, addendum.annotationTable);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (interfaceTable != null) {
|
|
|
|
int stride = (isInterface() ? 1 : 2);
|
|
|
|
for (int i = 0; i < interfaceTable.length; i += stride) {
|
|
|
|
((Class) interfaceTable[i]).link(loader);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (methodTable != null) {
|
|
|
|
for (int i = 0; i < methodTable.length; ++i) {
|
|
|
|
Method m = methodTable[i];
|
|
|
|
|
|
|
|
for (int j = 1; j < m.spec.length;) {
|
|
|
|
j = resolveSpec(loader, m.spec, j);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (m.addendum != null && m.addendum.annotationTable != null) {
|
|
|
|
resolveAnnotationTable(loader, m.addendum.annotationTable);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fieldTable != null) {
|
|
|
|
for (int i = 0; i < fieldTable.length; ++i) {
|
|
|
|
Field f = fieldTable[i];
|
|
|
|
|
|
|
|
resolveSpec(loader, f.spec, 0);
|
|
|
|
|
|
|
|
if (f.addendum != null && f.addendum.annotationTable != null) {
|
|
|
|
resolveAnnotationTable(loader, f.addendum.annotationTable);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
vmFlags |= LinkFlag;
|
|
|
|
}
|
|
|
|
} finally {
|
|
|
|
releaseClassLock();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-07-30 23:19:05 +00:00
|
|
|
public static Class forName(String name) throws ClassNotFoundException {
|
|
|
|
return forName
|
|
|
|
(name, true, Method.getCaller().getDeclaringClass().getClassLoader());
|
|
|
|
}
|
|
|
|
|
|
|
|
public static Class forName(String name, boolean initialize,
|
|
|
|
ClassLoader loader)
|
|
|
|
throws ClassNotFoundException
|
|
|
|
{
|
2009-06-07 02:29:15 +00:00
|
|
|
if (loader == null) {
|
|
|
|
loader = Class.class.loader;
|
|
|
|
}
|
2007-07-30 23:19:05 +00:00
|
|
|
Class c = loader.loadClass(name);
|
2009-08-18 20:26:28 +00:00
|
|
|
c.link(loader);
|
2007-08-01 23:48:36 +00:00
|
|
|
if (initialize) {
|
|
|
|
c.initialize();
|
2007-07-30 23:19:05 +00:00
|
|
|
}
|
|
|
|
return c;
|
|
|
|
}
|
2007-07-28 16:55:24 +00:00
|
|
|
|
|
|
|
private static native Class primitiveClass(char name);
|
2007-08-01 23:48:36 +00:00
|
|
|
|
|
|
|
private native void initialize();
|
2007-07-28 16:55:24 +00:00
|
|
|
|
2007-08-13 00:50:25 +00:00
|
|
|
public static Class forCanonicalName(String name) {
|
2009-08-10 13:46:05 +00:00
|
|
|
return forCanonicalName(null, name);
|
|
|
|
}
|
|
|
|
|
|
|
|
public static Class forCanonicalName(ClassLoader loader, String name) {
|
2007-07-28 16:55:24 +00:00
|
|
|
try {
|
|
|
|
if (name.startsWith("[")) {
|
2009-08-10 13:46:05 +00:00
|
|
|
return forName(name, true, loader);
|
2007-07-28 16:55:24 +00:00
|
|
|
} else if (name.startsWith("L")) {
|
2009-08-10 13:46:05 +00:00
|
|
|
return forName(name.substring(1, name.length() - 1), true, loader);
|
2007-07-28 16:55:24 +00:00
|
|
|
} else {
|
2007-07-29 23:32:23 +00:00
|
|
|
if (name.length() == 1) {
|
2007-07-28 16:55:24 +00:00
|
|
|
return primitiveClass(name.charAt(0));
|
|
|
|
} else {
|
|
|
|
throw new ClassNotFoundException(name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} catch (ClassNotFoundException e) {
|
|
|
|
throw new RuntimeException(e);
|
|
|
|
}
|
|
|
|
}
|
2007-07-24 01:44:20 +00:00
|
|
|
|
2007-08-13 00:50:25 +00:00
|
|
|
public Class getComponentType() {
|
|
|
|
if (isArray()) {
|
2009-08-13 14:57:06 +00:00
|
|
|
String n = getName();
|
|
|
|
if ("[Z".equals(n)) {
|
|
|
|
return primitiveClass('Z');
|
|
|
|
} else if ("[B".equals(n)) {
|
|
|
|
return primitiveClass('B');
|
|
|
|
} else if ("[S".equals(n)) {
|
|
|
|
return primitiveClass('S');
|
|
|
|
} else if ("[C".equals(n)) {
|
|
|
|
return primitiveClass('C');
|
|
|
|
} else if ("[I".equals(n)) {
|
|
|
|
return primitiveClass('I');
|
|
|
|
} else if ("[F".equals(n)) {
|
|
|
|
return primitiveClass('F');
|
|
|
|
} else if ("[J".equals(n)) {
|
|
|
|
return primitiveClass('J');
|
|
|
|
} else if ("[D".equals(n)) {
|
|
|
|
return primitiveClass('D');
|
|
|
|
}
|
|
|
|
|
|
|
|
if (staticTable == null) throw new AssertionError(name);
|
2007-11-20 20:40:07 +00:00
|
|
|
return (Class) staticTable;
|
2007-08-13 00:50:25 +00:00
|
|
|
} else {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-07-24 01:44:20 +00:00
|
|
|
public native boolean isAssignableFrom(Class c);
|
|
|
|
|
2007-07-27 02:39:53 +00:00
|
|
|
private Field findField(String name) {
|
2007-07-28 16:10:13 +00:00
|
|
|
if (fieldTable != null) {
|
2009-08-18 20:26:28 +00:00
|
|
|
link(loader);
|
|
|
|
|
2007-07-28 16:10:13 +00:00
|
|
|
for (int i = 0; i < fieldTable.length; ++i) {
|
|
|
|
if (fieldTable[i].getName().equals(name)) {
|
|
|
|
return fieldTable[i];
|
|
|
|
}
|
2007-07-24 01:44:20 +00:00
|
|
|
}
|
|
|
|
}
|
2007-07-27 02:39:53 +00:00
|
|
|
return null;
|
|
|
|
}
|
2007-07-24 01:44:20 +00:00
|
|
|
|
2007-07-27 02:39:53 +00:00
|
|
|
public Field getDeclaredField(String name) throws NoSuchFieldException {
|
|
|
|
Field f = findField(name);
|
|
|
|
if (f == null) {
|
|
|
|
throw new NoSuchFieldException(name);
|
|
|
|
} else {
|
|
|
|
return f;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public Field getField(String name) throws NoSuchFieldException {
|
|
|
|
for (Class c = this; c != null; c = c.super_) {
|
|
|
|
Field f = c.findField(name);
|
|
|
|
if (f != null) {
|
|
|
|
return f;
|
|
|
|
}
|
|
|
|
}
|
2007-07-24 01:44:20 +00:00
|
|
|
throw new NoSuchFieldException(name);
|
|
|
|
}
|
|
|
|
|
|
|
|
private static boolean match(Class[] a, Class[] b) {
|
|
|
|
if (a.length == b.length) {
|
|
|
|
for (int i = 0; i < a.length; ++i) {
|
|
|
|
if (! a[i].isAssignableFrom(b[i])) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-07-27 02:39:53 +00:00
|
|
|
private Method findMethod(String name, Class[] parameterTypes) {
|
2007-07-28 16:10:13 +00:00
|
|
|
if (methodTable != null) {
|
2009-08-18 20:26:28 +00:00
|
|
|
link(loader);
|
|
|
|
|
|
|
|
if (parameterTypes == null) {
|
2009-06-05 20:28:55 +00:00
|
|
|
parameterTypes = new Class[0];
|
2009-08-18 20:26:28 +00:00
|
|
|
}
|
|
|
|
|
2007-07-28 16:10:13 +00:00
|
|
|
for (int i = 0; i < methodTable.length; ++i) {
|
|
|
|
if (methodTable[i].getName().equals(name)
|
|
|
|
&& match(parameterTypes, methodTable[i].getParameterTypes()))
|
|
|
|
{
|
|
|
|
return methodTable[i];
|
|
|
|
}
|
2007-07-24 01:44:20 +00:00
|
|
|
}
|
|
|
|
}
|
2007-07-27 02:39:53 +00:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
public Method getDeclaredMethod(String name, Class ... parameterTypes)
|
|
|
|
throws NoSuchMethodException
|
|
|
|
{
|
2007-07-27 23:56:19 +00:00
|
|
|
if (name.startsWith("<")) {
|
|
|
|
throw new NoSuchMethodException(name);
|
|
|
|
}
|
|
|
|
Method m = findMethod(name, parameterTypes);
|
|
|
|
if (m == null) {
|
2007-07-27 02:39:53 +00:00
|
|
|
throw new NoSuchMethodException(name);
|
|
|
|
} else {
|
2007-07-27 23:56:19 +00:00
|
|
|
return m;
|
2007-07-27 02:39:53 +00:00
|
|
|
}
|
|
|
|
}
|
2007-07-24 01:44:20 +00:00
|
|
|
|
2007-07-27 02:39:53 +00:00
|
|
|
public Method getMethod(String name, Class ... parameterTypes)
|
|
|
|
throws NoSuchMethodException
|
|
|
|
{
|
2007-07-27 23:56:19 +00:00
|
|
|
if (name.startsWith("<")) {
|
|
|
|
throw new NoSuchMethodException(name);
|
|
|
|
}
|
2007-07-27 02:39:53 +00:00
|
|
|
for (Class c = this; c != null; c = c.super_) {
|
2007-07-27 23:56:19 +00:00
|
|
|
Method m = c.findMethod(name, parameterTypes);
|
|
|
|
if (m != null) {
|
|
|
|
return m;
|
2007-07-27 02:39:53 +00:00
|
|
|
}
|
|
|
|
}
|
2007-07-24 01:44:20 +00:00
|
|
|
throw new NoSuchMethodException(name);
|
|
|
|
}
|
2007-07-27 02:39:53 +00:00
|
|
|
|
|
|
|
public Constructor getConstructor(Class ... parameterTypes)
|
|
|
|
throws NoSuchMethodException
|
|
|
|
{
|
2007-07-27 23:56:19 +00:00
|
|
|
Method m = findMethod("<init>", parameterTypes);
|
|
|
|
if (m == null) {
|
|
|
|
throw new NoSuchMethodException();
|
|
|
|
} else {
|
|
|
|
return new Constructor(m);
|
|
|
|
}
|
2007-07-27 02:39:53 +00:00
|
|
|
}
|
|
|
|
|
2008-11-11 15:20:49 +00:00
|
|
|
public Constructor getDeclaredConstructor(Class ... parameterTypes)
|
|
|
|
throws NoSuchMethodException
|
|
|
|
{
|
|
|
|
Constructor c = null;
|
|
|
|
Constructor[] constructors = getDeclaredConstructors();
|
|
|
|
|
|
|
|
for (int i = 0; i < constructors.length; ++i) {
|
|
|
|
if (match(parameterTypes, constructors[i].getParameterTypes())) {
|
|
|
|
c = constructors[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (c == null) {
|
|
|
|
throw new NoSuchMethodException();
|
|
|
|
} else {
|
|
|
|
return c;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-07-27 23:56:19 +00:00
|
|
|
private int countConstructors(boolean publicOnly) {
|
2007-07-27 02:39:53 +00:00
|
|
|
int count = 0;
|
2007-07-28 16:10:13 +00:00
|
|
|
if (methodTable != null) {
|
|
|
|
for (int i = 0; i < methodTable.length; ++i) {
|
|
|
|
if (((! publicOnly)
|
|
|
|
|| ((methodTable[i].getModifiers() & Modifier.PUBLIC)) != 0)
|
|
|
|
&& methodTable[i].getName().equals("<init>"))
|
|
|
|
{
|
|
|
|
++ count;
|
|
|
|
}
|
2007-07-27 02:39:53 +00:00
|
|
|
}
|
|
|
|
}
|
2007-07-27 23:56:19 +00:00
|
|
|
return count;
|
|
|
|
}
|
2007-07-27 02:39:53 +00:00
|
|
|
|
2007-07-27 23:56:19 +00:00
|
|
|
public Constructor[] getDeclaredConstructors() {
|
|
|
|
Constructor[] array = new Constructor[countConstructors(false)];
|
2007-07-28 16:10:13 +00:00
|
|
|
if (methodTable != null) {
|
2009-08-18 20:26:28 +00:00
|
|
|
link(loader);
|
|
|
|
|
2007-07-28 16:10:13 +00:00
|
|
|
int index = 0;
|
|
|
|
for (int i = 0; i < methodTable.length; ++i) {
|
|
|
|
if (methodTable[i].getName().equals("<init>")) {
|
|
|
|
array[index++] = new Constructor(methodTable[i]);
|
|
|
|
}
|
2007-07-27 02:39:53 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return array;
|
|
|
|
}
|
|
|
|
|
2007-07-27 23:56:19 +00:00
|
|
|
public Constructor[] getConstructors() {
|
|
|
|
Constructor[] array = new Constructor[countConstructors(true)];
|
2007-07-28 16:10:13 +00:00
|
|
|
if (methodTable != null) {
|
2009-08-18 20:26:28 +00:00
|
|
|
link(loader);
|
|
|
|
|
2007-07-28 16:10:13 +00:00
|
|
|
int index = 0;
|
|
|
|
for (int i = 0; i < methodTable.length; ++i) {
|
|
|
|
if (((methodTable[i].getModifiers() & Modifier.PUBLIC) != 0)
|
|
|
|
&& methodTable[i].getName().equals("<init>"))
|
|
|
|
{
|
|
|
|
array[index++] = new Constructor(methodTable[i]);
|
|
|
|
}
|
2007-07-27 23:56:19 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return array;
|
2007-07-27 02:39:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public Field[] getDeclaredFields() {
|
2007-07-28 16:10:13 +00:00
|
|
|
if (fieldTable != null) {
|
|
|
|
Field[] array = new Field[fieldTable.length];
|
|
|
|
System.arraycopy(fieldTable, 0, array, 0, fieldTable.length);
|
|
|
|
return array;
|
|
|
|
} else {
|
|
|
|
return new Field[0];
|
|
|
|
}
|
2007-07-27 02:39:53 +00:00
|
|
|
}
|
|
|
|
|
2007-07-27 23:56:19 +00:00
|
|
|
private int countPublicFields() {
|
|
|
|
int count = 0;
|
2007-07-28 16:10:13 +00:00
|
|
|
if (fieldTable != null) {
|
|
|
|
for (int i = 0; i < fieldTable.length; ++i) {
|
|
|
|
if (((fieldTable[i].getModifiers() & Modifier.PUBLIC)) != 0) {
|
|
|
|
++ count;
|
|
|
|
}
|
2007-07-27 23:56:19 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
|
|
|
public Field[] getFields() {
|
|
|
|
Field[] array = new Field[countPublicFields()];
|
2007-07-28 16:10:13 +00:00
|
|
|
if (fieldTable != null) {
|
2009-08-18 20:26:28 +00:00
|
|
|
link(loader);
|
|
|
|
|
2007-11-20 20:40:07 +00:00
|
|
|
int ai = 0;
|
2007-07-28 16:10:13 +00:00
|
|
|
for (int i = 0; i < fieldTable.length; ++i) {
|
|
|
|
if (((fieldTable[i].getModifiers() & Modifier.PUBLIC)) != 0) {
|
2007-11-20 20:40:07 +00:00
|
|
|
array[ai++] = fieldTable[i];
|
2007-07-28 16:10:13 +00:00
|
|
|
}
|
2007-07-27 23:56:19 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return array;
|
|
|
|
}
|
|
|
|
|
|
|
|
private int countMethods(boolean publicOnly) {
|
2007-07-27 02:39:53 +00:00
|
|
|
int count = 0;
|
2007-08-20 02:57:32 +00:00
|
|
|
if (methodTable != null) {
|
|
|
|
for (int i = 0; i < methodTable.length; ++i) {
|
|
|
|
if (((! publicOnly)
|
|
|
|
|| ((methodTable[i].getModifiers() & Modifier.PUBLIC)) != 0)
|
|
|
|
&& (! methodTable[i].getName().startsWith("<")))
|
|
|
|
{
|
|
|
|
++ count;
|
|
|
|
}
|
2007-07-27 02:39:53 +00:00
|
|
|
}
|
|
|
|
}
|
2007-07-27 23:56:19 +00:00
|
|
|
return count;
|
|
|
|
}
|
2007-07-27 02:39:53 +00:00
|
|
|
|
2007-07-27 23:56:19 +00:00
|
|
|
public Method[] getDeclaredMethods() {
|
|
|
|
Method[] array = new Method[countMethods(false)];
|
2007-07-28 16:10:13 +00:00
|
|
|
if (methodTable != null) {
|
2009-08-18 20:26:28 +00:00
|
|
|
link(loader);
|
|
|
|
|
2007-11-20 20:40:07 +00:00
|
|
|
int ai = 0;
|
2007-07-28 16:10:13 +00:00
|
|
|
for (int i = 0; i < methodTable.length; ++i) {
|
|
|
|
if (! methodTable[i].getName().startsWith("<")) {
|
2007-11-20 20:40:07 +00:00
|
|
|
array[ai++] = methodTable[i];
|
2007-07-28 16:10:13 +00:00
|
|
|
}
|
2007-07-27 23:56:19 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return array;
|
|
|
|
}
|
|
|
|
|
|
|
|
public Method[] getMethods() {
|
|
|
|
Method[] array = new Method[countMethods(true)];
|
2007-07-28 16:10:13 +00:00
|
|
|
if (methodTable != null) {
|
2009-08-18 20:26:28 +00:00
|
|
|
link(loader);
|
|
|
|
|
2007-07-28 16:10:13 +00:00
|
|
|
int index = 0;
|
|
|
|
for (int i = 0; i < methodTable.length; ++i) {
|
|
|
|
if (((methodTable[i].getModifiers() & Modifier.PUBLIC) != 0)
|
|
|
|
&& (! methodTable[i].getName().startsWith("<")))
|
|
|
|
{
|
|
|
|
array[index++] = methodTable[i];
|
|
|
|
}
|
2007-07-27 02:39:53 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return array;
|
|
|
|
}
|
|
|
|
|
|
|
|
public Class[] getInterfaces() {
|
2007-07-28 16:10:13 +00:00
|
|
|
if (interfaceTable != null) {
|
2009-08-18 20:26:28 +00:00
|
|
|
link(loader);
|
|
|
|
|
2009-09-19 00:01:54 +00:00
|
|
|
int stride = (isInterface() ? 1 : 2);
|
|
|
|
Class[] array = new Class[interfaceTable.length / stride];
|
2007-07-28 16:10:13 +00:00
|
|
|
for (int i = 0; i < array.length; ++i) {
|
2009-09-19 00:01:54 +00:00
|
|
|
array[i] = (Class) interfaceTable[i * stride];
|
2007-07-28 16:10:13 +00:00
|
|
|
}
|
|
|
|
return array;
|
|
|
|
} else {
|
|
|
|
return new Class[0];
|
2007-07-27 02:39:53 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-11-17 18:39:29 +00:00
|
|
|
public T[] getEnumConstants() {
|
2007-11-20 20:40:07 +00:00
|
|
|
if (Enum.class.isAssignableFrom(this)) {
|
2007-11-17 18:39:29 +00:00
|
|
|
try {
|
|
|
|
return (T[]) getMethod("values").invoke(null);
|
|
|
|
} catch (Exception e) {
|
|
|
|
throw new Error();
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-07-27 02:39:53 +00:00
|
|
|
public ClassLoader getClassLoader() {
|
2007-07-30 23:19:05 +00:00
|
|
|
return loader;
|
2007-07-27 02:39:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public int getModifiers() {
|
|
|
|
return flags;
|
|
|
|
}
|
|
|
|
|
2008-04-01 22:48:58 +00:00
|
|
|
public boolean isInterface() {
|
|
|
|
return (flags & Modifier.INTERFACE) != 0;
|
|
|
|
}
|
|
|
|
|
2007-07-27 02:39:53 +00:00
|
|
|
public Class getSuperclass() {
|
|
|
|
return super_;
|
|
|
|
}
|
|
|
|
|
|
|
|
public boolean isArray() {
|
2009-08-18 20:26:28 +00:00
|
|
|
return arrayDimensions != 0;
|
2007-07-27 02:39:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public boolean isInstance(Object o) {
|
2007-11-20 20:40:07 +00:00
|
|
|
return o != null && isAssignableFrom(o.getClass());
|
2007-07-27 02:39:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public boolean isPrimitive() {
|
2007-08-18 17:53:30 +00:00
|
|
|
return (vmFlags & PrimitiveFlag) != 0;
|
2007-07-27 02:39:53 +00:00
|
|
|
}
|
2007-09-12 01:13:05 +00:00
|
|
|
|
2007-09-14 03:59:39 +00:00
|
|
|
public URL getResource(String path) {
|
2007-09-12 01:13:05 +00:00
|
|
|
if (! path.startsWith("/")) {
|
2007-09-14 03:59:39 +00:00
|
|
|
String name = new String(this.name, 0, this.name.length - 1, false);
|
|
|
|
int index = name.lastIndexOf('/');
|
|
|
|
if (index >= 0) {
|
|
|
|
path = name.substring(0, index) + "/" + path;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return getClassLoader().getResource(path);
|
|
|
|
}
|
|
|
|
|
|
|
|
public InputStream getResourceAsStream(String path) {
|
|
|
|
URL url = getResource(path);
|
|
|
|
try {
|
|
|
|
return (url == null ? null : url.openStream());
|
|
|
|
} catch (IOException e) {
|
|
|
|
return null;
|
2007-09-12 01:13:05 +00:00
|
|
|
}
|
|
|
|
}
|
2009-06-02 23:14:38 +00:00
|
|
|
|
2008-07-03 15:16:32 +00:00
|
|
|
public boolean desiredAssertionStatus() {
|
2008-07-03 16:49:08 +00:00
|
|
|
return false;
|
2008-07-03 15:16:32 +00:00
|
|
|
}
|
2009-06-02 23:14:38 +00:00
|
|
|
|
2009-08-13 14:57:06 +00:00
|
|
|
public <T> Class<? extends T> asSubclass(Class<T> c) {
|
|
|
|
if (! c.isAssignableFrom(this)) {
|
|
|
|
throw new ClassCastException();
|
|
|
|
}
|
|
|
|
|
|
|
|
return (Class<? extends T>) this;
|
|
|
|
}
|
|
|
|
|
2009-06-02 23:14:38 +00:00
|
|
|
public T cast(Object o) {
|
|
|
|
return (T) o;
|
|
|
|
}
|
|
|
|
|
|
|
|
public Object[] getSigners() {
|
2009-09-19 00:01:54 +00:00
|
|
|
return addendum == null ? null : addendum.signers;
|
2009-06-02 23:14:38 +00:00
|
|
|
}
|
|
|
|
|
2009-08-13 14:57:06 +00:00
|
|
|
public Package getPackage() {
|
|
|
|
if ((vmFlags & PrimitiveFlag) != 0 || isArray()) {
|
|
|
|
return null;
|
|
|
|
} else {
|
|
|
|
String name = getCanonicalName();
|
|
|
|
int index = name.lastIndexOf('.');
|
|
|
|
if (index >= 0) {
|
|
|
|
return new Package(name.substring(0, index),
|
|
|
|
null, null, null, null, null, null, null, null);
|
|
|
|
} else {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-09-19 00:01:54 +00:00
|
|
|
public boolean isAnnotationPresent
|
|
|
|
(Class<? extends Annotation> class_)
|
|
|
|
{
|
|
|
|
return getAnnotation(class_) != null;
|
|
|
|
}
|
|
|
|
|
|
|
|
private Annotation getAnnotation(Object[] a) {
|
|
|
|
if (a[0] == null) {
|
|
|
|
a[0] = Proxy.newProxyInstance
|
|
|
|
(loader, new Class[] { (Class) a[1] },
|
|
|
|
new AnnotationInvocationHandler(a));
|
|
|
|
}
|
|
|
|
return (Annotation) a[0];
|
|
|
|
}
|
|
|
|
|
|
|
|
public <T extends Annotation> T getAnnotation(Class<T> class_) {
|
|
|
|
for (Class c = this; c != null; c = c.super_) {
|
|
|
|
if (c.addendum != null && c.addendum.annotationTable != null) {
|
|
|
|
link(c.loader);
|
|
|
|
|
|
|
|
Object[] table = c.addendum.annotationTable;
|
|
|
|
for (int i = 0; i < table.length; ++i) {
|
|
|
|
Object[] a = (Object[]) table[i];
|
|
|
|
if (a[1] == class_) {
|
|
|
|
return (T) c.getAnnotation(a);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2009-06-02 23:14:38 +00:00
|
|
|
public Annotation[] getDeclaredAnnotations() {
|
2009-09-19 00:01:54 +00:00
|
|
|
if (addendum != null && addendum.annotationTable != null) {
|
|
|
|
link(loader);
|
|
|
|
|
|
|
|
Object[] table = addendum.annotationTable;
|
|
|
|
Annotation[] array = new Annotation[table.length];
|
|
|
|
for (int i = 0; i < table.length; ++i) {
|
|
|
|
array[i] = getAnnotation((Object[]) table[i]);
|
|
|
|
}
|
|
|
|
return array;
|
|
|
|
} else {
|
|
|
|
return new Annotation[0];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private int countAnnotations() {
|
|
|
|
int count = 0;
|
|
|
|
for (Class c = this; c != null; c = c.super_) {
|
|
|
|
if (c.addendum != null && c.addendum.annotationTable != null) {
|
|
|
|
count += c.addendum.annotationTable.length;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
|
|
|
public Annotation[] getAnnotations() {
|
|
|
|
Annotation[] array = new Annotation[countMethods(true)];
|
|
|
|
int i = 0;
|
|
|
|
for (Class c = this; c != null; c = c.super_) {
|
|
|
|
if (c.addendum != null && c.addendum.annotationTable != null) {
|
|
|
|
Object[] table = c.addendum.annotationTable;
|
|
|
|
for (int j = 0; j < table.length; ++j) {
|
|
|
|
array[i++] = getAnnotation((Object[]) table[j]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return array;
|
2009-06-02 23:14:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public boolean isEnum() {
|
|
|
|
throw new UnsupportedOperationException();
|
|
|
|
}
|
|
|
|
|
|
|
|
public TypeVariable<Class<T>>[] getTypeParameters() {
|
|
|
|
throw new UnsupportedOperationException();
|
|
|
|
}
|
|
|
|
|
|
|
|
public Method getEnclosingMethod() {
|
|
|
|
throw new UnsupportedOperationException();
|
|
|
|
}
|
|
|
|
|
|
|
|
public Constructor getEnclosingConstructor() {
|
|
|
|
throw new UnsupportedOperationException();
|
|
|
|
}
|
|
|
|
|
|
|
|
public Class getEnclosingClass() {
|
|
|
|
throw new UnsupportedOperationException();
|
|
|
|
}
|
|
|
|
|
|
|
|
public Class[] getDeclaredClasses() {
|
|
|
|
throw new UnsupportedOperationException();
|
|
|
|
}
|
|
|
|
|
|
|
|
public ProtectionDomain getProtectionDomain() {
|
2009-08-03 14:56:19 +00:00
|
|
|
Permissions p = new Permissions();
|
|
|
|
p.add(new AllPermission());
|
|
|
|
return new ProtectionDomain(null, p);
|
2009-06-02 23:14:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// for GNU Classpath compatibility:
|
|
|
|
void setSigners(Object[] signers) {
|
2009-08-05 00:31:14 +00:00
|
|
|
if (signers != null && signers.length > 0) {
|
2009-09-19 00:01:54 +00:00
|
|
|
if (addendum == null) {
|
|
|
|
addendum = new Addendum();
|
|
|
|
}
|
|
|
|
addendum.signers = signers;
|
2009-08-05 00:31:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-09-19 00:01:54 +00:00
|
|
|
private static class Addendum {
|
|
|
|
public Object[] annotationTable;
|
|
|
|
public Object[] signers;
|
2009-06-02 23:14:38 +00:00
|
|
|
}
|
2007-06-21 01:37:43 +00:00
|
|
|
}
|