Package com.esotericsoftware.kryo.io

Source Code of com.esotericsoftware.kryo.io.UnsafeMemoryInput

package com.esotericsoftware.kryo.io;

import java.io.IOException;
import java.io.InputStream;
import java.nio.ByteBuffer;

import sun.nio.ch.DirectBuffer;

import com.esotericsoftware.kryo.KryoException;
import static com.esotericsoftware.kryo.util.UnsafeUtil.*;

/** An optimized InputStream that reads data directly from the off-heap memory. Utility methods are provided for efficiently
* reading primitive types, arrays of primitive types and strings. It uses @link{java.misc.Unsafe} to achieve a very good
* performance.
*
* <p>
* Important notes:<br/>
* <li>Bulk operations, e.g. on arrays of primitive types, are always using native byte order.</li>
* <li>Fixed-size int, long, short, float and double elements are always read using native byte order.</li>
* <li>Best performance is achieved if no variable length encoding for integers is used.</li>
* <li>Serialized representation used as input for this class should always be produced using @link{UnsafeMemoryOutput}</li>
* </p>
* @author Roman Levenstein <romixlev@gmail.com> */
public final class UnsafeMemoryInput extends ByteBufferInput {
  /** Start address of the memory buffer The memory buffer should be non-movable, which normally means that is is allocated
   * off-heap */
  private long bufaddress;

  {
    varIntsEnabled = false;

  }

  /** Creates an uninitialized Input. {@link #setBuffer(byte[], int, int)} must be called before the Input is used. */
  public UnsafeMemoryInput () {
  }

  /** Creates a new Input for reading from a byte array.
   * @param bufferSize The initial and maximum size of the buffer. An exception is thrown if this size is exceeded. */
  public UnsafeMemoryInput (int bufferSize) {
    super(bufferSize);
    updateBufferAddress();
  }

  /** Creates a new Input for reading from a byte array.
   * @see #setBuffer(byte[]) */
  public UnsafeMemoryInput (byte[] buffer) {
    super(buffer);
    updateBufferAddress();
  }

  public UnsafeMemoryInput (ByteBuffer buffer) {
    super(buffer, 0, buffer.position());
    updateBufferAddress();
  }

  public UnsafeMemoryInput (long address, int maxBufferSize) {
    super(address, maxBufferSize);
    updateBufferAddress();
  }

  /** Creates a new Input for reading from an InputStream. A buffer size of 4096 is used. */
  public UnsafeMemoryInput (InputStream inputStream) {
    super(inputStream);
    updateBufferAddress();
  }

  /** Creates a new Input for reading from an InputStream. */
  public UnsafeMemoryInput (InputStream inputStream, int bufferSize) {
    super(inputStream, bufferSize);
    updateBufferAddress();
  }

  public void setBuffer (ByteBuffer buffer, int offset, int count) {
    super.setBuffer(buffer, offset, count);
    updateBufferAddress();
  }

  private void updateBufferAddress () {
    bufaddress = ((DirectBuffer)super.niobuffer).address();
  }

  /** Reads a 4 byte int. */
  public int readInt () throws KryoException {
    require(4);
    int result = unsafe().getInt(bufaddress + position);
    position += 4;
    return result;
  }

  /** Reads a 4 byte float. */
  public float readFloat () throws KryoException {
    require(4);
    float result = unsafe().getFloat(bufaddress + position);
    position += 4;
    return result;
  }

  /** Reads a 2 byte short. */
  public short readShort () throws KryoException {
    require(2);
    short result = unsafe().getShort(bufaddress + position);
    position += 2;
    return result;
  }

  /** Reads an 8 byte long. */
  public long readLong () throws KryoException {
    require(8);
    long result = unsafe().getLong(bufaddress + position);
    position += 8;
    return result;
  }

  /** Reads a 1 byte boolean. */
  public boolean readBoolean () throws KryoException {
    super.niobuffer.position(position);
    return super.readBoolean();
  }

  /** Reads a single byte. */
  public byte readByte () throws KryoException {
    super.niobuffer.position(position);
    return super.readByte();
  }

  /** Reads a 2 byte char. */
  public char readChar () throws KryoException {
    super.niobuffer.position(position);
    return super.readChar();
  }

  /** Reads an 8 byte double. */
  public double readDouble () throws KryoException {
    require(8);
    double result = unsafe().getDouble(bufaddress + position);
    position += 8;
    return result;
  }

  public int readInt (boolean optimizePositive) throws KryoException {
    if (!varIntsEnabled)
      return readInt();
    else
      return super.readInt(optimizePositive);
  }

  public long readLong (boolean optimizePositive) throws KryoException {
    if (!varIntsEnabled)
      return readLong();
    else
      return super.readLong(optimizePositive);
  }

  // Methods implementing bulk operations on arrays of primitive types

  /** {@inheritDoc} */
  final public int[] readInts (int length, boolean optimizePositive) throws KryoException {
    if (!varIntsEnabled) {
      int bytesToCopy = length << 2;
      int[] array = new int[length];
      readBytes(array, intArrayBaseOffset, 0, bytesToCopy);
      return array;
    } else
      return super.readInts(length, optimizePositive);
  }

  /** {@inheritDoc} */
  final public long[] readLongs (int length, boolean optimizePositive) throws KryoException {
    if (!varIntsEnabled) {
      int bytesToCopy = length << 3;
      long[] array = new long[length];
      readBytes(array, longArrayBaseOffset, 0, bytesToCopy);
      return array;
    } else
      return super.readLongs(length, optimizePositive);
  }

  /** {@inheritDoc} */
  final public float[] readFloats (int length) throws KryoException {
    int bytesToCopy = length << 2;
    float[] array = new float[length];
    readBytes(array, floatArrayBaseOffset, 0, bytesToCopy);
    return array;
  }

  /** {@inheritDoc} */
  final public short[] readShorts (int length) throws KryoException {
    int bytesToCopy = length << 1;
    short[] array = new short[length];
    readBytes(array, shortArrayBaseOffset, 0, bytesToCopy);
    return array;
  }

  /** {@inheritDoc} */
  final public char[] readChars (int length) throws KryoException {
    int bytesToCopy = length << 1;
    char[] array = new char[length];
    readBytes(array, charArrayBaseOffset, 0, bytesToCopy);
    return array;
  }

  /** {@inheritDoc} */
  final public double[] readDoubles (int length) throws KryoException {
    int bytesToCopy = length << 2;
    double[] array = new double[length];
    readBytes(array, doubleArrayBaseOffset, 0, bytesToCopy);
    return array;
  }

  final public void readBytes (Object dstObj, long offset, long count) throws KryoException {
    /* Unsafe supports efficient bulk reading into arrays of primitives only because of JVM limitations due to GC */
    if (dstObj.getClass().isArray())
      readBytes(dstObj, 0, offset, (int)count);
    else {
      throw new KryoException("Only bulk reads of arrays is supported");
    }
  }

  final private void readBytes (Object dstObj, long dstArrayTypeOffset, long offset, int count) throws KryoException {
    int copyCount = Math.min(limit - position, count);
    while (true) {
      unsafe().copyMemory(null, bufaddress + position, dstObj, dstArrayTypeOffset + offset, copyCount);
      position += copyCount;
      count -= copyCount;
      if (count == 0) break;
      offset += copyCount;
      copyCount = Math.min(count, capacity);
      require(copyCount);
    }
  }
}
TOP

Related Classes of com.esotericsoftware.kryo.io.UnsafeMemoryInput

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.