2009-12-03 02:08:29 +00:00
|
|
|
/* Copyright (c) 2008-2009, Avian Contributors
|
2008-02-19 18:06:52 +00: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. */
|
|
|
|
|
2007-07-21 17:50:26 +00:00
|
|
|
package java.lang.reflect;
|
|
|
|
|
2009-09-19 00:01:54 +00:00
|
|
|
import avian.AnnotationInvocationHandler;
|
|
|
|
|
|
|
|
import java.lang.annotation.Annotation;
|
|
|
|
|
2007-07-21 17:50:26 +00:00
|
|
|
public class Field<T> extends AccessibleObject {
|
2007-08-18 17:15:03 +00:00
|
|
|
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;
|
|
|
|
|
2007-07-21 17:50:26 +00:00
|
|
|
private byte vmFlags;
|
|
|
|
private byte code;
|
|
|
|
private short flags;
|
|
|
|
private short offset;
|
|
|
|
private byte[] name;
|
2009-09-19 00:01:54 +00:00
|
|
|
public byte[] spec;
|
2009-09-19 22:21:15 +00:00
|
|
|
public avian.Addendum addendum;
|
2007-07-21 17:50:26 +00:00
|
|
|
private Class<T> class_;
|
|
|
|
|
|
|
|
private Field() { }
|
|
|
|
|
|
|
|
public boolean isAccessible() {
|
|
|
|
return (vmFlags & Accessible) != 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void setAccessible(boolean v) {
|
|
|
|
if (v) vmFlags |= Accessible; else vmFlags &= ~Accessible;
|
|
|
|
}
|
2007-07-21 20:44:39 +00:00
|
|
|
|
|
|
|
public Class<T> getDeclaringClass() {
|
|
|
|
return class_;
|
|
|
|
}
|
|
|
|
|
|
|
|
public int getModifiers() {
|
|
|
|
return flags;
|
|
|
|
}
|
|
|
|
|
|
|
|
public String getName() {
|
|
|
|
return new String(name, 0, name.length - 1, false);
|
|
|
|
}
|
2007-07-24 01:44:20 +00:00
|
|
|
|
2007-08-13 00:50:25 +00:00
|
|
|
public Class getType() {
|
2009-08-10 13:48:44 +00:00
|
|
|
return Class.forCanonicalName(class_.getClassLoader(),
|
|
|
|
new String(spec, 0, spec.length - 1, false));
|
2007-08-13 00:50:25 +00:00
|
|
|
}
|
|
|
|
|
2007-08-18 17:15:03 +00:00
|
|
|
public Object get(Object instance) throws IllegalAccessException {
|
2007-11-02 21:08:14 +00:00
|
|
|
Object target;
|
2007-08-18 17:15:03 +00:00
|
|
|
if ((flags & Modifier.STATIC) != 0) {
|
2007-11-02 21:08:14 +00:00
|
|
|
target = class_.staticTable();
|
2007-08-18 17:15:03 +00:00
|
|
|
} else if (class_.isInstance(instance)) {
|
2007-11-02 21:08:14 +00:00
|
|
|
target = instance;
|
|
|
|
} else {
|
|
|
|
throw new IllegalArgumentException();
|
|
|
|
}
|
2007-08-18 17:15:03 +00:00
|
|
|
|
2007-11-02 21:08:14 +00:00
|
|
|
switch (code) {
|
|
|
|
case ByteField:
|
|
|
|
return Byte.valueOf((byte) getPrimitive(target, code, offset));
|
2007-08-18 17:15:03 +00:00
|
|
|
|
2007-11-02 21:08:14 +00:00
|
|
|
case BooleanField:
|
|
|
|
return Boolean.valueOf(getPrimitive(target, code, offset) != 0);
|
2007-08-18 17:15:03 +00:00
|
|
|
|
2007-11-02 21:08:14 +00:00
|
|
|
case CharField:
|
|
|
|
return Character.valueOf((char) getPrimitive(target, code, offset));
|
2007-08-18 17:15:03 +00:00
|
|
|
|
2007-11-02 21:08:14 +00:00
|
|
|
case ShortField:
|
|
|
|
return Short.valueOf((short) getPrimitive(target, code, offset));
|
2007-08-18 17:15:03 +00:00
|
|
|
|
2007-11-02 21:08:14 +00:00
|
|
|
case IntField:
|
|
|
|
return Integer.valueOf((int) getPrimitive(target, code, offset));
|
2007-08-18 17:15:03 +00:00
|
|
|
|
2007-11-02 21:08:14 +00:00
|
|
|
case LongField:
|
|
|
|
return Long.valueOf((int) getPrimitive(target, code, offset));
|
2007-08-18 17:15:03 +00:00
|
|
|
|
2007-11-02 21:08:14 +00:00
|
|
|
case FloatField:
|
|
|
|
return Float.valueOf
|
|
|
|
(Float.intBitsToFloat((int) getPrimitive(target, code, offset)));
|
2007-08-18 17:15:03 +00:00
|
|
|
|
2007-11-02 21:08:14 +00:00
|
|
|
case DoubleField:
|
|
|
|
return Double.valueOf
|
|
|
|
(Double.longBitsToDouble(getPrimitive(target, code, offset)));
|
2007-08-18 17:15:03 +00:00
|
|
|
|
2007-11-02 21:08:14 +00:00
|
|
|
case ObjectField:
|
2007-11-02 21:42:19 +00:00
|
|
|
return getObject(target, offset);
|
2007-11-02 21:08:14 +00:00
|
|
|
|
|
|
|
default:
|
|
|
|
throw new Error();
|
2007-08-18 17:15:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-06-02 23:14:38 +00:00
|
|
|
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();
|
|
|
|
}
|
|
|
|
|
2007-08-18 17:15:03 +00:00
|
|
|
public void set(Object instance, Object value)
|
|
|
|
throws IllegalAccessException
|
|
|
|
{
|
2007-11-02 21:08:14 +00:00
|
|
|
Object target;
|
2007-08-18 17:15:03 +00:00
|
|
|
if ((flags & Modifier.STATIC) != 0) {
|
2007-11-02 21:08:14 +00:00
|
|
|
target = class_.staticTable();
|
2007-08-18 17:15:03 +00:00
|
|
|
} else if (class_.isInstance(instance)) {
|
2007-11-02 21:08:14 +00:00
|
|
|
target = instance;
|
2007-08-18 17:15:03 +00:00
|
|
|
} else {
|
|
|
|
throw new IllegalArgumentException();
|
|
|
|
}
|
2007-11-02 21:08:14 +00:00
|
|
|
|
|
|
|
switch (code) {
|
|
|
|
case ByteField:
|
|
|
|
setPrimitive(target, code, offset, (Byte) value);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case BooleanField:
|
|
|
|
setPrimitive(target, code, offset, ((Boolean) value) ? 1 : 0);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CharField:
|
|
|
|
setPrimitive(target, code, offset, (Character) value);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ShortField:
|
|
|
|
setPrimitive(target, code, offset, (Short) value);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case IntField:
|
|
|
|
setPrimitive(target, code, offset, (Integer) value);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LongField:
|
|
|
|
setPrimitive(target, code, offset, (Long) value);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case FloatField:
|
|
|
|
setPrimitive(target, code, offset,
|
|
|
|
Float.floatToRawIntBits((Float) value));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case DoubleField:
|
|
|
|
setPrimitive(target, code, offset,
|
|
|
|
Double.doubleToRawLongBits((Double) value));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ObjectField:
|
2007-11-20 20:40:07 +00:00
|
|
|
if (value == null || getType().isInstance(value)) {
|
2007-11-02 21:08:14 +00:00
|
|
|
setObject(target, offset, value);
|
|
|
|
} else {
|
2007-11-20 20:40:07 +00:00
|
|
|
throw new IllegalArgumentException
|
2010-03-07 19:03:17 +00:00
|
|
|
("needed " + getType() + ", got " + value.getClass().getName() +
|
|
|
|
" when setting " + class_.getName() + "." + getName());
|
2007-11-02 21:08:14 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
throw new Error();
|
|
|
|
}
|
2007-08-18 17:15:03 +00:00
|
|
|
}
|
|
|
|
|
2009-09-19 00:01:54 +00:00
|
|
|
private Annotation getAnnotation(Object[] a) {
|
|
|
|
if (a[0] == null) {
|
|
|
|
a[0] = Proxy.newProxyInstance
|
|
|
|
(class_.getClassLoader(), new Class[] { (Class) a[1] },
|
|
|
|
new AnnotationInvocationHandler(a));
|
|
|
|
}
|
|
|
|
return (Annotation) a[0];
|
|
|
|
}
|
|
|
|
|
|
|
|
public <T extends Annotation> T getAnnotation(Class<T> class_) {
|
|
|
|
if (addendum != null && addendum.annotationTable != null) {
|
2009-09-19 22:21:15 +00:00
|
|
|
Object[] table = (Object[]) addendum.annotationTable;
|
2009-09-19 00:01:54 +00:00
|
|
|
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 (addendum != null && addendum.annotationTable != null) {
|
2009-09-19 22:21:15 +00:00
|
|
|
Object[] table = (Object[]) addendum.annotationTable;
|
2009-09-19 00:01:54 +00:00
|
|
|
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();
|
|
|
|
}
|
|
|
|
|
2009-06-02 23:14:38 +00:00
|
|
|
public boolean isEnumConstant() {
|
|
|
|
throw new UnsupportedOperationException();
|
|
|
|
}
|
|
|
|
|
2007-08-18 17:15:03 +00:00
|
|
|
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
|
|
|
|
2007-08-18 17:15:03 +00:00
|
|
|
private static native void setObject
|
|
|
|
(Object instance, int offset, Object value);
|
2009-09-19 00:01:54 +00:00
|
|
|
|
|
|
|
public static class Addendum {
|
2009-09-19 22:21:15 +00:00
|
|
|
public Object pool;
|
|
|
|
public Object annotationTable;
|
2009-09-19 00:01:54 +00:00
|
|
|
}
|
2007-07-21 17:50:26 +00:00
|
|
|
}
|