blob: 45e8b57f19e46ce92e958249f6cca570c6c9b0c5 [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You 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.
*/
package org.apache.ignite.internal.binary;
import java.io.Externalizable;
import java.io.IOException;
import java.io.ObjectInput;
import java.io.ObjectOutput;
import java.math.BigDecimal;
import java.math.BigInteger;
import java.nio.ByteBuffer;
import java.sql.Time;
import java.sql.Timestamp;
import java.util.Date;
import java.util.UUID;
import org.apache.ignite.IgniteCheckedException;
import org.apache.ignite.binary.BinaryObject;
import org.apache.ignite.binary.BinaryObjectException;
import org.apache.ignite.binary.BinaryType;
import org.apache.ignite.internal.GridDirectTransient;
import org.apache.ignite.internal.IgniteCodeGeneratingFail;
import org.apache.ignite.internal.binary.streams.BinaryHeapInputStream;
import org.apache.ignite.internal.processors.cache.CacheObject;
import org.apache.ignite.internal.processors.cache.CacheObjectAdapter;
import org.apache.ignite.internal.processors.cache.CacheObjectContext;
import org.apache.ignite.internal.processors.cache.CacheObjectValueContext;
import org.apache.ignite.internal.processors.cache.KeyCacheObject;
import org.apache.ignite.internal.processors.cache.binary.CacheObjectBinaryProcessorImpl;
import org.apache.ignite.internal.util.typedef.internal.U;
import org.apache.ignite.plugin.extensions.communication.MessageReader;
import org.apache.ignite.plugin.extensions.communication.MessageWriter;
import org.jetbrains.annotations.Nullable;
import static java.nio.charset.StandardCharsets.UTF_8;
/**
* Binary object implementation.
*/
@IgniteCodeGeneratingFail // Fields arr and start should not be generated by MessageCodeGenerator.
public final class BinaryObjectImpl extends BinaryObjectExImpl implements Externalizable, KeyCacheObject {
/** */
private static final long serialVersionUID = 0L;
/** */
@GridDirectTransient
private BinaryContext ctx;
/** */
private byte[] arr;
/** */
private int start;
/** */
@GridDirectTransient
private Object obj;
/** */
@GridDirectTransient
private boolean detachAllowed;
/** */
private int part = -1;
/**
* For {@link Externalizable}.
*/
public BinaryObjectImpl() {
// No-op.
}
/**
* @param ctx Context.
* @param arr Array.
* @param start Start.
*/
public BinaryObjectImpl(BinaryContext ctx, byte[] arr, int start) {
assert ctx != null;
assert arr != null;
this.ctx = ctx;
this.arr = arr;
this.start = start;
}
/** {@inheritDoc} */
@Override public KeyCacheObject copy(int part) {
if (this.part == part)
return this;
BinaryObjectImpl cp = new BinaryObjectImpl(ctx, arr, start);
cp.part = part;
return cp;
}
/** {@inheritDoc} */
@Override public int partition() {
return part;
}
/** {@inheritDoc} */
@Override public void partition(int part) {
this.part = part;
}
/** {@inheritDoc} */
@Override public byte cacheObjectType() {
return TYPE_BINARY;
}
/** {@inheritDoc} */
@Override public boolean isPlatformType() {
return false;
}
/** {@inheritDoc} */
@Override public boolean internal() {
return false;
}
/** {@inheritDoc} */
@Nullable @Override public <T> T value(CacheObjectValueContext ctx, boolean cpy) {
Object obj0 = obj;
if (obj0 == null || (cpy && needCopy(ctx)))
obj0 = deserializeValue(ctx);
return (T)obj0;
}
/** {@inheritDoc} */
@Override public byte[] valueBytes(CacheObjectValueContext ctx) throws IgniteCheckedException {
if (detached())
return array();
int len = length();
byte[] arr0 = new byte[len];
U.arrayCopy(arr, start, arr0, 0, len);
return arr0;
}
/** {@inheritDoc} */
@Override public boolean putValue(ByteBuffer buf) throws IgniteCheckedException {
return putValue(buf, 0, CacheObjectAdapter.objectPutSize(length()));
}
/** {@inheritDoc} */
@Override public int putValue(long addr) throws IgniteCheckedException {
return CacheObjectAdapter.putValue(addr, cacheObjectType(), arr, start);
}
/** {@inheritDoc} */
@Override public boolean putValue(final ByteBuffer buf, int off, int len) throws IgniteCheckedException {
return CacheObjectAdapter.putValue(cacheObjectType(), buf, off, len, arr, start);
}
/** {@inheritDoc} */
@Override public int valueBytesLength(CacheObjectContext ctx) throws IgniteCheckedException {
return CacheObjectAdapter.objectPutSize(length());
}
/** {@inheritDoc} */
@Override public CacheObject prepareForCache(CacheObjectContext ctx) {
if (detached())
return this;
return (BinaryObjectImpl)detach();
}
/** {@inheritDoc} */
@Override public void finishUnmarshal(CacheObjectValueContext ctx, ClassLoader ldr) throws IgniteCheckedException {
CacheObjectBinaryProcessorImpl binaryProc = (CacheObjectBinaryProcessorImpl)ctx.kernalContext().cacheObjects();
this.ctx = binaryProc.binaryContext();
binaryProc.waitMetadataWriteIfNeeded(typeId());
}
/** {@inheritDoc} */
@Override public void prepareMarshal(CacheObjectValueContext ctx) throws IgniteCheckedException {
// No-op.
}
/** {@inheritDoc} */
@Override public int length() {
return BinaryPrimitives.readInt(arr, start + GridBinaryMarshaller.TOTAL_LEN_POS);
}
/**
* @return Detached binary object.
*/
public BinaryObjectImpl detach() {
if (!detachAllowed || detached())
return this;
int len = length();
byte[] arr0 = new byte[len];
U.arrayCopy(arr, start, arr0, 0, len);
return new BinaryObjectImpl(ctx, arr0, 0);
}
/**
* @return Detached or not.
*/
public boolean detached() {
return start == 0 && length() == arr.length;
}
/**
* @param detachAllowed Detach allowed flag.
*/
public void detachAllowed(boolean detachAllowed) {
this.detachAllowed = detachAllowed;
}
/** {@inheritDoc} */
@Override public BinaryContext context() {
return ctx;
}
/**
* @param ctx Context.
*/
public void context(BinaryContext ctx) {
this.ctx = ctx;
}
/** {@inheritDoc} */
@Override public byte[] array() {
return arr;
}
/** {@inheritDoc} */
@Override public int start() {
return start;
}
/** {@inheritDoc} */
@Override public long offheapAddress() {
return 0;
}
/** {@inheritDoc} */
@Override public boolean hasArray() {
return true;
}
/** {@inheritDoc} */
@Override public boolean isFlagSet(short flag) {
short flags = BinaryPrimitives.readShort(arr, start + GridBinaryMarshaller.FLAGS_POS);
return BinaryUtils.isFlagSet(flags, flag);
}
/** {@inheritDoc} */
@Override public int typeId() {
int off = start + GridBinaryMarshaller.TYPE_ID_POS;
int typeId = BinaryPrimitives.readInt(arr, off);
if (typeId == GridBinaryMarshaller.UNREGISTERED_TYPE_ID) {
off = start + GridBinaryMarshaller.DFLT_HDR_LEN;
assert arr[off] == GridBinaryMarshaller.STRING : arr[off];
int len = BinaryPrimitives.readInt(arr, ++off);
String clsName = new String(arr, off + 4, len, UTF_8);
typeId = ctx.typeId(clsName);
}
return typeId;
}
/** {@inheritDoc} */
@Nullable @Override public BinaryType type() throws BinaryObjectException {
return BinaryUtils.typeProxy(ctx, this);
}
/** {@inheritDoc} */
@Nullable @Override public BinaryType rawType() throws BinaryObjectException {
return BinaryUtils.type(ctx, this);
}
/** {@inheritDoc} */
@Nullable @Override public <F> F field(String fieldName) throws BinaryObjectException {
return (F) reader(null, false).unmarshalField(fieldName);
}
/** {@inheritDoc} */
@Nullable @Override public <F> F field(int fieldId) throws BinaryObjectException {
return (F) reader(null, false).unmarshalField(fieldId);
}
/** {@inheritDoc} */
@Override public BinarySerializedFieldComparator createFieldComparator() {
int schemaOff = BinaryPrimitives.readInt(arr, start + GridBinaryMarshaller.SCHEMA_OR_RAW_OFF_POS);
short flags = BinaryPrimitives.readShort(arr, start + GridBinaryMarshaller.FLAGS_POS);
int fieldIdLen = BinaryUtils.isCompactFooter(flags) ? 0 : BinaryUtils.FIELD_ID_LEN;
int fieldOffLen = BinaryUtils.fieldOffsetLength(flags);
int orderBase = start + schemaOff + fieldIdLen;
int orderMultiplier = fieldIdLen + fieldOffLen;
return new BinarySerializedFieldComparator(this, arr, 0L, start, orderBase, orderMultiplier, fieldOffLen);
}
/** {@inheritDoc} */
@Override public int dataStartOffset() {
int typeId = BinaryPrimitives.readInt(arr, start + GridBinaryMarshaller.TYPE_ID_POS);
if (typeId == GridBinaryMarshaller.UNREGISTERED_TYPE_ID) {
int len = BinaryPrimitives.readInt(arr, start + GridBinaryMarshaller.DFLT_HDR_LEN + 1);
return start + GridBinaryMarshaller.DFLT_HDR_LEN + len + 5;
} else
return start + GridBinaryMarshaller.DFLT_HDR_LEN;
}
/** {@inheritDoc} */
@Override public int footerStartOffset() {
short flags = BinaryPrimitives.readShort(arr, start + GridBinaryMarshaller.FLAGS_POS);
if (!BinaryUtils.hasSchema(flags))
return start + length();
return start + BinaryPrimitives.readInt(arr, start + GridBinaryMarshaller.SCHEMA_OR_RAW_OFF_POS);
}
/** {@inheritDoc} */
@Nullable @Override public <F> F fieldByOrder(int order) {
if (order == BinarySchema.ORDER_NOT_FOUND)
return null;
Object val;
// Calculate field position.
int schemaOff = BinaryPrimitives.readInt(arr, start + GridBinaryMarshaller.SCHEMA_OR_RAW_OFF_POS);
short flags = BinaryPrimitives.readShort(arr, start + GridBinaryMarshaller.FLAGS_POS);
int fieldIdLen = BinaryUtils.isCompactFooter(flags) ? 0 : BinaryUtils.FIELD_ID_LEN;
int fieldOffLen = BinaryUtils.fieldOffsetLength(flags);
int fieldOffsetPos = start + schemaOff + order * (fieldIdLen + fieldOffLen) + fieldIdLen;
int fieldPos;
if (fieldOffLen == BinaryUtils.OFFSET_1)
fieldPos = start + ((int)BinaryPrimitives.readByte(arr, fieldOffsetPos) & 0xFF);
else if (fieldOffLen == BinaryUtils.OFFSET_2)
fieldPos = start + ((int)BinaryPrimitives.readShort(arr, fieldOffsetPos) & 0xFFFF);
else
fieldPos = start + BinaryPrimitives.readInt(arr, fieldOffsetPos);
// Read header and try performing fast lookup for well-known types (the most common types go first).
byte hdr = BinaryPrimitives.readByte(arr, fieldPos);
switch (hdr) {
case GridBinaryMarshaller.INT:
val = BinaryPrimitives.readInt(arr, fieldPos + 1);
break;
case GridBinaryMarshaller.LONG:
val = BinaryPrimitives.readLong(arr, fieldPos + 1);
break;
case GridBinaryMarshaller.BOOLEAN:
val = BinaryPrimitives.readBoolean(arr, fieldPos + 1);
break;
case GridBinaryMarshaller.SHORT:
val = BinaryPrimitives.readShort(arr, fieldPos + 1);
break;
case GridBinaryMarshaller.BYTE:
val = BinaryPrimitives.readByte(arr, fieldPos + 1);
break;
case GridBinaryMarshaller.CHAR:
val = BinaryPrimitives.readChar(arr, fieldPos + 1);
break;
case GridBinaryMarshaller.FLOAT:
val = BinaryPrimitives.readFloat(arr, fieldPos + 1);
break;
case GridBinaryMarshaller.DOUBLE:
val = BinaryPrimitives.readDouble(arr, fieldPos + 1);
break;
case GridBinaryMarshaller.STRING: {
int dataLen = BinaryPrimitives.readInt(arr, fieldPos + 1);
val = new String(arr, fieldPos + 5, dataLen, UTF_8);
break;
}
case GridBinaryMarshaller.DATE: {
long time = BinaryPrimitives.readLong(arr, fieldPos + 1);
val = new Date(time);
break;
}
case GridBinaryMarshaller.TIMESTAMP: {
long time = BinaryPrimitives.readLong(arr, fieldPos + 1);
int nanos = BinaryPrimitives.readInt(arr, fieldPos + 1 + 8);
Timestamp ts = new Timestamp(time);
ts.setNanos(ts.getNanos() + nanos);
val = ts;
break;
}
case GridBinaryMarshaller.TIME: {
long time = BinaryPrimitives.readLong(arr, fieldPos + 1);
val = new Time(time);
break;
}
case GridBinaryMarshaller.UUID: {
long most = BinaryPrimitives.readLong(arr, fieldPos + 1);
long least = BinaryPrimitives.readLong(arr, fieldPos + 1 + 8);
val = new UUID(most, least);
break;
}
case GridBinaryMarshaller.DECIMAL: {
int scale = BinaryPrimitives.readInt(arr, fieldPos + 1);
int dataLen = BinaryPrimitives.readInt(arr, fieldPos + 5);
byte[] data = BinaryPrimitives.readByteArray(arr, fieldPos + 9, dataLen);
boolean negative = data[0] < 0;
if (negative)
data[0] &= 0x7F;
BigInteger intVal = new BigInteger(data);
if (negative)
intVal = intVal.negate();
val = new BigDecimal(intVal, scale);
break;
}
case GridBinaryMarshaller.NULL:
val = null;
break;
default:
val = BinaryUtils.unmarshal(BinaryHeapInputStream.create(arr, fieldPos), ctx, null);
break;
}
return (F)val;
}
/** {@inheritDoc} */
@Override public boolean writeFieldByOrder(int order, ByteBuffer buf) {
// Calculate field position.
int schemaOffset = BinaryPrimitives.readInt(arr, start + GridBinaryMarshaller.SCHEMA_OR_RAW_OFF_POS);
short flags = BinaryPrimitives.readShort(arr, start + GridBinaryMarshaller.FLAGS_POS);
int fieldIdLen = BinaryUtils.isCompactFooter(flags) ? 0 : BinaryUtils.FIELD_ID_LEN;
int fieldOffsetLen = BinaryUtils.fieldOffsetLength(flags);
int fieldOffsetPos = start + schemaOffset + order * (fieldIdLen + fieldOffsetLen) + fieldIdLen;
int fieldPos;
if (fieldOffsetLen == BinaryUtils.OFFSET_1)
fieldPos = start + ((int)BinaryPrimitives.readByte(arr, fieldOffsetPos) & 0xFF);
else if (fieldOffsetLen == BinaryUtils.OFFSET_2)
fieldPos = start + ((int)BinaryPrimitives.readShort(arr, fieldOffsetPos) & 0xFFFF);
else
fieldPos = start + BinaryPrimitives.readInt(arr, fieldOffsetPos);
// Read header and try performing fast lookup for well-known types (the most common types go first).
byte hdr = BinaryPrimitives.readByte(arr, fieldPos);
int totalLen;
switch (hdr) {
case GridBinaryMarshaller.NULL:
totalLen = 1;
break;
case GridBinaryMarshaller.INT:
case GridBinaryMarshaller.FLOAT:
totalLen = 5;
break;
case GridBinaryMarshaller.LONG:
case GridBinaryMarshaller.DOUBLE:
case GridBinaryMarshaller.DATE:
case GridBinaryMarshaller.TIME:
totalLen = 9;
break;
case GridBinaryMarshaller.BOOLEAN:
totalLen = 2;
break;
case GridBinaryMarshaller.SHORT:
totalLen = 3;
break;
case GridBinaryMarshaller.BYTE:
totalLen = 2;
break;
case GridBinaryMarshaller.CHAR:
totalLen = 3;
break;
case GridBinaryMarshaller.STRING: {
int dataLen = BinaryPrimitives.readInt(arr, fieldPos + 1);
totalLen = dataLen + 5;
break;
}
case GridBinaryMarshaller.TIMESTAMP:
totalLen = 13;
break;
case GridBinaryMarshaller.UUID:
totalLen = 17;
break;
case GridBinaryMarshaller.DECIMAL: {
int dataLen = BinaryPrimitives.readInt(arr, fieldPos + 5);
totalLen = dataLen + 9;
break;
}
case GridBinaryMarshaller.OBJ:
totalLen = BinaryPrimitives.readInt(arr, fieldPos + GridBinaryMarshaller.TOTAL_LEN_POS);
break;
case GridBinaryMarshaller.OPTM_MARSH:
totalLen = BinaryPrimitives.readInt(arr, fieldPos + 1);
break;
default:
throw new UnsupportedOperationException("Failed to write field of the given type " +
"(field type is not supported): " + hdr);
}
if (buf.remaining() < totalLen)
return false;
buf.put(arr, fieldPos, totalLen);
return true;
}
/** {@inheritDoc} */
@Nullable @Override protected <F> F field(BinaryReaderHandles rCtx, String fieldName) {
return (F)reader(rCtx, false).unmarshalField(fieldName);
}
/** {@inheritDoc} */
@Override public boolean hasField(String fieldName) {
return reader(null, false).findFieldByName(fieldName);
}
/** {@inheritDoc} */
@Nullable @Override public <T> T deserialize() throws BinaryObjectException {
Object obj0 = obj;
if (obj0 == null)
obj0 = deserializeValue(null);
return (T)obj0;
}
/** {@inheritDoc} */
@Override public BinaryObject clone() throws CloneNotSupportedException {
return super.clone();
}
/** {@inheritDoc} */
@Override public int hashCode() {
return BinaryPrimitives.readInt(arr, start + GridBinaryMarshaller.HASH_CODE_POS);
}
/** {@inheritDoc} */
@Override public boolean hasSchema() {
short flags = BinaryPrimitives.readShort(arr, start + GridBinaryMarshaller.FLAGS_POS);
return BinaryUtils.hasSchema(flags);
}
/** {@inheritDoc} */
@Override public int schemaId() {
return BinaryPrimitives.readInt(arr, start + GridBinaryMarshaller.SCHEMA_ID_POS);
}
/** {@inheritDoc} */
@Override public BinarySchema createSchema() {
return reader(null, false).getOrCreateSchema();
}
/** {@inheritDoc} */
@Override public void onAckReceived() {
// No-op.
}
/** {@inheritDoc} */
@Override public void writeExternal(ObjectOutput out) throws IOException {
if (detachAllowed) {
int len = length();
out.writeInt(len);
out.write(arr, start, len);
out.writeInt(0);
}
else {
out.writeInt(arr.length);
out.write(arr);
out.writeInt(start);
}
}
/** {@inheritDoc} */
@Override public void readExternal(ObjectInput in) throws IOException, ClassNotFoundException {
ctx = GridBinaryMarshaller.threadLocalContext();
arr = new byte[in.readInt()];
in.readFully(arr);
start = in.readInt();
}
/** {@inheritDoc} */
@Override public boolean writeTo(ByteBuffer buf, MessageWriter writer) {
writer.setBuffer(buf);
if (!writer.isHeaderWritten()) {
if (!writer.writeHeader(directType(), fieldsCount()))
return false;
writer.onHeaderWritten();
}
switch (writer.state()) {
case 0:
if (!writer.writeByteArray("arr",
arr,
detachAllowed ? start : 0,
detachAllowed ? length() : arr.length))
return false;
writer.incrementState();
case 1:
if (!writer.writeInt("part", part))
return false;
writer.incrementState();
case 2:
if (!writer.writeInt("start", detachAllowed ? 0 : start))
return false;
writer.incrementState();
}
return true;
}
/** {@inheritDoc} */
@Override public boolean readFrom(ByteBuffer buf, MessageReader reader) {
reader.setBuffer(buf);
if (!reader.beforeMessageRead())
return false;
switch (reader.state()) {
case 0:
arr = reader.readByteArray("arr");
if (!reader.isLastRead())
return false;
reader.incrementState();
case 1:
part = reader.readInt("part");
if (!reader.isLastRead())
return false;
reader.incrementState();
case 2:
start = reader.readInt("start");
if (!reader.isLastRead())
return false;
reader.incrementState();
}
return reader.afterMessageRead(BinaryObjectImpl.class);
}
/** {@inheritDoc} */
@Override public short directType() {
return 113;
}
/** {@inheritDoc} */
@Override public byte fieldsCount() {
return 3;
}
/**
* Runs value deserialization regardless of whether obj already has the deserialized value.
* Will set obj if descriptor is configured to keep deserialized values.
* @param coCtx CacheObjectContext.
* @return Object.
*/
private Object deserializeValue(@Nullable CacheObjectValueContext coCtx) {
BinaryReaderExImpl reader = reader(null, coCtx != null ?
coCtx.kernalContext().config().getClassLoader() : ctx.configuration().getClassLoader(), true);
Object obj0 = reader.deserialize();
BinaryClassDescriptor desc = reader.descriptor();
assert desc != null;
if (coCtx != null && coCtx.storeValue())
obj = obj0;
return obj0;
}
/**
* @param ctx Context.
* @return {@code True} need to copy value returned to user.
*/
private boolean needCopy(CacheObjectValueContext ctx) {
return ctx.copyOnGet() && obj != null && !ctx.kernalContext().cacheObjects().immutable(obj);
}
/**
* Create new reader for this object.
*
* @param rCtx Reader context.
* @param ldr Class loader.
* @param forUnmarshal {@code True} if reader is need to unmarshal object.
* @return Reader.
*/
private BinaryReaderExImpl reader(@Nullable BinaryReaderHandles rCtx, @Nullable ClassLoader ldr,
boolean forUnmarshal) {
if (ldr == null)
ldr = ctx.configuration().getClassLoader();
return new BinaryReaderExImpl(ctx,
BinaryHeapInputStream.create(arr, start),
ldr,
rCtx,
forUnmarshal);
}
/**
* Create new reader for this object.
*
* @param rCtx Reader context.
* @param forUnmarshal {@code True} if reader is need to unmarshal object.
* @return Reader.
*/
private BinaryReaderExImpl reader(@Nullable BinaryReaderHandles rCtx, boolean forUnmarshal) {
return reader(rCtx, null, forUnmarshal);
}
/**
* Compare two objects for DML operation.
*
* @param first First.
* @param second Second.
* @return Comparison result.
*/
@SuppressWarnings("unchecked")
public static int compareForDml(Object first, Object second) {
boolean firstBinary = first instanceof BinaryObjectImpl;
boolean secondBinary = second instanceof BinaryObjectImpl;
if (firstBinary) {
if (secondBinary)
return compareForDml0((BinaryObjectImpl)first, (BinaryObjectImpl)second);
else
return 1; // Go to the right part.
}
else {
if (secondBinary)
return -1; // Go to the left part.
else
return ((Comparable)first).compareTo(second);
}
}
/**
* Internal DML comparison routine.
*
* @param first First item.
* @param second Second item.
* @return Comparison result.
*/
private static int compareForDml0(BinaryObjectImpl first, BinaryObjectImpl second) {
int res = Integer.compare(first.typeId(), second.typeId());
if (res == 0) {
res = Integer.compare(first.hashCode(), second.hashCode());
if (res == 0) {
// Pessimistic case: need to perform binary comparison.
int firstDataStart = first.dataStartOffset();
int secondDataStart = second.dataStartOffset();
int firstLen = first.footerStartOffset() - firstDataStart;
int secondLen = second.footerStartOffset() - secondDataStart;
res = Integer.compare(firstLen, secondLen);
if (res == 0) {
for (int i = 0; i < firstLen; i++) {
byte firstByte = first.arr[firstDataStart + i];
byte secondByte = second.arr[secondDataStart + i];
res = Byte.compare(firstByte, secondByte);
if (res != 0)
break;
}
}
}
}
return res;
}
/** {@inheritDoc} */
@Override public String toString() {
if (arr == null || ctx == null)
return "BinaryObjectImpl [arr= " + (arr != null) + ", ctx=" + (ctx != null) + ", start=" + start + "]";
return super.toString();
}
}