corda/classpath/java/lang/reflect/Field.java
2015-03-13 12:52:59 -06:00

402 lines
10 KiB
Java

/* Copyright (c) 2008-2015, 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.reflect;
import avian.VMField;
import avian.AnnotationInvocationHandler;
import avian.SystemClassLoader;
import avian.Classes;
import java.lang.annotation.Annotation;
public class Field<T> extends AccessibleObject implements Member {
private static final int VoidField = 0;
private static final int ByteField = 1;
private static final int CharField = 2;
private static final int DoubleField = 3;
private static final int FloatField = 4;
private static final int IntField = 5;
private static final int LongField = 6;
private static final int ShortField = 7;
private static final int BooleanField = 8;
private static final int ObjectField = 9;
private final VMField vmField;
private boolean accessible = true;
public Field(VMField vmField) {
this.vmField = vmField;
}
public boolean isAccessible() {
return accessible;
}
public void setAccessible(boolean v) {
accessible = v;
}
public Class<T> getDeclaringClass() {
return SystemClassLoader.getClass(vmField.class_);
}
public int getModifiers() {
return vmField.flags;
}
public boolean isSynthetic() {
return (vmField.flags & ACC_SYNTHETIC) != 0;
}
public String getName() {
return getName(vmField);
}
public static String getName(VMField vmField) {
return Classes.makeString(vmField.name, 0, vmField.name.length - 1);
}
public Class getType() {
return Classes.forCanonicalName
(vmField.class_.loader,
Classes.makeString(vmField.spec, 0, vmField.spec.length - 1));
}
public Type getGenericType() {
if (vmField.addendum == null || vmField.addendum.signature == null) {
return getType();
}
String signature = Classes.toString((byte[]) vmField.addendum.signature);
return SignatureParser.parse(vmField.class_.loader, signature, getDeclaringClass());
}
public Object get(Object instance) throws IllegalAccessException {
Object target;
if ((vmField.flags & Modifier.STATIC) != 0) {
target = vmField.class_.staticTable;
} else if (Class.isInstance(vmField.class_, instance)) {
target = instance;
} else {
throw new IllegalArgumentException();
}
Classes.initialize(vmField.class_);
switch (vmField.code) {
case ByteField:
return Byte.valueOf
((byte) getPrimitive(target, vmField.code, vmField.offset));
case BooleanField:
return Boolean.valueOf
(getPrimitive(target, vmField.code, vmField.offset) != 0);
case CharField:
return Character.valueOf
((char) getPrimitive(target, vmField.code, vmField.offset));
case ShortField:
return Short.valueOf
((short) getPrimitive(target, vmField.code, vmField.offset));
case IntField:
return Integer.valueOf
((int) getPrimitive(target, vmField.code, vmField.offset));
case LongField:
return Long.valueOf
(getPrimitive(target, vmField.code, vmField.offset));
case FloatField:
return Float.valueOf
(Float.intBitsToFloat
((int) getPrimitive(target, vmField.code, vmField.offset)));
case DoubleField:
return Double.valueOf
(Double.longBitsToDouble
(getPrimitive(target, vmField.code, vmField.offset)));
case ObjectField:
return getObject(target, vmField.offset);
default:
throw new Error();
}
}
public boolean getBoolean(Object instance) throws IllegalAccessException {
return ((Boolean) get(instance)).booleanValue();
}
public byte getByte(Object instance) throws IllegalAccessException {
return ((Byte) get(instance)).byteValue();
}
public short getShort(Object instance) throws IllegalAccessException {
return ((Short) get(instance)).shortValue();
}
public char getChar(Object instance) throws IllegalAccessException {
return ((Character) get(instance)).charValue();
}
public int getInt(Object instance) throws IllegalAccessException {
return ((Integer) get(instance)).intValue();
}
public float getFloat(Object instance) throws IllegalAccessException {
return ((Float) get(instance)).floatValue();
}
public long getLong(Object instance) throws IllegalAccessException {
return ((Long) get(instance)).longValue();
}
public double getDouble(Object instance) throws IllegalAccessException {
return ((Double) get(instance)).doubleValue();
}
private boolean matchType(Object value) {
switch (vmField.code) {
case ByteField:
return value instanceof Byte;
case BooleanField:
return value instanceof Boolean;
case CharField:
return value instanceof Character;
case ShortField:
return value instanceof Short;
case IntField:
return value instanceof Integer;
case LongField:
return value instanceof Long;
case FloatField:
return value instanceof Float;
case DoubleField:
return value instanceof Double;
case ObjectField:
return value == null || getType().isInstance(value);
default:
throw new Error();
}
}
public void set(Object instance, Object value)
throws IllegalAccessException
{
Object target;
if ((vmField.flags & Modifier.STATIC) != 0) {
target = vmField.class_.staticTable;
} else if (Class.isInstance(vmField.class_, instance)) {
target = instance;
} else {
throw new IllegalArgumentException();
}
if (! matchType(value)) {
throw new IllegalArgumentException();
}
Classes.initialize(vmField.class_);
switch (vmField.code) {
case ByteField:
setPrimitive(target, vmField.code, vmField.offset, (Byte) value);
break;
case BooleanField:
setPrimitive
(target, vmField.code, vmField.offset, ((Boolean) value) ? 1 : 0);
break;
case CharField:
setPrimitive(target, vmField.code, vmField.offset, (Character) value);
break;
case ShortField:
setPrimitive(target, vmField.code, vmField.offset, (Short) value);
break;
case IntField:
setPrimitive(target, vmField.code, vmField.offset, (Integer) value);
break;
case LongField:
setPrimitive(target, vmField.code, vmField.offset, (Long) value);
break;
case FloatField:
setPrimitive(target, vmField.code, vmField.offset,
Float.floatToRawIntBits((Float) value));
break;
case DoubleField:
setPrimitive(target, vmField.code, vmField.offset,
Double.doubleToRawLongBits((Double) value));
break;
case ObjectField:
setObject(target, vmField.offset, value);
break;
default:
throw new Error();
}
}
private void set(Object instance, long value)
throws IllegalAccessException
{
Object target;
if ((vmField.flags & Modifier.STATIC) != 0) {
target = vmField.class_.staticTable;
} else if (Class.isInstance(vmField.class_, instance)) {
target = instance;
} else {
throw new IllegalArgumentException();
}
Classes.initialize(vmField.class_);
switch (vmField.code) {
case ByteField:
case BooleanField:
case CharField:
case ShortField:
case IntField:
case LongField:
case FloatField:
case DoubleField:
setPrimitive(target, vmField.code, vmField.offset, value);
break;
default:
throw new IllegalArgumentException
("needed " + getType() + ", got primitive type when setting "
+ Class.getName(vmField.class_) + "." + getName());
}
}
public void setByte(Object instance, byte value)
throws IllegalAccessException
{
set(instance, value & 0xff);
}
public void setBoolean(Object instance, boolean value)
throws IllegalAccessException
{
set(instance, value ? 1 : 0);
}
public void setChar(Object instance, char value)
throws IllegalAccessException
{
set(instance, value & 0xffff);
}
public void setShort(Object instance, short value)
throws IllegalAccessException
{
set(instance, value & 0xffff);
}
public void setInt(Object instance, int value)
throws IllegalAccessException
{
set(instance, value & 0xffffffffl);
}
public void setLong(Object instance, long value)
throws IllegalAccessException
{
set(instance, value);
}
public void setFloat(Object instance, float value)
throws IllegalAccessException
{
set(instance, Float.floatToIntBits(value));
}
public void setDouble(Object instance, double value)
throws IllegalAccessException
{
set(instance, Double.doubleToLongBits(value));
}
private Annotation getAnnotation(Object[] a) {
if (a[0] == null) {
a[0] = Proxy.newProxyInstance
(vmField.class_.loader, new Class[] { (Class) a[1] },
new AnnotationInvocationHandler(a));
}
return (Annotation) a[0];
}
private boolean hasAnnotations() {
return vmField.addendum != null
&& vmField.addendum.annotationTable != null;
}
public <T extends Annotation> T getAnnotation(Class<T> class_) {
if (hasAnnotations()) {
Object[] table = (Object[]) vmField.addendum.annotationTable;
for (int i = 0; i < table.length; ++i) {
Object[] a = (Object[]) table[i];
if (a[1] == class_) {
return (T) getAnnotation(a);
}
}
}
return null;
}
public Annotation[] getAnnotations() {
if (hasAnnotations()) {
Object[] table = (Object[]) vmField.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];
}
}
public Annotation[] getDeclaredAnnotations() {
return getAnnotations();
}
private static native long getPrimitive
(Object instance, int code, int offset);
private static native Object getObject
(Object instance, int offset);
private static native void setPrimitive
(Object instance, int code, int offset, long value);
private static native void setObject
(Object instance, int offset, Object value);
}