2013-07-03 02:52:38 +00:00
|
|
|
/* Copyright (c) 2008-2013, 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-29 01:29:01 +00:00
|
|
|
package java.io;
|
|
|
|
|
2013-10-29 15:20:55 +00:00
|
|
|
import static java.io.ObjectOutputStream.STREAM_MAGIC;
|
|
|
|
import static java.io.ObjectOutputStream.STREAM_VERSION;
|
|
|
|
import static java.io.ObjectOutputStream.TC_NULL;
|
|
|
|
import static java.io.ObjectOutputStream.TC_REFERENCE;
|
|
|
|
import static java.io.ObjectOutputStream.TC_CLASSDESC;
|
|
|
|
import static java.io.ObjectOutputStream.TC_OBJECT;
|
|
|
|
import static java.io.ObjectOutputStream.TC_STRING;
|
|
|
|
import static java.io.ObjectOutputStream.TC_ARRAY;
|
|
|
|
import static java.io.ObjectOutputStream.TC_CLASS;
|
|
|
|
import static java.io.ObjectOutputStream.TC_BLOCKDATA;
|
|
|
|
import static java.io.ObjectOutputStream.TC_ENDBLOCKDATA;
|
|
|
|
import static java.io.ObjectOutputStream.TC_RESET;
|
|
|
|
import static java.io.ObjectOutputStream.TC_BLOCKDATALONG;
|
|
|
|
import static java.io.ObjectOutputStream.TC_EXCEPTION;
|
|
|
|
import static java.io.ObjectOutputStream.TC_LONGSTRING;
|
|
|
|
import static java.io.ObjectOutputStream.TC_PROXYCLASSDESC;
|
|
|
|
import static java.io.ObjectOutputStream.TC_ENUM;
|
|
|
|
import static java.io.ObjectOutputStream.SC_WRITE_METHOD;
|
|
|
|
import static java.io.ObjectOutputStream.SC_BLOCK_DATA;
|
|
|
|
import static java.io.ObjectOutputStream.SC_SERIALIZABLE;
|
|
|
|
import static java.io.ObjectOutputStream.SC_EXTERNALIZABLE;
|
|
|
|
import static java.io.ObjectOutputStream.SC_ENUM;
|
2013-10-30 17:46:35 +00:00
|
|
|
import static java.io.ObjectOutputStream.getReadOrWriteMethod;
|
2013-10-29 15:20:55 +00:00
|
|
|
|
2011-07-01 14:43:43 +00:00
|
|
|
import avian.VMClass;
|
|
|
|
|
2007-08-13 00:50:25 +00:00
|
|
|
import java.util.HashMap;
|
|
|
|
import java.lang.reflect.Array;
|
|
|
|
import java.lang.reflect.Field;
|
2013-10-30 17:46:35 +00:00
|
|
|
import java.lang.reflect.Method;
|
2007-08-13 00:50:25 +00:00
|
|
|
import java.lang.reflect.Modifier;
|
|
|
|
|
2013-10-29 15:20:55 +00:00
|
|
|
public class ObjectInputStream extends InputStream implements DataInput {
|
2007-07-29 01:29:01 +00:00
|
|
|
private final InputStream in;
|
|
|
|
|
2013-10-29 15:20:55 +00:00
|
|
|
public ObjectInputStream(InputStream in) throws IOException {
|
2007-07-29 01:29:01 +00:00
|
|
|
this.in = in;
|
2013-10-29 15:20:55 +00:00
|
|
|
short signature = (short)rawShort();
|
|
|
|
if (signature != STREAM_MAGIC) {
|
|
|
|
throw new IOException("Unrecognized signature: 0x"
|
|
|
|
+ Integer.toHexString(signature));
|
|
|
|
}
|
|
|
|
int version = rawShort();
|
|
|
|
if (version != STREAM_VERSION) {
|
|
|
|
throw new IOException("Unsupported version: " + version);
|
|
|
|
}
|
2007-07-29 01:29:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public int read() throws IOException {
|
|
|
|
return in.read();
|
|
|
|
}
|
|
|
|
|
2013-10-29 15:20:55 +00:00
|
|
|
private int rawByte() throws IOException {
|
|
|
|
int c = read();
|
|
|
|
if (c < 0) {
|
|
|
|
throw new EOFException();
|
|
|
|
}
|
|
|
|
return c;
|
2007-07-29 01:29:01 +00:00
|
|
|
}
|
|
|
|
|
2013-10-29 15:20:55 +00:00
|
|
|
private int rawShort() throws IOException {
|
|
|
|
return (rawByte() << 8) | rawByte();
|
2007-07-29 01:29:01 +00:00
|
|
|
}
|
2007-08-12 21:01:47 +00:00
|
|
|
|
2013-10-29 15:20:55 +00:00
|
|
|
private int rawInt() throws IOException {
|
|
|
|
return (rawShort() << 16) | rawShort();
|
2007-08-12 21:01:47 +00:00
|
|
|
}
|
|
|
|
|
2013-10-29 15:20:55 +00:00
|
|
|
private long rawLong() throws IOException {
|
|
|
|
return ((rawInt() & 0xffffffffl) << 32) | rawInt();
|
2007-08-12 21:01:47 +00:00
|
|
|
}
|
|
|
|
|
2013-10-29 15:20:55 +00:00
|
|
|
private String rawString() throws IOException {
|
|
|
|
int length = rawShort();
|
|
|
|
byte[] array = new byte[length];
|
|
|
|
readFully(array);
|
|
|
|
return new String(array);
|
2007-08-12 21:01:47 +00:00
|
|
|
}
|
|
|
|
|
2013-10-29 15:20:55 +00:00
|
|
|
public int read(byte[] b, int offset, int length) throws IOException {
|
|
|
|
return in.read(b, offset, length);
|
2007-08-12 21:01:47 +00:00
|
|
|
}
|
|
|
|
|
2013-10-29 15:20:55 +00:00
|
|
|
public void readFully(byte[] b) throws IOException {
|
|
|
|
readFully(b, 0, b.length);
|
2007-08-12 21:01:47 +00:00
|
|
|
}
|
|
|
|
|
2013-10-29 15:20:55 +00:00
|
|
|
public void readFully(byte[] b, int offset, int length) throws IOException {
|
|
|
|
while (length > 0) {
|
|
|
|
int count = read(b, offset, length);
|
|
|
|
if (count < 0) {
|
|
|
|
throw new EOFException("Reached EOF " + length + " bytes too early");
|
|
|
|
}
|
|
|
|
offset += count;
|
|
|
|
length -= count;
|
|
|
|
}
|
2007-08-12 21:01:47 +00:00
|
|
|
}
|
|
|
|
|
2013-10-29 15:20:55 +00:00
|
|
|
public String readLine() throws IOException {
|
|
|
|
int c = read();
|
|
|
|
if (c < 0) {
|
|
|
|
return null;
|
|
|
|
} else if (c == '\n') {
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
StringBuilder builder = new StringBuilder();
|
|
|
|
for (;;) {
|
|
|
|
builder.append((char)c);
|
|
|
|
c = read();
|
|
|
|
if (c < 0 || c == '\n') {
|
|
|
|
return builder.toString();
|
|
|
|
}
|
|
|
|
}
|
2007-08-12 21:01:47 +00:00
|
|
|
}
|
|
|
|
|
2013-10-29 15:20:55 +00:00
|
|
|
public void close() throws IOException {
|
|
|
|
in.close();
|
2007-08-12 21:01:47 +00:00
|
|
|
}
|
|
|
|
|
2013-10-29 15:20:55 +00:00
|
|
|
private int remainingBlockData;
|
2007-08-12 21:01:47 +00:00
|
|
|
|
2013-10-29 15:20:55 +00:00
|
|
|
private int rawBlockDataByte() throws IOException {
|
|
|
|
while (remainingBlockData <= 0) {
|
|
|
|
int b = rawByte();
|
|
|
|
if (b == TC_BLOCKDATA) {
|
|
|
|
remainingBlockData = rawByte();
|
|
|
|
} else {
|
|
|
|
throw new UnsupportedOperationException("Unknown token: 0x"
|
|
|
|
+ Integer.toHexString(b));
|
|
|
|
}
|
2007-08-12 21:01:47 +00:00
|
|
|
}
|
2013-10-29 15:20:55 +00:00
|
|
|
--remainingBlockData;
|
|
|
|
return rawByte();
|
|
|
|
}
|
|
|
|
|
|
|
|
private int rawBlockDataShort() throws IOException {
|
|
|
|
return (rawBlockDataByte() << 8) | rawBlockDataByte();
|
2007-08-12 21:01:47 +00:00
|
|
|
}
|
|
|
|
|
2013-10-29 15:20:55 +00:00
|
|
|
private int rawBlockDataInt() throws IOException {
|
|
|
|
return (rawBlockDataShort() << 16) | rawBlockDataShort();
|
|
|
|
}
|
2007-08-12 21:01:47 +00:00
|
|
|
|
2013-10-29 15:20:55 +00:00
|
|
|
private long rawBlockDataLong() throws IOException {
|
|
|
|
return ((rawBlockDataInt() & 0xffffffffl) << 32) | rawBlockDataInt();
|
2007-08-12 21:01:47 +00:00
|
|
|
}
|
|
|
|
|
2013-10-29 15:20:55 +00:00
|
|
|
public boolean readBoolean() throws IOException {
|
|
|
|
return rawBlockDataByte() != 0;
|
|
|
|
}
|
2007-08-12 21:01:47 +00:00
|
|
|
|
2013-10-29 15:20:55 +00:00
|
|
|
public byte readByte() throws IOException {
|
|
|
|
return (byte)rawBlockDataByte();
|
2007-08-12 21:01:47 +00:00
|
|
|
}
|
|
|
|
|
2013-10-29 15:20:55 +00:00
|
|
|
public char readChar() throws IOException {
|
|
|
|
return (char)rawBlockDataShort();
|
2007-08-12 21:01:47 +00:00
|
|
|
}
|
|
|
|
|
2013-10-29 15:20:55 +00:00
|
|
|
public short readShort() throws IOException {
|
|
|
|
return (short)rawBlockDataShort();
|
2007-08-12 21:01:47 +00:00
|
|
|
}
|
|
|
|
|
2013-10-29 15:20:55 +00:00
|
|
|
public int readInt() throws IOException {
|
|
|
|
return rawBlockDataInt();
|
2007-08-13 00:50:25 +00:00
|
|
|
}
|
|
|
|
|
2013-10-29 15:20:55 +00:00
|
|
|
public long readLong() throws IOException {
|
|
|
|
return rawBlockDataLong();
|
2007-08-12 21:01:47 +00:00
|
|
|
}
|
|
|
|
|
2013-10-29 15:20:55 +00:00
|
|
|
public float readFloat() throws IOException {
|
|
|
|
return Float.intBitsToFloat(rawBlockDataInt());
|
|
|
|
}
|
|
|
|
|
|
|
|
public double readDouble() throws IOException {
|
|
|
|
return Double.longBitsToDouble(rawBlockDataLong());
|
|
|
|
}
|
2007-08-12 21:01:47 +00:00
|
|
|
|
2013-10-29 15:20:55 +00:00
|
|
|
public int readUnsignedByte() throws IOException {
|
|
|
|
return rawBlockDataByte();
|
|
|
|
}
|
2007-08-13 00:50:25 +00:00
|
|
|
|
2013-10-29 15:20:55 +00:00
|
|
|
public int readUnsignedShort() throws IOException {
|
|
|
|
return rawBlockDataShort();
|
2007-08-12 21:01:47 +00:00
|
|
|
}
|
|
|
|
|
2013-10-29 15:20:55 +00:00
|
|
|
public String readUTF() throws IOException {
|
|
|
|
int length = rawBlockDataShort();
|
|
|
|
if (remainingBlockData < length) {
|
|
|
|
throw new IOException("Short block data: "
|
|
|
|
+ remainingBlockData + " < " + length);
|
|
|
|
}
|
|
|
|
byte[] bytes = new byte[length];
|
|
|
|
readFully(bytes);
|
|
|
|
remainingBlockData -= length;
|
|
|
|
return new String(bytes, "UTF-8");
|
|
|
|
}
|
2007-08-13 00:50:25 +00:00
|
|
|
|
2013-10-29 15:20:55 +00:00
|
|
|
public int skipBytes(int count) throws IOException {
|
|
|
|
int i = 0;
|
|
|
|
while (i < count) {
|
|
|
|
if (read() < 0) {
|
|
|
|
return i;
|
2007-08-12 21:01:47 +00:00
|
|
|
}
|
2013-10-29 15:20:55 +00:00
|
|
|
++i;
|
2007-08-12 21:01:47 +00:00
|
|
|
}
|
2013-10-29 15:20:55 +00:00
|
|
|
return count;
|
|
|
|
}
|
2007-08-12 21:01:47 +00:00
|
|
|
|
2013-10-30 13:59:51 +00:00
|
|
|
private static Class charToPrimitiveType(int c) {
|
|
|
|
if (c == 'B') {
|
|
|
|
return Byte.TYPE;
|
|
|
|
} else if (c == 'C') {
|
|
|
|
return Character.TYPE;
|
|
|
|
} else if (c == 'D') {
|
|
|
|
return Double.TYPE;
|
|
|
|
} else if (c == 'F') {
|
|
|
|
return Float.TYPE;
|
|
|
|
} else if (c == 'I') {
|
|
|
|
return Integer.TYPE;
|
|
|
|
} else if (c == 'J') {
|
|
|
|
return Long.TYPE;
|
|
|
|
} else if (c == 'S') {
|
|
|
|
return Short.TYPE;
|
|
|
|
} else if (c == 'Z') {
|
|
|
|
return Boolean.TYPE;
|
|
|
|
}
|
|
|
|
throw new RuntimeException("Unhandled char: " + (char)c);
|
|
|
|
}
|
|
|
|
|
|
|
|
private void expectToken(int token) throws IOException {
|
|
|
|
int c = rawByte();
|
|
|
|
if (c != token) {
|
|
|
|
throw new UnsupportedOperationException("Unexpected token: 0x"
|
|
|
|
+ Integer.toHexString(c));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private void field(Field field, Object o)
|
|
|
|
throws IOException, IllegalArgumentException, IllegalAccessException,
|
|
|
|
ClassNotFoundException
|
|
|
|
{
|
|
|
|
Class type = field.getType();
|
|
|
|
if (!type.isPrimitive()) {
|
|
|
|
field.set(o, readObject());
|
|
|
|
} else {
|
|
|
|
if (type == Byte.TYPE) {
|
|
|
|
field.setByte(o, (byte)rawByte());
|
|
|
|
} else if (type == Character.TYPE) {
|
|
|
|
field.setChar(o, (char)rawShort());
|
|
|
|
} else if (type == Double.TYPE) {
|
|
|
|
field.setDouble(o, Double.longBitsToDouble(rawLong()));
|
|
|
|
} else if (type == Float.TYPE) {
|
|
|
|
field.setFloat(o, Float.intBitsToFloat(rawInt()));
|
|
|
|
} else if (type == Integer.TYPE) {
|
|
|
|
field.setInt(o, rawInt());
|
|
|
|
} else if (type == Long.TYPE) {
|
|
|
|
field.setLong(o, rawLong());
|
|
|
|
} else if (type == Short.TYPE) {
|
|
|
|
field.setShort(o, (short)rawShort());
|
|
|
|
} else if (type == Boolean.TYPE) {
|
|
|
|
field.setBoolean(o, rawByte() != 0);
|
|
|
|
} else {
|
|
|
|
throw new IOException("Unhandled type: " + type);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public Object readObject() throws IOException, ClassNotFoundException {
|
2013-10-30 03:28:22 +00:00
|
|
|
int c = rawByte();
|
|
|
|
if (c == TC_NULL) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
if (c == TC_STRING) {
|
|
|
|
int length = rawShort();
|
|
|
|
byte[] bytes = new byte[length];
|
|
|
|
readFully(bytes);
|
|
|
|
return new String(bytes, "UTF-8");
|
|
|
|
}
|
|
|
|
if (c != TC_OBJECT) {
|
|
|
|
throw new IOException("Unexpected token: 0x"
|
|
|
|
+ Integer.toHexString(c));
|
|
|
|
}
|
2013-10-30 13:59:51 +00:00
|
|
|
|
|
|
|
// class desc
|
|
|
|
expectToken(TC_CLASSDESC);
|
|
|
|
String className = rawString();
|
|
|
|
ClassLoader loader = Thread.currentThread().getContextClassLoader();
|
|
|
|
Class clazz = loader.loadClass(className);
|
|
|
|
long serialVersionUID = rawLong();
|
|
|
|
try {
|
|
|
|
Field field = clazz.getField("serialVersionUID");
|
|
|
|
long expected = field.getLong(null);
|
|
|
|
if (expected != serialVersionUID) {
|
|
|
|
throw new IOException("Incompatible serial version UID: 0x"
|
|
|
|
+ Long.toHexString(serialVersionUID) + " != 0x"
|
|
|
|
+ Long.toHexString(expected));
|
|
|
|
}
|
|
|
|
} catch (Exception ignored) { }
|
|
|
|
|
|
|
|
int flags = rawByte();
|
2013-10-30 17:46:35 +00:00
|
|
|
if ((flags & ~(SC_SERIALIZABLE | SC_WRITE_METHOD)) != 0) {
|
2013-10-30 13:59:51 +00:00
|
|
|
throw new UnsupportedOperationException("Cannot handle flags: 0x"
|
|
|
|
+ Integer.toHexString(flags));
|
|
|
|
}
|
|
|
|
|
|
|
|
int fieldCount = rawShort();
|
|
|
|
Field[] fields = new Field[fieldCount];
|
|
|
|
for (int i = 0; i < fields.length; i++) {
|
|
|
|
int typeChar = rawByte();
|
|
|
|
String fieldName = rawString();
|
|
|
|
try {
|
|
|
|
fields[i] = clazz.getDeclaredField(fieldName);
|
|
|
|
} catch (Exception e) {
|
|
|
|
throw new IOException(e);
|
|
|
|
}
|
|
|
|
Class type;
|
|
|
|
if (typeChar == '[' || typeChar == 'L') {
|
|
|
|
expectToken(TC_STRING);
|
|
|
|
String typeName = rawString();
|
|
|
|
type = loader.loadClass(typeName);
|
|
|
|
} else {
|
|
|
|
type = charToPrimitiveType(typeChar);
|
|
|
|
}
|
|
|
|
if (fields[i].getType() != type) {
|
|
|
|
throw new IOException("Unexpected type of field " + fieldName
|
|
|
|
+ ": expected " + fields[i].getType() + " but got " + type);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
expectToken(TC_ENDBLOCKDATA);
|
|
|
|
expectToken(TC_NULL);
|
|
|
|
|
|
|
|
try {
|
|
|
|
Object o = makeInstance(clazz.vmClass);
|
|
|
|
|
2013-10-30 17:46:35 +00:00
|
|
|
boolean customized = (flags & SC_WRITE_METHOD) != 0;
|
|
|
|
Method readMethod = customized ?
|
|
|
|
getReadOrWriteMethod(o, "readObject") : null;
|
|
|
|
if (readMethod == null) {
|
|
|
|
if (customized) {
|
|
|
|
throw new IOException("Could not find required readObject method in "
|
|
|
|
+ clazz);
|
|
|
|
}
|
|
|
|
defaultReadObject(o, fields);
|
|
|
|
} else {
|
|
|
|
current = o;
|
|
|
|
currentFields = fields;
|
|
|
|
readMethod.invoke(o, this);
|
|
|
|
current = null;
|
|
|
|
currentFields = null;
|
|
|
|
expectToken(TC_ENDBLOCKDATA);
|
2013-10-30 13:59:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return o;
|
|
|
|
} catch (IOException e) {
|
|
|
|
throw e;
|
|
|
|
} catch (Exception e) {
|
|
|
|
throw new IOException(e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-10-30 17:46:35 +00:00
|
|
|
private Object current;
|
|
|
|
private Field[] currentFields;
|
|
|
|
|
2013-10-30 13:59:51 +00:00
|
|
|
public void defaultReadObject() throws IOException {
|
2013-10-30 17:46:35 +00:00
|
|
|
defaultReadObject(current, currentFields);
|
|
|
|
}
|
|
|
|
|
|
|
|
private void defaultReadObject(Object o, Field[] fields) throws IOException {
|
|
|
|
try {
|
|
|
|
for (Field field : fields) {
|
|
|
|
field(field, o);
|
|
|
|
}
|
|
|
|
} catch (IOException e) {
|
|
|
|
throw e;
|
|
|
|
} catch (Exception e) {
|
|
|
|
throw new IOException(e);
|
|
|
|
}
|
2013-10-30 13:59:51 +00:00
|
|
|
}
|
2007-08-13 00:50:25 +00:00
|
|
|
|
2013-10-29 15:20:55 +00:00
|
|
|
private static native Object makeInstance(VMClass c);
|
2007-07-29 01:29:01 +00:00
|
|
|
}
|