blob: c068d2369c7f7e10661f3e7cab4e074f5261f818 [file] [log] [blame]
/*
* Copyright (C) 2008 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
import other.OtherPackage;
import java.lang.reflect.Field;
/*
* Test field access through reflection.
*/
public class Main {
public static void main(String[] args) {
SubOther.main(null);
try {
GetNonexistent.main(null);
System.err.println("Not expected to succeed");
} catch (VerifyError fe) {
// dalvik
System.out.println("Got expected failure");
} catch (NoSuchFieldError nsfe) {
// reference
System.out.println("Got expected failure");
}
}
/*
* Get the field specified by "field" from "obj".
*
* "type" determines which "get" call is made, e.g. 'B' turns into
* field.getByte().
*
* The "expectedException" must match the class of the exception thrown,
* or be null if no exception was expected.
*
* On success, the boxed value retrieved is returned.
*/
public Object getValue(Field field, Object obj, char type,
Class expectedException) {
Object result = null;
try {
switch (type) {
case 'Z':
result = new Boolean(field.getBoolean(obj));
break;
case 'B':
result = new Byte(field.getByte(obj));
break;
case 'S':
result = new Short(field.getShort(obj));
break;
case 'C':
result = new Character(field.getChar(obj));
break;
case 'I':
result = new Integer(field.getInt(obj));
break;
case 'J':
result = new Long(field.getLong(obj));
break;
case 'F':
result = new Float(field.getFloat(obj));
break;
case 'D':
result = new Double(field.getDouble(obj));
break;
case 'L':
result = field.get(obj);
break;
default:
throw new RuntimeException("bad type '" + type + "'");
}
/* success; expected? */
if (expectedException != null) {
Throwable th = new Throwable();
System.err.println("ERROR: call succeeded, was expecting "
+ expectedException);
th.printStackTrace();
}
} catch (Exception ex) {
if (expectedException == null) {
System.err.println("ERROR: call failed unexpectedly: "
+ ex.getClass());
ex.printStackTrace();
} else {
if (!expectedException.equals(ex.getClass())) {
System.err.println("ERROR: incorrect exception: wanted "
+ expectedException.getName() + ", got "
+ ex.getClass());
ex.printStackTrace();
}
}
}
return result;
}
}
/*
* Local class with some fields.
*/
class SamePackage {
public byte pubByteField;
protected byte protByteField;
protected Object protObjectField;
private float privFloatField;
}
/*
* This is a sub-class of OtherPackage, which should be allowed to access
* the various protected fields.
*/
class SubOther extends OtherPackage {
protected long protLongField = 0x1122334455667788L;
/*
* Perform the various tests.
*
* localInst.getValue() is performed using an instance of Main as the
* source of the reflection call. otherInst.getValue() uses a subclass
* of OtherPackage as the source.
*/
public static void main(String[] args) {
SubOther subOther = new SubOther();
subOther.doTests();
}
public void doTests() {
Class localClass = SamePackage.class;
Class otherClass = OtherPackage.class;
Field localPubByteField, localProtByteField, localProtObjectField,
localPrivFloatField;
Field otherPubCharField, otherProtShortField, otherProtObjectField,
otherPkgDoubleField;
Field subProtLongField;
Main localInst = new Main();
SamePackage samePkgInst = new SamePackage();
OtherPackage otherPkgInst = new OtherPackage();
Object plainObj = new Object();
/*
* Locate the various fields.
*/
try {
localPubByteField = localClass.getDeclaredField("pubByteField");
localProtByteField = localClass.getDeclaredField("protByteField");
localProtObjectField = localClass.getDeclaredField("protObjectField");
localPrivFloatField = localClass.getDeclaredField("privFloatField");
otherPubCharField = otherClass.getDeclaredField("pubCharField");
otherProtShortField = otherClass.getDeclaredField("protShortField");
otherProtObjectField = otherClass.getDeclaredField("protObjectField");
otherPkgDoubleField = otherClass.getDeclaredField("pkgDoubleField");
subProtLongField = getClass().getDeclaredField("protLongField");
} catch (NoSuchFieldException nsfe) {
throw new RuntimeException(nsfe);
}
/*
* Get a public field from a class in the same package.
*/
localInst.getValue(localPubByteField, samePkgInst, 'B', null);
/*
* Get a protected field from a class in the same package.
*/
this.getValue(localProtByteField, samePkgInst, 'B', null);
/*
* Get a private field from a class in the same package.
*/
this.getValue(localPrivFloatField, samePkgInst, 'F',
IllegalAccessException.class);
/*
* Get a protected field from otherInst's superclass.
*
* We can get at "this.protShortField" but not
* "otherPkgInst.protShortField" because we can only access
* protected fields in instances of our class -- being a subclass
* of OtherPackage does not allow us to modify protected fields in
* all other subclasses of OtherPackage.
*/
this.getValue(otherProtShortField, this, 'S',
null);
this.getValue(otherProtShortField, otherPkgInst, 'S',
IllegalAccessException.class);
this.getValue(otherPkgDoubleField, otherPkgInst, 'D',
IllegalAccessException.class);
/*
* Null object. Different exceptions based on which package
* we would be trying to access and whether or not our object
* has the correct type.
*/
localInst.getValue(localPubByteField, null, 'B',
NullPointerException.class);
this.getValue(subProtLongField, null, 'J',
NullPointerException.class);
this.getValue(localPrivFloatField, null, 'F',
IllegalAccessException.class);
localInst.getValue(otherProtShortField, null, 'S',
IllegalAccessException.class);
this.getValue(otherProtShortField, null, 'S',
IllegalAccessException.class);
this.getValue(otherPkgDoubleField, null, 'D',
IllegalAccessException.class);
localInst.getValue(otherProtShortField, null, 'Z',
IllegalAccessException.class);
/* -- Dalvik VM currently throws NPE
this.getValue(subProtLongField, null, 'Z',
IllegalArgumentException.class);
*/
/*
* Valid object, wrong field type.
*/
this.getValue(subProtLongField, this, 'J',
null);
this.getValue(localProtByteField, samePkgInst, 'Z',
IllegalArgumentException.class);
this.getValue(subProtLongField, this, 'Z',
IllegalArgumentException.class);
this.getValue(localPrivFloatField, this, 'Z',
IllegalAccessException.class);
this.getValue(localPrivFloatField, this, 'Z',
IllegalAccessException.class);
localInst.getValue(otherProtShortField, otherPkgInst, 'Z',
IllegalAccessException.class);
this.getValue(otherProtShortField, otherPkgInst, 'Z',
IllegalAccessException.class);
/*
* Wrong object.
*/
this.getValue(subProtLongField, plainObj, 'J',
IllegalArgumentException.class);
/* wrong object + private field */
this.getValue(localPrivFloatField, plainObj, 'F',
IllegalAccessException.class);
/* wrong object + wrong field type */
this.getValue(subProtLongField, plainObj, 'Z',
IllegalArgumentException.class);
/* wrong object + invalid access */
localInst.getValue(otherProtShortField, plainObj, 'S',
IllegalAccessException.class);
this.getValue(otherProtShortField, plainObj, 'S',
IllegalAccessException.class);
System.out.println("good");
}
/*
* [this is a clone of Main.getValue() -- the class issuing the
* reflection call is significant]
*/
public Object getValue(Field field, Object obj, char type,
Class expectedException) {
Object result = null;
try {
switch (type) {
case 'Z':
result = new Boolean(field.getBoolean(obj));
break;
case 'B':
result = new Byte(field.getByte(obj));
break;
case 'S':
result = new Short(field.getShort(obj));
break;
case 'C':
result = new Character(field.getChar(obj));
break;
case 'I':
result = new Integer(field.getInt(obj));
break;
case 'J':
result = new Long(field.getLong(obj));
break;
case 'F':
result = new Float(field.getFloat(obj));
break;
case 'D':
result = new Double(field.getDouble(obj));
break;
case 'L':
result = field.get(obj);
break;
default:
throw new RuntimeException("bad type '" + type + "'");
}
/* success; expected? */
if (expectedException != null) {
Throwable th = new Throwable();
System.err.println("ERROR: call succeeded, was expecting "
+ expectedException);
th.printStackTrace();
}
} catch (Exception ex) {
if (expectedException == null) {
System.err.println("ERROR: call failed unexpectedly: "
+ ex.getClass());
ex.printStackTrace();
} else {
if (!expectedException.equals(ex.getClass())) {
System.err.println("ERROR: incorrect exception: wanted "
+ expectedException.getName() + ", got "
+ ex.getClass());
ex.printStackTrace();
}
}
}
return result;
}
}