corda/classpath/java/lang/Class.java

423 lines
10 KiB
Java
Raw Normal View History

/* 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. */
package java.lang;
2007-07-26 20:39:53 -06:00
import java.lang.reflect.Constructor;
2007-07-23 19:44:20 -06:00
import java.lang.reflect.Method;
import java.lang.reflect.Field;
2007-07-27 17:56:19 -06:00
import java.lang.reflect.Modifier;
import java.lang.reflect.InvocationTargetException;
import java.io.InputStream;
2007-09-13 21:59:39 -06:00
import java.io.IOException;
import java.net.URL;
2007-07-23 19:44:20 -06:00
public final class Class <T> {
private static final int PrimitiveFlag = 1 << 4;
private short flags;
private byte vmFlags;
private byte arrayDimensions;
private short fixedSize;
private short arrayElementSize;
private int[] objectMask;
private byte[] name;
private Class super_;
2007-07-23 19:44:20 -06:00
private Object[] interfaceTable;
private Method[] virtualTable;
private Field[] fieldTable;
private Method[] methodTable;
2007-11-02 15:08:14 -06:00
private Object staticTable;
2007-07-30 17:19:05 -06:00
private ClassLoader loader;
private Class() { }
2007-08-19 20:57:32 -06:00
public String toString() {
return getName();
}
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;
}
public String getName() {
return new String
(replace('/', '.', name, 0, name.length - 1), 0, name.length - 1, false);
}
2007-07-23 19:44:20 -06:00
2007-11-02 15:08:14 -06:00
public Object staticTable() {
return staticTable;
}
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);
}
}
2007-07-30 17:19:05 -06: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
{
Class c = loader.loadClass(name);
if (initialize) {
c.initialize();
2007-07-30 17:19:05 -06:00
}
return c;
}
private static native Class primitiveClass(char name);
private native void initialize();
public static Class forCanonicalName(String name) {
try {
if (name.startsWith("[")) {
return forName(name);
} else if (name.startsWith("L")) {
return forName(name.substring(1, name.length() - 1));
} else {
if (name.length() == 1) {
return primitiveClass(name.charAt(0));
} else {
throw new ClassNotFoundException(name);
}
}
} catch (ClassNotFoundException e) {
throw new RuntimeException(e);
}
}
2007-07-23 19:44:20 -06:00
public Class getComponentType() {
if (isArray()) {
return (Class) staticTable;
} else {
return null;
}
}
2007-07-23 19:44:20 -06:00
public native boolean isAssignableFrom(Class c);
2007-07-26 20:39:53 -06:00
private Field findField(String name) {
2007-07-28 10:10:13 -06:00
if (fieldTable != null) {
for (int i = 0; i < fieldTable.length; ++i) {
if (fieldTable[i].getName().equals(name)) {
return fieldTable[i];
}
2007-07-23 19:44:20 -06:00
}
}
2007-07-26 20:39:53 -06:00
return null;
}
2007-07-23 19:44:20 -06:00
2007-07-26 20:39:53 -06: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-23 19:44:20 -06: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-26 20:39:53 -06:00
private Method findMethod(String name, Class[] parameterTypes) {
2007-07-28 10:10:13 -06:00
if (methodTable != null) {
for (int i = 0; i < methodTable.length; ++i) {
if (methodTable[i].getName().equals(name)
&& match(parameterTypes, methodTable[i].getParameterTypes()))
{
return methodTable[i];
}
2007-07-23 19:44:20 -06:00
}
}
2007-07-26 20:39:53 -06:00
return null;
}
public Method getDeclaredMethod(String name, Class ... parameterTypes)
throws NoSuchMethodException
{
2007-07-27 17:56:19 -06:00
if (name.startsWith("<")) {
throw new NoSuchMethodException(name);
}
Method m = findMethod(name, parameterTypes);
if (m == null) {
2007-07-26 20:39:53 -06:00
throw new NoSuchMethodException(name);
} else {
2007-07-27 17:56:19 -06:00
return m;
2007-07-26 20:39:53 -06:00
}
}
2007-07-23 19:44:20 -06:00
2007-07-26 20:39:53 -06:00
public Method getMethod(String name, Class ... parameterTypes)
throws NoSuchMethodException
{
2007-07-27 17:56:19 -06:00
if (name.startsWith("<")) {
throw new NoSuchMethodException(name);
}
2007-07-26 20:39:53 -06:00
for (Class c = this; c != null; c = c.super_) {
2007-07-27 17:56:19 -06:00
Method m = c.findMethod(name, parameterTypes);
if (m != null) {
return m;
2007-07-26 20:39:53 -06:00
}
}
2007-07-23 19:44:20 -06:00
throw new NoSuchMethodException(name);
}
2007-07-26 20:39:53 -06:00
public Constructor getConstructor(Class ... parameterTypes)
throws NoSuchMethodException
{
2007-07-27 17:56:19 -06:00
Method m = findMethod("<init>", parameterTypes);
if (m == null) {
throw new NoSuchMethodException();
} else {
return new Constructor(m);
}
2007-07-26 20:39:53 -06:00
}
2007-07-27 17:56:19 -06:00
private int countConstructors(boolean publicOnly) {
2007-07-26 20:39:53 -06:00
int count = 0;
2007-07-28 10:10:13 -06: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-26 20:39:53 -06:00
}
}
2007-07-27 17:56:19 -06:00
return count;
}
2007-07-26 20:39:53 -06:00
2007-07-27 17:56:19 -06:00
public Constructor[] getDeclaredConstructors() {
Constructor[] array = new Constructor[countConstructors(false)];
2007-07-28 10:10:13 -06:00
if (methodTable != null) {
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-26 20:39:53 -06:00
}
}
return array;
}
2007-07-27 17:56:19 -06:00
public Constructor[] getConstructors() {
Constructor[] array = new Constructor[countConstructors(true)];
2007-07-28 10:10:13 -06:00
if (methodTable != null) {
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 17:56:19 -06:00
}
}
return array;
2007-07-26 20:39:53 -06:00
}
public Field[] getDeclaredFields() {
2007-07-28 10:10:13 -06: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-26 20:39:53 -06:00
}
2007-07-27 17:56:19 -06:00
private int countPublicFields() {
int count = 0;
2007-07-28 10:10:13 -06:00
if (fieldTable != null) {
for (int i = 0; i < fieldTable.length; ++i) {
if (((fieldTable[i].getModifiers() & Modifier.PUBLIC)) != 0) {
++ count;
}
2007-07-27 17:56:19 -06:00
}
}
return count;
}
public Field[] getFields() {
Field[] array = new Field[countPublicFields()];
2007-07-28 10:10:13 -06:00
if (fieldTable != null) {
int ai = 0;
2007-07-28 10:10:13 -06:00
for (int i = 0; i < fieldTable.length; ++i) {
if (((fieldTable[i].getModifiers() & Modifier.PUBLIC)) != 0) {
array[ai++] = fieldTable[i];
2007-07-28 10:10:13 -06:00
}
2007-07-27 17:56:19 -06:00
}
}
return array;
}
private int countMethods(boolean publicOnly) {
2007-07-26 20:39:53 -06:00
int count = 0;
2007-08-19 20:57:32 -06: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-26 20:39:53 -06:00
}
}
2007-07-27 17:56:19 -06:00
return count;
}
2007-07-26 20:39:53 -06:00
2007-07-27 17:56:19 -06:00
public Method[] getDeclaredMethods() {
Method[] array = new Method[countMethods(false)];
2007-07-28 10:10:13 -06:00
if (methodTable != null) {
int ai = 0;
2007-07-28 10:10:13 -06:00
for (int i = 0; i < methodTable.length; ++i) {
if (! methodTable[i].getName().startsWith("<")) {
array[ai++] = methodTable[i];
2007-07-28 10:10:13 -06:00
}
2007-07-27 17:56:19 -06:00
}
}
return array;
}
public Method[] getMethods() {
Method[] array = new Method[countMethods(true)];
2007-07-28 10:10:13 -06:00
if (methodTable != null) {
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-26 20:39:53 -06:00
}
}
return array;
}
public Class[] getInterfaces() {
2007-07-28 10:10:13 -06:00
if (interfaceTable != null) {
Class[] array = new Class[interfaceTable.length / 2];
for (int i = 0; i < array.length; ++i) {
array[i] = (Class) interfaceTable[i * 2];
}
return array;
} else {
return new Class[0];
2007-07-26 20:39:53 -06:00
}
}
2007-11-17 11:39:29 -07:00
public T[] getEnumConstants() {
if (Enum.class.isAssignableFrom(this)) {
2007-11-17 11:39:29 -07:00
try {
return (T[]) getMethod("values").invoke(null);
} catch (Exception e) {
throw new Error();
}
} else {
return null;
}
}
2007-07-26 20:39:53 -06:00
public ClassLoader getClassLoader() {
2007-07-30 17:19:05 -06:00
return loader;
2007-07-26 20:39:53 -06:00
}
public int getModifiers() {
return flags;
}
2008-04-01 16:48:58 -06:00
public boolean isInterface() {
return (flags & Modifier.INTERFACE) != 0;
}
2007-07-26 20:39:53 -06:00
public Class getSuperclass() {
return super_;
}
public boolean isArray() {
return arrayElementSize != 0;
}
public boolean isInstance(Object o) {
return o != null && isAssignableFrom(o.getClass());
2007-07-26 20:39:53 -06:00
}
public boolean isPrimitive() {
return (vmFlags & PrimitiveFlag) != 0;
2007-07-26 20:39:53 -06:00
}
2007-09-13 21:59:39 -06:00
public URL getResource(String path) {
if (! path.startsWith("/")) {
2007-09-13 21:59:39 -06: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;
}
}
public boolean desiredAssertionStatus() {
return false;
}
}