summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorNeil Fuller <nfuller@google.com>2014-08-28 13:00:38 +0000
committerAndroid Git Automerger <android-git-automerger@android.com>2014-08-28 13:00:38 +0000
commit38157bed6b89d9d39f4541d1bdef10518e5728af (patch)
tree9ee21322b5311aa2e4161a9e7d0a972e1a330668
parent0753af6a0deb63ed09700800a70ca8b408cbe022 (diff)
parent7f9653f5d02beff35b8704b3a075f2e581db6135 (diff)
downloadlibcore-38157bed6b89d9d39f4541d1bdef10518e5728af.zip
libcore-38157bed6b89d9d39f4541d1bdef10518e5728af.tar.gz
libcore-38157bed6b89d9d39f4541d1bdef10518e5728af.tar.bz2
am 7f9653f5: am 55848681: Add additional field checks for deserialization.
* commit '7f9653f5d02beff35b8704b3a075f2e581db6135': Add additional field checks for deserialization.
-rw-r--r--luni/src/main/java/java/io/ObjectInputStream.java11
-rw-r--r--luni/src/main/java/java/io/ObjectOutputStream.java8
-rw-r--r--luni/src/main/java/java/io/ObjectStreamClass.java36
-rw-r--r--luni/src/test/java/libcore/java/io/SerializationTest.java92
4 files changed, 129 insertions, 18 deletions
diff --git a/luni/src/main/java/java/io/ObjectInputStream.java b/luni/src/main/java/java/io/ObjectInputStream.java
index cd7f736..3a89b52 100644
--- a/luni/src/main/java/java/io/ObjectInputStream.java
+++ b/luni/src/main/java/java/io/ObjectInputStream.java
@@ -23,7 +23,6 @@ import java.lang.reflect.Array;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
-import java.lang.reflect.Modifier;
import java.lang.reflect.Proxy;
import java.util.ArrayList;
import java.util.Arrays;
@@ -1064,12 +1063,10 @@ public class ObjectInputStream extends InputStream implements ObjectInput, Objec
}
for (ObjectStreamField fieldDesc : fields) {
- Field field = classDesc.getReflectionField(fieldDesc);
- if (field != null && Modifier.isTransient(field.getModifiers())) {
- field = null; // No setting transient fields! (http://b/4471249)
- }
- // We may not have been able to find the field, or it may be transient, but we still
- // need to read the value and do the other checking...
+ // checkAndGetReflectionField() can return null if it was not able to find the field or
+ // if it is transient or static. We still need to read the data and do the other
+ // checking...
+ Field field = classDesc.checkAndGetReflectionField(fieldDesc);
try {
Class<?> type = fieldDesc.getTypeInternal();
if (type == byte.class) {
diff --git a/luni/src/main/java/java/io/ObjectOutputStream.java b/luni/src/main/java/java/io/ObjectOutputStream.java
index bb6711b..f67919d 100644
--- a/luni/src/main/java/java/io/ObjectOutputStream.java
+++ b/luni/src/main/java/java/io/ObjectOutputStream.java
@@ -928,9 +928,11 @@ public class ObjectOutputStream extends OutputStream implements ObjectOutput, Ob
for (ObjectStreamField fieldDesc : classDesc.fields()) {
try {
Class<?> type = fieldDesc.getTypeInternal();
- Field field = classDesc.getReflectionField(fieldDesc);
+ Field field = classDesc.checkAndGetReflectionField(fieldDesc);
if (field == null) {
- throw new InvalidClassException(classDesc.getName() + " doesn't have a field " + fieldDesc.getName() + " of type " + type);
+ throw new InvalidClassException(classDesc.getName()
+ + " doesn't have a serializable field " + fieldDesc.getName()
+ + " of type " + type);
}
if (type == byte.class) {
output.writeByte(field.getByte(obj));
@@ -1728,7 +1730,7 @@ public class ObjectOutputStream extends OutputStream implements ObjectOutput, Ob
// Only write field "name" for enum class, which is the second field of
// enum, that is fields[1]. Ignore all non-fields and fields.length < 2
if (fields != null && fields.length > 1) {
- Field field = classDesc.getSuperclass().getReflectionField(fields[1]);
+ Field field = classDesc.getSuperclass().checkAndGetReflectionField(fields[1]);
if (field == null) {
throw new NoSuchFieldError();
}
diff --git a/luni/src/main/java/java/io/ObjectStreamClass.java b/luni/src/main/java/java/io/ObjectStreamClass.java
index 9b7f2c9..1a27c9d 100644
--- a/luni/src/main/java/java/io/ObjectStreamClass.java
+++ b/luni/src/main/java/java/io/ObjectStreamClass.java
@@ -185,26 +185,46 @@ public class ObjectStreamClass implements Serializable {
return constructor;
}
- Field getReflectionField(ObjectStreamField osf) {
+ /**
+ * Returns the {@link Field} referred to by {@link ObjectStreamField} for the class described by
+ * this {@link ObjectStreamClass}. A {@code null} value is returned if the local definition of
+ * the field does not meet the criteria for a serializable / deserializable field, i.e. the
+ * field must be non-static and non-transient. Caching of each field lookup is performed. The
+ * first time a field is returned it is made accessible with a call to
+ * {@link Field#setAccessible(boolean)}.
+ */
+ Field checkAndGetReflectionField(ObjectStreamField osf) {
synchronized (reflectionFields) {
Field field = reflectionFields.get(osf);
- if (field != null) {
+ // null might indicate a cache miss or a hit and a non-serializable field so we
+ // check for a mapping.
+ if (field != null || reflectionFields.containsKey(osf)) {
return field;
}
}
+ Field field;
try {
Class<?> declaringClass = forClass();
- Field field = declaringClass.getDeclaredField(osf.getName());
- field.setAccessible(true);
- synchronized (reflectionFields) {
- reflectionFields.put(osf, field);
+ field = declaringClass.getDeclaredField(osf.getName());
+
+ int modifiers = field.getModifiers();
+ if (Modifier.isStatic(modifiers) || Modifier.isTransient(modifiers)) {
+ // No serialization or deserialization of transient or static fields!
+ // See http://b/4471249 and http://b/17202597.
+ field = null;
+ } else {
+ field.setAccessible(true);
}
- return reflectionFields.get(osf);
} catch (NoSuchFieldException ex) {
// The caller messed up. We'll return null and won't try to resolve this again.
- return null;
+ field = null;
+ }
+
+ synchronized (reflectionFields) {
+ reflectionFields.put(osf, field);
}
+ return field;
}
/*
diff --git a/luni/src/test/java/libcore/java/io/SerializationTest.java b/luni/src/test/java/libcore/java/io/SerializationTest.java
index 1002cf1..32bc402 100644
--- a/luni/src/test/java/libcore/java/io/SerializationTest.java
+++ b/luni/src/test/java/libcore/java/io/SerializationTest.java
@@ -20,6 +20,7 @@ import junit.framework.TestCase;
import java.io.InvalidClassException;
import java.io.InvalidObjectException;
+import java.io.NotSerializableException;
import java.io.ObjectStreamClass;
import java.io.ObjectStreamField;
import java.io.Serializable;
@@ -52,6 +53,97 @@ public final class SerializationTest extends TestCase {
private int nonTransientInt;
}
+ public void testSerializeFieldMadeStatic() throws Exception {
+ // Does ObjectStreamClass have the right idea?
+ ObjectStreamClass osc = ObjectStreamClass.lookup(FieldMadeStatic.class);
+ ObjectStreamField[] fields = osc.getFields();
+ assertEquals(0, fields.length);
+
+ // This was created by serializing a FieldMadeStatic with a non-static staticInt
+ String s = "aced0005737200316c6962636f72652e6a6176612e696f2e53657269616c697a6174696f6e54657"
+ + "374244669656c644d6164655374617469630000000000000000020001490009737461746963496e7"
+ + "47870000022b8";
+ FieldMadeStatic deserialized = (FieldMadeStatic) SerializationTester.deserializeHex(s);
+ // The field data is simply ignored if it is static.
+ assertEquals(9999, deserialized.staticInt);
+ }
+
+ static class FieldMadeStatic implements Serializable {
+ private static final long serialVersionUID = 0L;
+ // private int staticInt = 8888;
+ private static int staticInt = 9999;
+ }
+
+ // We can serialize an object that has an unserializable field providing it is null.
+ public void testDeserializeNullUnserializableField() throws Exception {
+ // This was created by creating a new SerializableContainer and not setting the
+ // unserializable field. A canned serialized form is used so we can tell if the static
+ // initializers were executed during deserialization.
+ // SerializationTester.serializeHex(new SerializableContainer());
+ String s = "aced0005737200376c6962636f72652e6a6176612e696f2e53657269616c697a6174696f6e54657"
+ + "3742453657269616c697a61626c65436f6e7461696e657200000000000000000200014c000e756e7"
+ + "3657269616c697a61626c657400334c6c6962636f72652f6a6176612f696f2f53657269616c697a6"
+ + "174696f6e546573742457617353657269616c697a61626c653b787070";
+
+ serializableContainerInitializedFlag = false;
+ wasSerializableInitializedFlag = false;
+
+ SerializableContainer sc = (SerializableContainer) SerializationTester.deserializeHex(s);
+ assertNull(sc.unserializable);
+
+ // Confirm the container was initialized, but the class for the null field was not.
+ assertTrue(serializableContainerInitializedFlag);
+ assertFalse(wasSerializableInitializedFlag);
+ }
+
+ public static boolean serializableContainerInitializedFlag = false;
+
+ static class SerializableContainer implements Serializable {
+ private static final long serialVersionUID = 0L;
+ private Object unserializable = null;
+
+ static {
+ serializableContainerInitializedFlag = true;
+ }
+ }
+
+ // We must not serialize an object that has a non-null unserializable field.
+ public void testSerializeUnserializableField() throws Exception {
+ SerializableContainer sc = new SerializableContainer();
+ sc.unserializable = new WasSerializable();
+ try {
+ SerializationTester.serializeHex(sc);
+ fail();
+ } catch (NotSerializableException expected) {
+ }
+ }
+
+ // It must not be possible to deserialize an object if a field is no longer serializable.
+ public void testDeserializeUnserializableField() throws Exception {
+ // This was generated by creating a SerializableContainer and setting the unserializable
+ // field to a WasSerializable when it was still Serializable. A canned serialized form is
+ // used so we can tell if the static initializers were executed during deserialization.
+ // SerializableContainer sc = new SerializableContainer();
+ // sc.unserializable = new WasSerializable();
+ // SerializationTester.serializeHex(sc);
+ String s = "aced0005737200376c6962636f72652e6a6176612e696f2e53657269616c697a6174696f6e54657"
+ + "3742453657269616c697a61626c65436f6e7461696e657200000000000000000200014c000e756e7"
+ + "3657269616c697a61626c657400124c6a6176612f6c616e672f4f626a6563743b7870737200316c6"
+ + "962636f72652e6a6176612e696f2e53657269616c697a6174696f6e5465737424576173536572696"
+ + "16c697a61626c65000000000000000002000149000169787000000000";
+
+ serializableContainerInitializedFlag = false;
+ wasSerializableInitializedFlag = false;
+ try {
+ SerializationTester.deserializeHex(s);
+ fail();
+ } catch (InvalidClassException expected) {
+ }
+ // Confirm neither the container nor the contained class was initialized.
+ assertFalse(serializableContainerInitializedFlag);
+ assertFalse(wasSerializableInitializedFlag);
+ }
+
public void testSerialVersionUidChange() throws Exception {
// this was created by serializing a SerialVersionUidChanged with serialVersionUID = 0L
String s = "aced0005737200396c6962636f72652e6a6176612e696f2e53657269616c697a6174696f6e54657"