Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

8333796: Add missing serialization functionality to sun.reflect.ReflectionFactory #19702

Closed
wants to merge 40 commits into from
Closed
Changes from all commits
Commits
Show all changes
40 commits
Select commit Hold shift + click to select a range
430b5d3
WIP: Generate readObject/writeObject variants
dmlloyd Jun 12, 2024
12eb7e8
Use `unreflectGetter` for access checking
dmlloyd Jun 13, 2024
2f0642a
Move bootstrap method to `ConstantBootstraps`
dmlloyd Jun 13, 2024
d36f05e
Restrict using new methods with enums, records, hidden
dmlloyd Jun 13, 2024
93d02ec
Add `@since` tag
dmlloyd Jun 13, 2024
797caca
Small optimization in `fieldSetterForSerialization`
dmlloyd Jun 13, 2024
265ef31
Copyright
dmlloyd Jun 13, 2024
a78ad0d
More cleanups, forbid usage on Externalizable classes
dmlloyd Jun 13, 2024
d53fc77
Use Modifier.isFinal()
dmlloyd Jun 13, 2024
12acab0
Copyright
dmlloyd Jun 13, 2024
48aeac5
Minor docs fix, use Modifier.isXxx() on constant bootstrap, use conve…
dmlloyd Jun 13, 2024
11738ff
Try a safer access checking strategy
dmlloyd Jun 13, 2024
0941a5d
Improve documentation of `findSetterForSerialization`
dmlloyd Jun 13, 2024
d88006b
Centralize the check for class serializability, add more cases therei…
dmlloyd Jun 14, 2024
b81b1c7
Instead of using a public constant bootstrap, define the readObject h…
dmlloyd Jun 27, 2024
64d32cc
Interim access solution, plus test
dmlloyd Jun 27, 2024
4e68264
Avoid security manager error and improve test name
dmlloyd Jun 27, 2024
8d46c28
Move generator to JLI
dmlloyd Jun 27, 2024
2c9b1c5
Move bytecode gen back to `jdk.internal.reflect`, use unprivileged lo…
dmlloyd Jun 28, 2024
a9418da
Unused field
dmlloyd Jun 28, 2024
0d3def8
Neaten if statement
dmlloyd Jun 28, 2024
d137f43
Fix security manager permissions for test
dmlloyd Jun 28, 2024
236a073
Apply review feedback
dmlloyd Jul 1, 2024
d6f5ccd
Missed one
dmlloyd Jul 1, 2024
e23d01f
Add tests for static accessors
dmlloyd Jul 1, 2024
b2cc30d
More tests and fixes
dmlloyd Jul 1, 2024
a02fb6e
Restrict field set to those defined by serialPersistentFields
dmlloyd Jul 1, 2024
e733465
More tests
dmlloyd Jul 5, 2024
7155da1
Rework using facilities found in ObjectStreamClass
dmlloyd Aug 27, 2024
1d8f1b2
Eliminate cache
dmlloyd Aug 27, 2024
2a734c5
Address review comment
dmlloyd Aug 29, 2024
bf658a3
Address review feedback
dmlloyd Sep 20, 2024
34d4523
Test fixes and finish renaming
dmlloyd Sep 24, 2024
38f7b79
Address review feedback
dmlloyd Sep 25, 2024
11bc95e
Move `serialPersistentFields` for a degree of method order consistency
dmlloyd Oct 18, 2024
997dba0
Round out the documentation of the new methods to explain the support…
dmlloyd Oct 22, 2024
d57c434
Merge remote-tracking branch 'upstream-jdk/master' into serialization
dmlloyd Nov 13, 2024
77c56b1
Merge remote-tracking branch 'upstream-jdk/master' into serialization
dmlloyd Nov 13, 2024
7a85d62
Address review feedback
dmlloyd Nov 13, 2024
8b83bd7
Merge remote-tracking branch 'upstream-jdk/master' into serialization
dmlloyd Nov 14, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
177 changes: 177 additions & 0 deletions src/java.base/share/classes/java/io/ObjectStreamReflection.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,177 @@
/*
* Copyright (c) 2024, 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. Oracle designates this
* particular file as subject to the "Classpath" exception as provided
* by Oracle in the LICENSE file that accompanied this code.
*
* 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.
*/

package java.io;

import java.lang.invoke.MethodHandle;
import java.lang.invoke.MethodHandles;
import java.lang.invoke.MethodType;

import jdk.internal.access.JavaObjectStreamReflectionAccess;
import jdk.internal.access.SharedSecrets;
import jdk.internal.util.ByteArray;

/**
* Utilities relating to serialization and deserialization of objects.
*/
final class ObjectStreamReflection {

// todo: these could be constants
private static final MethodHandle DRO_HANDLE;
private static final MethodHandle DWO_HANDLE;

static {
try {
MethodHandles.Lookup lookup = MethodHandles.lookup();
MethodType droType = MethodType.methodType(void.class, ObjectStreamClass.class, Object.class, ObjectInputStream.class);
DRO_HANDLE = lookup.findStatic(ObjectStreamReflection.class, "defaultReadObject", droType);
MethodType dwoType = MethodType.methodType(void.class, ObjectStreamClass.class, Object.class, ObjectOutputStream.class);
DWO_HANDLE = lookup.findStatic(ObjectStreamReflection.class, "defaultWriteObject", dwoType);
} catch (NoSuchMethodException | IllegalAccessException e) {
throw new InternalError(e);
}
}

/**
* Populate a serializable object from data acquired from the stream's
* {@link java.io.ObjectInputStream.GetField} object independently of
* the actual {@link ObjectInputStream} implementation which may
* arbitrarily override the {@link ObjectInputStream#readFields()} method
* in order to deserialize using a custom object format.
* <p>
* The fields are populated using the mechanism defined in {@link ObjectStreamClass},
* which requires objects and primitives to each be packed into a separate array
* whose relative field offsets are defined in the {@link ObjectStreamField}
* corresponding to each field.
* Utility methods on the {@code ObjectStreamClass} instance are then used
* to validate and perform the actual field accesses.
*
* @param streamClass the object stream class of the object (must not be {@code null})
* @param obj the object to deserialize (must not be {@code null})
* @param ois the object stream (must not be {@code null})
* @throws IOException if the call to {@link ObjectInputStream#readFields}
* or one of its field accessors throws this exception type
* @throws ClassNotFoundException if the call to {@link ObjectInputStream#readFields}
* or one of its field accessors throws this exception type
*/
private static void defaultReadObject(ObjectStreamClass streamClass, Object obj, ObjectInputStream ois)
throws IOException, ClassNotFoundException {
ObjectInputStream.GetField getField = ois.readFields();
byte[] bytes = new byte[streamClass.getPrimDataSize()];
Object[] objs = new Object[streamClass.getNumObjFields()];
for (ObjectStreamField field : streamClass.getFields(false)) {
int offset = field.getOffset();
String fieldName = field.getName();
switch (field.getTypeCode()) {
case 'B' -> bytes[offset] = getField.get(fieldName, (byte) 0);
case 'C' -> ByteArray.setChar(bytes, offset, getField.get(fieldName, (char) 0));
case 'D' -> ByteArray.setDoubleRaw(bytes, offset, getField.get(fieldName, 0.0));
case 'F' -> ByteArray.setFloatRaw(bytes, offset, getField.get(fieldName, 0.0f));
case 'I' -> ByteArray.setInt(bytes, offset, getField.get(fieldName, 0));
case 'J' -> ByteArray.setLong(bytes, offset, getField.get(fieldName, 0L));
case 'S' -> ByteArray.setShort(bytes, offset, getField.get(fieldName, (short) 0));
case 'Z' -> ByteArray.setBoolean(bytes, offset, getField.get(fieldName, false));
case '[', 'L' -> objs[offset] = getField.get(fieldName, null);
default -> throw new IllegalStateException();
}
}
streamClass.checkObjFieldValueTypes(obj, objs);
streamClass.setPrimFieldValues(obj, bytes);
streamClass.setObjFieldValues(obj, objs);
}

/**
* Populate and write a stream's {@link java.io.ObjectOutputStream.PutField} object
* from field data acquired from a serializable object independently of
* the actual {@link ObjectOutputStream} implementation which may
* arbitrarily override the {@link ObjectOutputStream#putFields()}
* and {@link ObjectOutputStream#writeFields()} methods
* in order to deserialize using a custom object format.
* <p>
* The fields are accessed using the mechanism defined in {@link ObjectStreamClass},
* which causes objects and primitives to each be packed into a separate array
* whose relative field offsets are defined in the {@link ObjectStreamField}
* corresponding to each field.
*
* @param streamClass the object stream class of the object (must not be {@code null})
* @param obj the object to serialize (must not be {@code null})
* @param oos the object stream (must not be {@code null})
* @throws IOException if the call to {@link ObjectInputStream#readFields}
* or one of its field accessors throws this exception type
*/
private static void defaultWriteObject(ObjectStreamClass streamClass, Object obj, ObjectOutputStream oos)
throws IOException {
ObjectOutputStream.PutField putField = oos.putFields();
byte[] bytes = new byte[streamClass.getPrimDataSize()];
Object[] objs = new Object[streamClass.getNumObjFields()];
streamClass.getPrimFieldValues(obj, bytes);
streamClass.getObjFieldValues(obj, objs);
for (ObjectStreamField field : streamClass.getFields(false)) {
int offset = field.getOffset();
String fieldName = field.getName();
switch (field.getTypeCode()) {
case 'B' -> putField.put(fieldName, bytes[offset]);
case 'C' -> putField.put(fieldName, ByteArray.getChar(bytes, offset));
case 'D' -> putField.put(fieldName, ByteArray.getDouble(bytes, offset));
case 'F' -> putField.put(fieldName, ByteArray.getFloat(bytes, offset));
case 'I' -> putField.put(fieldName, ByteArray.getInt(bytes, offset));
case 'J' -> putField.put(fieldName, ByteArray.getLong(bytes, offset));
case 'S' -> putField.put(fieldName, ByteArray.getShort(bytes, offset));
case 'Z' -> putField.put(fieldName, ByteArray.getBoolean(bytes, offset));
case '[', 'L' -> putField.put(fieldName, objs[offset]);
default -> throw new IllegalStateException();
}
}
oos.writeFields();
}

static final class Access implements JavaObjectStreamReflectionAccess {
static {
SharedSecrets.setJavaObjectStreamReflectionAccess(new Access());
}

public MethodHandle defaultReadObject(Class<?> clazz) {
return handleForClass(DRO_HANDLE, clazz, ObjectInputStream.class);
}

public MethodHandle defaultWriteObject(Class<?> clazz) {
return handleForClass(DWO_HANDLE, clazz, ObjectOutputStream.class);
}

private static MethodHandle handleForClass(final MethodHandle handle, final Class<?> clazz, final Class<?> ioClass) {
ObjectStreamClass streamClass = ObjectStreamClass.lookup(clazz);
if (streamClass != null) {
try {
streamClass.checkDefaultSerialize();
return handle.bindTo(streamClass)
.asType(MethodType.methodType(void.class, clazz, ioClass));
} catch (InvalidClassException e) {
// ignore and return null
}
}
return null;
}
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,33 @@
/*
* Copyright (c) 2024, 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. Oracle designates this
* particular file as subject to the "Classpath" exception as provided
* by Oracle in the LICENSE file that accompanied this code.
*
* 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.
*/

package jdk.internal.access;

import java.lang.invoke.MethodHandle;

public interface JavaObjectStreamReflectionAccess {
MethodHandle defaultReadObject(Class<?> clazz);
MethodHandle defaultWriteObject(Class<?> clazz);
}
16 changes: 16 additions & 0 deletions src/java.base/share/classes/jdk/internal/access/SharedSecrets.java
Original file line number Diff line number Diff line change
@@ -79,6 +79,7 @@ public class SharedSecrets {
private static JavaObjectInputStreamReadString javaObjectInputStreamReadString;
private static JavaObjectInputStreamAccess javaObjectInputStreamAccess;
private static JavaObjectInputFilterAccess javaObjectInputFilterAccess;
private static JavaObjectStreamReflectionAccess javaObjectStreamReflectionAccess;
private static JavaNetInetAddressAccess javaNetInetAddressAccess;
private static JavaNetHttpCookieAccess javaNetHttpCookieAccess;
private static JavaNetUriAccess javaNetUriAccess;
@@ -459,6 +460,21 @@ public static void setJavaObjectInputFilterAccess(JavaObjectInputFilterAccess ac
javaObjectInputFilterAccess = access;
}

public static JavaObjectStreamReflectionAccess getJavaObjectStreamReflectionAccess() {
var access = javaObjectStreamReflectionAccess;
if (access == null) {
try {
Class.forName("java.io.ObjectStreamReflection$Access", true, null);
access = javaObjectStreamReflectionAccess;
} catch (ClassNotFoundException e) {}
}
return access;
}

public static void setJavaObjectStreamReflectionAccess(JavaObjectStreamReflectionAccess access) {
javaObjectStreamReflectionAccess = access;
}

public static void setJavaIORandomAccessFileAccess(JavaIORandomAccessFileAccess jirafa) {
javaIORandomAccessFileAccess = jirafa;
}
Original file line number Diff line number Diff line change
@@ -29,6 +29,7 @@
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.ObjectStreamClass;
import java.io.ObjectStreamField;
import java.io.OptionalDataException;
import java.io.Serializable;
import java.lang.invoke.MethodHandle;
@@ -39,7 +40,10 @@
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.lang.reflect.Proxy;
import java.security.PrivilegedAction;
import java.util.Set;

import jdk.internal.access.JavaLangReflectAccess;
import jdk.internal.access.SharedSecrets;
import jdk.internal.misc.VM;
@@ -66,6 +70,7 @@ public class ReflectionFactory {
private static volatile Method hasStaticInitializerMethod;

private final JavaLangReflectAccess langReflectAccess;

private ReflectionFactory() {
this.langReflectAccess = SharedSecrets.getJavaLangReflectAccess();
}
@@ -363,6 +368,46 @@ private final MethodHandle findReadWriteObjectForSerialization(Class<?> cl,
}
}

public final MethodHandle defaultReadObjectForSerialization(Class<?> cl) {
if (hasDefaultOrNoSerialization(cl)) {
return null;
}

return SharedSecrets.getJavaObjectStreamReflectionAccess().defaultReadObject(cl);
}

public final MethodHandle defaultWriteObjectForSerialization(Class<?> cl) {
if (hasDefaultOrNoSerialization(cl)) {
return null;
}

return SharedSecrets.getJavaObjectStreamReflectionAccess().defaultWriteObject(cl);
}

/**
* These are specific leaf classes which appear to be Serializable, but which
* have special semantics according to the serialization specification. We
* could theoretically include array classes here, but it is easier and clearer
* to just use `Class#isArray` instead.
*/
private static final Set<Class<?>> nonSerializableLeafClasses = Set.of(
Class.class,
String.class,
ObjectStreamClass.class
);

private static boolean hasDefaultOrNoSerialization(Class<?> cl) {
return ! Serializable.class.isAssignableFrom(cl)
|| cl.isInterface()
|| cl.isArray()
|| Proxy.isProxyClass(cl)
|| Externalizable.class.isAssignableFrom(cl)
|| cl.isEnum()
|| cl.isRecord()
|| cl.isHidden()
|| nonSerializableLeafClasses.contains(cl);
}

/**
* Returns a MethodHandle for {@code writeReplace} on the serializable class
* or null if no match found.
@@ -468,6 +513,28 @@ public final Constructor<OptionalDataException> newOptionalDataExceptionForSeria
}
}

public final ObjectStreamField[] serialPersistentFields(Class<?> cl) {
if (! Serializable.class.isAssignableFrom(cl) || cl.isInterface() || cl.isEnum()) {
return null;
}

try {
Field field = cl.getDeclaredField("serialPersistentFields");
int mods = field.getModifiers();
if (! (Modifier.isStatic(mods) && Modifier.isPrivate(mods) && Modifier.isFinal(mods))) {
return null;
}
if (field.getType() != ObjectStreamField[].class) {
return null;
}
field.setAccessible(true);
ObjectStreamField[] array = (ObjectStreamField[]) field.get(null);
return array != null && array.length > 0 ? array.clone() : array;
} catch (ReflectiveOperationException e) {
return null;
}
}

//--------------------------------------------------------------------------
//
// Internals only below this point
@@ -556,5 +623,4 @@ private static boolean packageEquals(Class<?> cl1, Class<?> cl2) {
return cl1.getClassLoader() == cl2.getClassLoader() &&
cl1.getPackageName() == cl2.getPackageName();
}

}
Loading