032067264f
Reviewed-by: alanb
583 lines
26 KiB
Java
583 lines
26 KiB
Java
/*
|
|
* Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved.
|
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
|
*
|
|
* This code is free software; you can redistribute it and/or modify it
|
|
* under the terms of the GNU General Public License version 2 only, as
|
|
* published by the Free Software Foundation.
|
|
*
|
|
* This code is distributed in the hope that it will be useful, but WITHOUT
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
|
* version 2 for more details (a copy is included in the LICENSE file that
|
|
* accompanied this code).
|
|
*
|
|
* You should have received a copy of the GNU General Public License version
|
|
* 2 along with this work; if not, write to the Free Software Foundation,
|
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
*
|
|
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
|
* or visit www.oracle.com if you need additional information or have any
|
|
* questions.
|
|
*/
|
|
|
|
/**
|
|
* @test
|
|
* @bug 8277451
|
|
* @run testng/othervm -Djdk.reflect.useDirectMethodHandle=true NegativeTest
|
|
* @run testng/othervm -Djdk.reflect.useDirectMethodHandle=false NegativeTest
|
|
* @summary Test exception thrown due to bad receiver and bad value on
|
|
* Field with and without setAccessible(true)
|
|
*/
|
|
|
|
import java.lang.reflect.Field;
|
|
import java.lang.reflect.Modifier;
|
|
|
|
import org.testng.annotations.DataProvider;
|
|
import org.testng.annotations.Test;
|
|
import static org.testng.Assert.*;
|
|
|
|
public class NegativeTest {
|
|
static class Fields {
|
|
public static int si;
|
|
public static char sc;
|
|
public static byte sb;
|
|
public static short ss;
|
|
public static long sl;
|
|
public static double sd;
|
|
public static float sf;
|
|
public static boolean sz;
|
|
public static String so;
|
|
|
|
public static final int sfi = 10;
|
|
public static final char sfc = 'a';
|
|
public static final byte sfb = 1;
|
|
public static final short sfs = 2;
|
|
public static final long sfl = 1000L;
|
|
public static final double sfd = 1.0;
|
|
public static final float sff = 2.0f;
|
|
public static final boolean sfz = true;
|
|
public static final String sfo = "abc";
|
|
|
|
public int i;
|
|
public char c;
|
|
public byte b;
|
|
public short s;
|
|
public long l;
|
|
public double d;
|
|
public float f;
|
|
public boolean z;
|
|
public String o;
|
|
|
|
public final int fi = 10;
|
|
public final char fc = 'a';
|
|
public final byte fb = 1;
|
|
public final short fs = 2;
|
|
public final long fl = 1000L;
|
|
public final double fd = 1.0;
|
|
public final float ff = 2.0f;
|
|
public final boolean fz = true;
|
|
public final String fo = "abc";
|
|
}
|
|
|
|
static final Field i_field = field("i", false);
|
|
static final Field c_field = field("c", false);
|
|
static final Field b_field = field("b", false);
|
|
static final Field s_field = field("s", false);
|
|
static final Field l_field = field("l", false);
|
|
static final Field d_field = field("d", false);
|
|
static final Field f_field = field("f", false);
|
|
static final Field z_field = field("z", false);
|
|
static final Field o_field = field("o", false);
|
|
static final Field fi_field = field("fi", false);
|
|
static final Field fc_field = field("fc", false);
|
|
static final Field fb_field = field("fb", false);
|
|
static final Field fs_field = field("fs", false);
|
|
static final Field fl_field = field("fl", false);
|
|
static final Field fd_field = field("fd", false);
|
|
static final Field ff_field = field("ff", false);
|
|
static final Field fz_field = field("fz", false);
|
|
static final Field fo_field = field("fo", false);
|
|
|
|
static final Field override_i_field = field("i", true);
|
|
static final Field override_c_field = field("c", true);
|
|
static final Field override_b_field = field("b", true);
|
|
static final Field override_s_field = field("s", true);
|
|
static final Field override_l_field = field("l", true);
|
|
static final Field override_d_field = field("d", true);
|
|
static final Field override_f_field = field("f", true);
|
|
static final Field override_z_field = field("z", true);
|
|
static final Field override_o_field = field("o", true);
|
|
static final Field override_fi_field = field("fi", true);
|
|
static final Field override_fc_field = field("fc", true);
|
|
static final Field override_fb_field = field("fb", true);
|
|
static final Field override_fs_field = field("fs", true);
|
|
static final Field override_fl_field = field("fl", true);
|
|
static final Field override_fd_field = field("fd", true);
|
|
static final Field override_ff_field = field("ff", true);
|
|
static final Field override_fz_field = field("fz", true);
|
|
static final Field override_fo_field = field("fo", true);
|
|
|
|
static final Field si_field = field("si", false);
|
|
static final Field sc_field = field("sc", false);
|
|
static final Field sb_field = field("sb", false);
|
|
static final Field ss_field = field("ss", false);
|
|
static final Field sl_field = field("sl", false);
|
|
static final Field sd_field = field("sd", false);
|
|
static final Field sf_field = field("sf", false);
|
|
static final Field sz_field = field("sz", false);
|
|
static final Field so_field = field("so", false);
|
|
static final Field sfi_field = field("sfi", false);
|
|
static final Field sfc_field = field("sfc", false);
|
|
static final Field sfb_field = field("sfb", false);
|
|
static final Field sfs_field = field("sfs", false);
|
|
static final Field sfl_field = field("sfl", false);
|
|
static final Field sfd_field = field("sfd", false);
|
|
static final Field sff_field = field("sff", false);
|
|
static final Field sfz_field = field("sfz", false);
|
|
static final Field sfo_field = field("sfo", false);
|
|
|
|
static final Field override_si_field = field("si", true);
|
|
static final Field override_sc_field = field("sc", true);
|
|
static final Field override_sb_field = field("sb", true);
|
|
static final Field override_ss_field = field("ss", true);
|
|
static final Field override_sl_field = field("sl", true);
|
|
static final Field override_sd_field = field("sd", true);
|
|
static final Field override_sf_field = field("sf", true);
|
|
static final Field override_sz_field = field("sz", true);
|
|
static final Field override_so_field = field("so", true);
|
|
static final Field override_sfi_field = field("sfi", true);
|
|
static final Field override_sfc_field = field("sfc", true);
|
|
static final Field override_sfb_field = field("sfb", true);
|
|
static final Field override_sfs_field = field("sfs", true);
|
|
static final Field override_sfl_field = field("sfl", true);
|
|
static final Field override_sfd_field = field("sfd", true);
|
|
static final Field override_sff_field = field("sff", true);
|
|
static final Field override_sfz_field = field("sfz", true);
|
|
static final Field override_sfo_field = field("sfo", true);
|
|
|
|
private static Field field(String name, boolean suppressAccessCheck) {
|
|
try {
|
|
Field f = Fields.class.getDeclaredField(name);
|
|
if (suppressAccessCheck) {
|
|
f.setAccessible(true);
|
|
}
|
|
return f;
|
|
} catch (ReflectiveOperationException e) {
|
|
throw new RuntimeException(e);
|
|
}
|
|
}
|
|
|
|
@DataProvider(name = "instanceFields")
|
|
private Object[][] instanceFields() {
|
|
return new Object[][]{
|
|
new Object[]{i_field},
|
|
new Object[]{c_field},
|
|
new Object[]{b_field},
|
|
new Object[]{s_field},
|
|
new Object[]{l_field},
|
|
new Object[]{d_field},
|
|
new Object[]{f_field},
|
|
new Object[]{z_field},
|
|
new Object[]{o_field},
|
|
new Object[]{override_i_field},
|
|
new Object[]{override_c_field},
|
|
new Object[]{override_b_field},
|
|
new Object[]{override_s_field},
|
|
new Object[]{override_l_field},
|
|
new Object[]{override_d_field},
|
|
new Object[]{override_f_field},
|
|
new Object[]{override_z_field},
|
|
new Object[]{override_o_field},
|
|
// final instance fields
|
|
new Object[]{fi_field},
|
|
new Object[]{fc_field},
|
|
new Object[]{fb_field},
|
|
new Object[]{fs_field},
|
|
new Object[]{fl_field},
|
|
new Object[]{fd_field},
|
|
new Object[]{ff_field},
|
|
new Object[]{fz_field},
|
|
new Object[]{fo_field},
|
|
new Object[]{override_fi_field},
|
|
new Object[]{override_fc_field},
|
|
new Object[]{override_fb_field},
|
|
new Object[]{override_fs_field},
|
|
new Object[]{override_fl_field},
|
|
new Object[]{override_fd_field},
|
|
new Object[]{override_ff_field},
|
|
new Object[]{override_fz_field},
|
|
new Object[]{override_fo_field},
|
|
};
|
|
}
|
|
private static Fields INSTANCE = new Fields();
|
|
|
|
/*
|
|
* Test Field::get on a good receiver, a bad receiver and null.
|
|
*
|
|
* IllegalArgumentException is thrown if the receiver is of
|
|
* a bad type. NullPointerException is thrown if the receiver is null.
|
|
*/
|
|
@Test(dataProvider = "instanceFields")
|
|
public void testReceiver(Field f) throws ReflectiveOperationException {
|
|
f.get(INSTANCE); // good receiver
|
|
|
|
testBadReceiver(f);
|
|
testNullReceiver(f);
|
|
}
|
|
|
|
/*
|
|
* IllegalArgumentException should be thrown for bad receiver type
|
|
*/
|
|
private void testBadReceiver(Field f) throws ReflectiveOperationException {
|
|
assertFalse(Modifier.isStatic(f.getModifiers())); // instance field
|
|
Object badObj = new NegativeTest();
|
|
try {
|
|
f.get(badObj);
|
|
fail("expected IllegalArgumentException");
|
|
} catch (IllegalArgumentException e) {
|
|
// expected
|
|
}
|
|
Class<?> fType = f.getType();
|
|
if (fType.isPrimitive()) {
|
|
try {
|
|
switch (fType.descriptorString()) {
|
|
case "B" -> f.getByte(badObj);
|
|
case "C" -> f.getChar(badObj);
|
|
case "D" -> f.getDouble(badObj);
|
|
case "F" -> f.getFloat(badObj);
|
|
case "I" -> f.getInt(badObj);
|
|
case "J" -> f.getLong(badObj);
|
|
case "S" -> f.getShort(badObj);
|
|
case "Z" -> f.getBoolean(badObj);
|
|
}
|
|
fail("expected IllegalArgumentException");
|
|
} catch (IllegalArgumentException e) {
|
|
// expected
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* NullPointerException should be thrown for null receiver
|
|
*/
|
|
private void testNullReceiver(Field f) throws ReflectiveOperationException {
|
|
assertFalse(Modifier.isStatic(f.getModifiers())); // instance field
|
|
try {
|
|
f.get(null);
|
|
fail("expected NullPointerException");
|
|
} catch (NullPointerException e) {
|
|
// expected
|
|
}
|
|
|
|
Class<?> fType = f.getType();
|
|
if (fType.isPrimitive()) {
|
|
try {
|
|
switch (fType.descriptorString()) {
|
|
case "B" -> f.getByte(null);
|
|
case "C" -> f.getChar(null);
|
|
case "D" -> f.getDouble(null);
|
|
case "F" -> f.getFloat(null);
|
|
case "I" -> f.getInt(null);
|
|
case "J" -> f.getLong(null);
|
|
case "S" -> f.getShort(null);
|
|
case "Z" -> f.getBoolean(null);
|
|
}
|
|
fail("expected NullPointerException");
|
|
} catch (NullPointerException e) {
|
|
// expected
|
|
}
|
|
}
|
|
}
|
|
|
|
@DataProvider(name = "writeableFields")
|
|
private Object[][] writeableFields() {
|
|
Fields obj = new Fields();
|
|
return new Object[][]{
|
|
// instance fields with and without setAccessible(true)
|
|
new Object[]{i_field, obj, Integer.valueOf(10)},
|
|
new Object[]{c_field, obj, Character.valueOf('c')},
|
|
new Object[]{b_field, obj, Byte.valueOf((byte)1)},
|
|
new Object[]{s_field, obj, Short.valueOf((short)2)},
|
|
new Object[]{l_field, obj, Long.valueOf(1000)},
|
|
new Object[]{d_field, obj, Double.valueOf(1.2)},
|
|
new Object[]{f_field, obj, Float.valueOf(2.5f)},
|
|
new Object[]{z_field, obj, Boolean.valueOf(true)},
|
|
new Object[]{o_field, obj, "good-value"},
|
|
new Object[]{override_i_field, obj, Integer.valueOf(10)},
|
|
new Object[]{override_c_field, obj, Character.valueOf('c')},
|
|
new Object[]{override_b_field, obj, Byte.valueOf((byte)1)},
|
|
new Object[]{override_s_field, obj, Short.valueOf((short)2)},
|
|
new Object[]{override_l_field, obj, Long.valueOf(1000)},
|
|
new Object[]{override_d_field, obj, Double.valueOf(1.2)},
|
|
new Object[]{override_f_field, obj, Float.valueOf(2.5f)},
|
|
new Object[]{override_z_field, obj, Boolean.valueOf(true)},
|
|
new Object[]{override_o_field, obj, "good-value"},
|
|
// instance final fields with setAccessible(true)
|
|
new Object[]{override_fi_field, obj, Integer.valueOf(10)},
|
|
new Object[]{override_fc_field, obj, Character.valueOf('c')},
|
|
new Object[]{override_fb_field, obj, Byte.valueOf((byte)1)},
|
|
new Object[]{override_fs_field, obj, Short.valueOf((short)2)},
|
|
new Object[]{override_fl_field, obj, Long.valueOf(1000)},
|
|
new Object[]{override_fd_field, obj, Double.valueOf(1.2)},
|
|
new Object[]{override_ff_field, obj, Float.valueOf(2.5f)},
|
|
new Object[]{override_fz_field, obj, Boolean.valueOf(true)},
|
|
new Object[]{override_fo_field, obj, "good-value"},
|
|
// static fields with and without setAccessible(true)
|
|
new Object[]{si_field, null, Integer.valueOf(10)},
|
|
new Object[]{sc_field, null, Character.valueOf('c')},
|
|
new Object[]{sb_field, null, Byte.valueOf((byte)1)},
|
|
new Object[]{ss_field, null, Short.valueOf((short)2)},
|
|
new Object[]{sl_field, null, Long.valueOf(1000)},
|
|
new Object[]{sd_field, null, Double.valueOf(1.2)},
|
|
new Object[]{sf_field, null, Float.valueOf(2.5f)},
|
|
new Object[]{sz_field, null, Boolean.valueOf(true)},
|
|
new Object[]{so_field, null, "good-value"},
|
|
new Object[]{override_si_field, null, Integer.valueOf(10)},
|
|
new Object[]{override_sc_field, null, Character.valueOf('c')},
|
|
new Object[]{override_sb_field, null, Byte.valueOf((byte)1)},
|
|
new Object[]{override_ss_field, null, Short.valueOf((short)2)},
|
|
new Object[]{override_sl_field, null, Long.valueOf(1000)},
|
|
new Object[]{override_sd_field, null, Double.valueOf(1.2)},
|
|
new Object[]{override_sf_field, null, Float.valueOf(2.5f)},
|
|
new Object[]{override_sz_field, null, Boolean.valueOf(true)},
|
|
new Object[]{override_so_field, null, "good-value"},
|
|
};
|
|
}
|
|
|
|
/*
|
|
* Test Field::set with a good and bad value.
|
|
* Test setting to null if the field type is primitive.
|
|
*
|
|
* IllegalArgumentException is thrown if the value is of a bad type or null.
|
|
* NullPointerException is thrown if the receiver of an instance field is null.
|
|
* The receiver is checked
|
|
*/
|
|
@Test(dataProvider = "writeableFields")
|
|
public void testSetValue(Field f, Object obj, Object value) throws IllegalAccessException {
|
|
f.set(obj, value);
|
|
Class<?> fType = f.getType();
|
|
if (fType.isPrimitive()) {
|
|
switch (fType.descriptorString()) {
|
|
case "B" -> f.setByte(obj, ((Byte) value).byteValue());
|
|
case "C" -> f.setChar(obj, ((Character) value).charValue());
|
|
case "D" -> f.setDouble(obj, ((Double) value).doubleValue());
|
|
case "F" -> f.setFloat(obj, ((Float) value).floatValue());
|
|
case "I" -> f.setInt(obj, ((Integer) value).intValue());
|
|
case "J" -> f.setLong(obj, ((Long) value).longValue());
|
|
case "S" -> f.setShort(obj, ((Short) value).shortValue());
|
|
case "Z" -> f.setBoolean(obj, ((Boolean) value).booleanValue());
|
|
}
|
|
|
|
// test null value only if it's primitive type
|
|
try {
|
|
f.set(obj, null);
|
|
fail("expected IllegalArgumentException");
|
|
} catch (IllegalArgumentException e) {
|
|
// expected
|
|
}
|
|
}
|
|
|
|
Object badValue = new NegativeTest();
|
|
try {
|
|
f.set(obj, badValue);
|
|
fail("expected IllegalArgumentException");
|
|
} catch (IllegalArgumentException e) {
|
|
// expected
|
|
}
|
|
}
|
|
|
|
@DataProvider(name = "readOnlyFinalFields")
|
|
private Object[][] readOnlyFinalFields() {
|
|
Object obj = INSTANCE;
|
|
return new Object[][]{
|
|
// instance final fields
|
|
new Object[]{fi_field, obj, Integer.valueOf(10)},
|
|
new Object[]{fc_field, obj, Character.valueOf('c')},
|
|
new Object[]{fb_field, obj, Byte.valueOf((byte)1)},
|
|
new Object[]{fs_field, obj, Short.valueOf((short)2)},
|
|
new Object[]{fl_field, obj, Long.valueOf(1000)},
|
|
new Object[]{fd_field, obj, Double.valueOf(1.2)},
|
|
new Object[]{ff_field, obj, Float.valueOf(2.5f)},
|
|
new Object[]{fz_field, obj, Boolean.valueOf(true)},
|
|
new Object[]{fo_field, obj, "good-value"},
|
|
// static final fields
|
|
new Object[]{sfi_field, null, Integer.valueOf(10)},
|
|
new Object[]{sfc_field, null, Character.valueOf('c')},
|
|
new Object[]{sfb_field, null, Byte.valueOf((byte)1)},
|
|
new Object[]{sfs_field, null, Short.valueOf((short)2)},
|
|
new Object[]{sfl_field, null, Long.valueOf(1000)},
|
|
new Object[]{sfd_field, null, Double.valueOf(1.2)},
|
|
new Object[]{sff_field, null, Float.valueOf(2.5f)},
|
|
new Object[]{sfz_field, null, Boolean.valueOf(true)},
|
|
new Object[]{sfo_field, null, "good-value"},
|
|
new Object[]{override_sfi_field, null, Integer.valueOf(10)},
|
|
new Object[]{override_sfc_field, null, Character.valueOf('c')},
|
|
new Object[]{override_sfb_field, null, Byte.valueOf((byte)1)},
|
|
new Object[]{override_sfs_field, null, Short.valueOf((short)2)},
|
|
new Object[]{override_sfl_field, null, Long.valueOf(1000)},
|
|
new Object[]{override_sfd_field, null, Double.valueOf(1.2)},
|
|
new Object[]{override_sff_field, null, Float.valueOf(2.5f)},
|
|
new Object[]{override_sfz_field, null, Boolean.valueOf(true)},
|
|
new Object[]{override_sfo_field, null, "good-value"},
|
|
};
|
|
}
|
|
|
|
/*
|
|
* Test Field::set on a read-only final field.
|
|
* IllegalAccessException is thrown regardless of whether the value
|
|
* is of a bad type or not.
|
|
*/
|
|
@Test(dataProvider = "readOnlyFinalFields")
|
|
public void testSetValueOnFinalField(Field f, Object obj, Object value) {
|
|
assertTrue(Modifier.isFinal(f.getModifiers()));
|
|
try {
|
|
f.set(obj, value);
|
|
fail("expected IllegalAccessException");
|
|
} catch (IllegalAccessException e) {
|
|
// expected
|
|
}
|
|
|
|
Class<?> fType = f.getType();
|
|
if (fType.isPrimitive()) {
|
|
try {
|
|
switch (fType.descriptorString()) {
|
|
case "B" -> f.setByte(obj, ((Byte)value).byteValue());
|
|
case "C" -> f.setChar(obj, ((Character)value).charValue());
|
|
case "D" -> f.setDouble(obj, ((Double)value).doubleValue());
|
|
case "F" -> f.setFloat(obj, ((Float)value).floatValue());
|
|
case "I" -> f.setInt(obj, ((Integer)value).intValue());
|
|
case "J" -> f.setLong(obj, ((Long)value).longValue());
|
|
case "S" -> f.setShort(obj, ((Short)value).shortValue());
|
|
case "Z" -> f.setBoolean(obj, ((Boolean)value).booleanValue());
|
|
}
|
|
fail("expected IllegalAccessException");
|
|
} catch (IllegalAccessException e) {
|
|
// expected
|
|
}
|
|
|
|
// test null value only if it's primitive type
|
|
try {
|
|
f.set(obj, null);
|
|
fail("expected IllegalAccessException");
|
|
} catch (IllegalAccessException e) {
|
|
// expected
|
|
}
|
|
}
|
|
|
|
Object badValue = new NegativeTest();
|
|
try {
|
|
f.set(obj, badValue);
|
|
fail("expected IllegalAccessException");
|
|
} catch (IllegalAccessException e) {
|
|
// expected
|
|
}
|
|
}
|
|
|
|
|
|
|
|
@DataProvider(name = "finalInstanceFields")
|
|
private Object[][] finalInstanceFields() {
|
|
return new Object[][]{
|
|
new Object[]{fi_field, Integer.valueOf(10)},
|
|
new Object[]{fc_field, Character.valueOf('c')},
|
|
new Object[]{fb_field, Byte.valueOf((byte) 1)},
|
|
new Object[]{fs_field, Short.valueOf((short) 2)},
|
|
new Object[]{fl_field, Long.valueOf(1000)},
|
|
new Object[]{fd_field, Double.valueOf(1.2)},
|
|
new Object[]{ff_field, Float.valueOf(2.5f)},
|
|
new Object[]{fz_field, Boolean.valueOf(true)},
|
|
new Object[]{fo_field, "good-value"},
|
|
};
|
|
}
|
|
|
|
/*
|
|
* Test Field::set on a final instance field with either a bad receiver
|
|
* or null. IllegalArgumentException is thrown if the receiver is of
|
|
* a bad type. NullPointerException is thrown if the receiver is null.
|
|
* The receiver is checked before the access check is performed and
|
|
* also before the value is checked.
|
|
*/
|
|
@Test(dataProvider = "finalInstanceFields")
|
|
public void testReceiverOnFinalField(Field f, Object value) {
|
|
assertTrue(Modifier.isFinal(f.getModifiers()));
|
|
Object badReceiver = new NegativeTest();
|
|
// set the field with a bad receiver with a good value
|
|
try {
|
|
f.set(badReceiver, value);
|
|
fail("expected IllegalArgumentException");
|
|
} catch (IllegalArgumentException e) {
|
|
// expected
|
|
} catch (IllegalAccessException e) {
|
|
throw new RuntimeException("Expected IllegalArgumentException but got: " + e.getMessage(), e);
|
|
}
|
|
|
|
// set the field with a bad receiver with a bad value
|
|
Object badValue = new NegativeTest();
|
|
try {
|
|
f.set(badReceiver, badValue);
|
|
fail("expected IllegalArgumentException");
|
|
} catch (IllegalArgumentException e) {
|
|
// expected
|
|
} catch (IllegalAccessException e) {
|
|
throw new RuntimeException("Expected IllegalArgumentException but got: " + e.getMessage(), e);
|
|
}
|
|
|
|
// set the field with a null receiver with a good value
|
|
try {
|
|
f.set(null, value);
|
|
fail("expected NullPointerException");
|
|
} catch (NullPointerException e) {
|
|
// expected
|
|
} catch (IllegalAccessException e) {
|
|
throw new RuntimeException("Expected NullPointerException but got: " + e.getMessage(), e);
|
|
}
|
|
// set the field with a null receiver with a bad value
|
|
try {
|
|
f.set(null, badValue);
|
|
fail("expected NullPointerException");
|
|
} catch (NullPointerException e) {
|
|
// expected
|
|
} catch (IllegalAccessException e) {
|
|
throw new RuntimeException("Expected NullPointerException but got: " + e.getMessage(), e);
|
|
}
|
|
|
|
Class<?> fType = f.getType();
|
|
if (fType.isPrimitive()) {
|
|
// test bad receiver
|
|
try {
|
|
switch (fType.descriptorString()) {
|
|
case "B" -> f.setByte(badReceiver, ((Byte) value).byteValue());
|
|
case "C" -> f.setChar(badReceiver, ((Character) value).charValue());
|
|
case "D" -> f.setDouble(badReceiver, ((Double) value).doubleValue());
|
|
case "F" -> f.setFloat(badReceiver, ((Float) value).floatValue());
|
|
case "I" -> f.setInt(badReceiver, ((Integer) value).intValue());
|
|
case "J" -> f.setLong(badReceiver, ((Long) value).longValue());
|
|
case "S" -> f.setShort(badReceiver, ((Short) value).shortValue());
|
|
case "Z" -> f.setBoolean(badReceiver, ((Boolean) value).booleanValue());
|
|
}
|
|
} catch (IllegalArgumentException e) {
|
|
} catch (IllegalAccessException e) {
|
|
throw new RuntimeException("Expected IllegalArgumentException but got: " + e.getMessage(), e);
|
|
}
|
|
// test null receiver
|
|
try {
|
|
switch (fType.descriptorString()) {
|
|
case "B" -> f.setByte(null, ((Byte) value).byteValue());
|
|
case "C" -> f.setChar(null, ((Character) value).charValue());
|
|
case "D" -> f.setDouble(null, ((Double) value).doubleValue());
|
|
case "F" -> f.setFloat(null, ((Float) value).floatValue());
|
|
case "I" -> f.setInt(null, ((Integer) value).intValue());
|
|
case "J" -> f.setLong(null, ((Long) value).longValue());
|
|
case "S" -> f.setShort(null, ((Short) value).shortValue());
|
|
case "Z" -> f.setBoolean(null, ((Boolean) value).booleanValue());
|
|
}
|
|
} catch (NullPointerException e) {
|
|
// expected
|
|
} catch (IllegalAccessException e) {
|
|
throw new RuntimeException("Expected NullPointerException but got: " + e.getMessage(), e);
|
|
}
|
|
}
|
|
}
|
|
}
|