corda/classpath/java/lang/reflect/Field.java

356 lines
9.5 KiB
Java
Raw Normal View History

/* Copyright (c) 2008-2013, 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 {
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;
}
2007-07-21 20:44:39 +00:00
public Class<T> getDeclaringClass() {
return SystemClassLoader.getClass(vmField.class_);
2007-07-21 20:44:39 +00:00
}
public int getModifiers() {
return vmField.flags;
2007-07-21 20:44:39 +00:00
}
public String getName() {
return getName(vmField);
}
public static String getName(VMField vmField) {
return new String(vmField.name, 0, vmField.name.length - 1, false);
2007-07-21 20:44:39 +00:00
}
2007-07-24 01:44:20 +00:00
public Class getType() {
2013-02-22 18:55:01 +00:00
return Classes.forCanonicalName
(vmField.class_.loader,
new String(vmField.spec, 0, vmField.spec.length - 1, false));
}
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);
}
public Object get(Object instance) throws IllegalAccessException {
2007-11-02 21:08:14 +00:00
Object target;
if ((vmField.flags & Modifier.STATIC) != 0) {
target = vmField.class_.staticTable;
} else if (Class.isInstance(vmField.class_, instance)) {
2007-11-02 21:08:14 +00:00
target = instance;
} else {
throw new IllegalArgumentException();
}
switch (vmField.code) {
2007-11-02 21:08:14 +00:00
case ByteField:
return Byte.valueOf
((byte) getPrimitive(target, vmField.code, vmField.offset));
2007-11-02 21:08:14 +00:00
case BooleanField:
return Boolean.valueOf
(getPrimitive(target, vmField.code, vmField.offset) != 0);
2007-11-02 21:08:14 +00:00
case CharField:
return Character.valueOf
((char) getPrimitive(target, vmField.code, vmField.offset));
2007-11-02 21:08:14 +00:00
case ShortField:
return Short.valueOf
((short) getPrimitive(target, vmField.code, vmField.offset));
2007-11-02 21:08:14 +00:00
case IntField:
return Integer.valueOf
((int) getPrimitive(target, vmField.code, vmField.offset));
2007-11-02 21:08:14 +00:00
case LongField:
return Long.valueOf
(getPrimitive(target, vmField.code, vmField.offset));
2007-11-02 21:08:14 +00:00
case FloatField:
return Float.valueOf
(Float.intBitsToFloat
((int) getPrimitive(target, vmField.code, vmField.offset)));
2007-11-02 21:08:14 +00:00
case DoubleField:
return Double.valueOf
(Double.longBitsToDouble
(getPrimitive(target, vmField.code, vmField.offset)));
2007-11-02 21:08:14 +00:00
case ObjectField:
return getObject(target, vmField.offset);
2007-11-02 21:08:14 +00:00
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();
}
public void set(Object instance, Object value)
throws IllegalAccessException
{
2007-11-02 21:08:14 +00:00
Object target;
if ((vmField.flags & Modifier.STATIC) != 0) {
target = vmField.class_.staticTable;
} else if (Class.isInstance(vmField.class_, instance)) {
2007-11-02 21:08:14 +00:00
target = instance;
} else {
throw new IllegalArgumentException();
}
2007-11-02 21:08:14 +00:00
switch (vmField.code) {
2007-11-02 21:08:14 +00:00
case ByteField:
setPrimitive(target, vmField.code, vmField.offset, (Byte) value);
2007-11-02 21:08:14 +00:00
break;
case BooleanField:
setPrimitive
(target, vmField.code, vmField.offset, ((Boolean) value) ? 1 : 0);
2007-11-02 21:08:14 +00:00
break;
case CharField:
setPrimitive(target, vmField.code, vmField.offset, (Character) value);
2007-11-02 21:08:14 +00:00
break;
case ShortField:
setPrimitive(target, vmField.code, vmField.offset, (Short) value);
2007-11-02 21:08:14 +00:00
break;
case IntField:
setPrimitive(target, vmField.code, vmField.offset, (Integer) value);
2007-11-02 21:08:14 +00:00
break;
case LongField:
setPrimitive(target, vmField.code, vmField.offset, (Long) value);
2007-11-02 21:08:14 +00:00
break;
case FloatField:
setPrimitive(target, vmField.code, vmField.offset,
2007-11-02 21:08:14 +00:00
Float.floatToRawIntBits((Float) value));
break;
case DoubleField:
setPrimitive(target, vmField.code, vmField.offset,
2007-11-02 21:08:14 +00:00
Double.doubleToRawLongBits((Double) value));
break;
case ObjectField:
if (value == null || getType().isInstance(value)) {
setObject(target, vmField.offset, value);
2007-11-02 21:08:14 +00:00
} else {
throw new IllegalArgumentException
("needed " + getType() + ", got "
+ value.getClass().getName() +
" when setting " + Class.getName(vmField.class_) + "." + getName());
2007-11-02 21:08:14 +00:00
}
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();
}
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];
}
public <T extends Annotation> T getAnnotation(Class<T> class_) {
if (vmField.addendum != null && vmField.addendum.annotationTable != null) {
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 (vmField.addendum.annotationTable != null) {
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);
2007-07-27 23:56:19 +00:00
private static native void setObject
(Object instance, int offset, Object value);
}