All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.apache.hadoop.hive.ql.io.NonSyncDataInputBuffer Maven / Gradle / Ivy

/**
 * 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.hadoop.hive.ql.io;

import java.io.DataInput;
import java.io.EOFException;
import java.io.FilterInputStream;
import java.io.IOException;
import java.io.PushbackInputStream;
import java.io.UTFDataFormatException;

import org.apache.hadoop.hive.common.io.NonSyncByteArrayInputStream;

/**
 * A thread-not-safe version of Hadoop's DataInputBuffer, which removes all
 * synchronized modifiers.
 */
public class NonSyncDataInputBuffer extends FilterInputStream implements
    DataInput {

  private final NonSyncByteArrayInputStream buffer;

  byte[] buff = new byte[16];

  /** Constructs a new empty buffer. */
  public NonSyncDataInputBuffer() {
    this(new NonSyncByteArrayInputStream());
  }

  private NonSyncDataInputBuffer(NonSyncByteArrayInputStream buffer) {
    super(buffer);
    this.buffer = buffer;
  }

  /** Resets the data that the buffer reads. */
  public void reset(byte[] input, int length) {
    buffer.reset(input, 0, length);
  }

  /** Resets the data that the buffer reads. */
  public void reset(byte[] input, int start, int length) {
    buffer.reset(input, start, length);
  }

  /** Returns the current position in the input. */
  public int getPosition() {
    return buffer.getPosition();
  }

  /** Returns the length of the input. */
  public int getLength() {
    return buffer.getLength();
  }

  /**
   * Reads bytes from the source stream into the byte array buffer.
   * The number of bytes actually read is returned.
   * 
   * @param buffer
   *          the buffer to read bytes into
   * @return the number of bytes actually read or -1 if end of stream.
   * 
   * @throws IOException
   *           If a problem occurs reading from this DataInputStream.
   * 
   */
  @Override
  public final int read(byte[] buffer) throws IOException {
    return in.read(buffer, 0, buffer.length);
  }

  /**
   * Read at most length bytes from this DataInputStream and stores
   * them in byte array buffer starting at offset.
   * Answer the number of bytes actually read or -1 if no bytes were read and
   * end of stream was encountered.
   * 
   * @param buffer
   *          the byte array in which to store the read bytes.
   * @param offset
   *          the offset in buffer to store the read bytes.
   * @param length
   *          the maximum number of bytes to store in buffer.
   * @return the number of bytes actually read or -1 if end of stream.
   * 
   * @throws IOException
   *           If a problem occurs reading from this DataInputStream.
   * 
   */
  @Override
  public final int read(byte[] buffer, int offset, int length)
      throws IOException {
    return in.read(buffer, offset, length);
  }

  /**
   * Reads a boolean from this stream.
   * 
   * @return the next boolean value from the source stream.
   * 
   * @throws IOException
   *           If a problem occurs reading from this DataInputStream.
   * 
   */
  @Override
  public final boolean readBoolean() throws IOException {
    int temp = in.read();
    if (temp < 0) {
      throw new EOFException();
    }
    return temp != 0;
  }

  /**
   * Reads an 8-bit byte value from this stream.
   * 
   * @return the next byte value from the source stream.
   * 
   * @throws IOException
   *           If a problem occurs reading from this DataInputStream.
   * 
   */
  @Override
  public final byte readByte() throws IOException {
    int temp = in.read();
    if (temp < 0) {
      throw new EOFException();
    }
    return (byte) temp;
  }

  /**
   * Reads a 16-bit character value from this stream.
   * 
   * @return the next char value from the source stream.
   * 
   * @throws IOException
   *           If a problem occurs reading from this DataInputStream.
   * 
   */
  private int readToBuff(int count) throws IOException {
    int offset = 0;

    while (offset < count) {
      int bytesRead = in.read(buff, offset, count - offset);
      if (bytesRead == -1) {
        return bytesRead;
      }
      offset += bytesRead;
    }
    return offset;
  }

  @Override
  public final char readChar() throws IOException {
    if (readToBuff(2) < 0) {
      throw new EOFException();
    }
    return (char) (((buff[0] & 0xff) << 8) | (buff[1] & 0xff));

  }

  /**
   * Reads a 64-bit double value from this stream.
   * 
   * @return the next double value from the source stream.
   * 
   * @throws IOException
   *           If a problem occurs reading from this DataInputStream.
   * 
   */
  @Override
  public final double readDouble() throws IOException {
    return Double.longBitsToDouble(readLong());
  }

  /**
   * Reads a 32-bit float value from this stream.
   * 
   * @return the next float value from the source stream.
   * 
   * @throws IOException
   *           If a problem occurs reading from this DataInputStream.
   * 
   */
  @Override
  public final float readFloat() throws IOException {
    return Float.intBitsToFloat(readInt());
  }

  /**
   * Reads bytes from this stream into the byte array buffer. This
   * method will block until buffer.length number of bytes have
   * been read.
   * 
   * @param buffer
   *          to read bytes into
   * 
   * @throws IOException
   *           If a problem occurs reading from this DataInputStream.
   * 
   */
  @Override
  public final void readFully(byte[] buffer) throws IOException {
    readFully(buffer, 0, buffer.length);
  }

  /**
   * Reads bytes from this stream and stores them in the byte array
   * buffer starting at the position offset. This
   * method blocks until count bytes have been read.
   * 
   * @param buffer
   *          the byte array into which the data is read
   * @param offset
   *          the offset the operation start at
   * @param length
   *          the maximum number of bytes to read
   * 
   * @throws IOException
   *           if a problem occurs while reading from this stream
   * @throws EOFException
   *           if reaches the end of the stream before enough bytes have been
   *           read
   */
  @Override
  public final void readFully(byte[] buffer, int offset, int length)
      throws IOException {
    if (length < 0) {
      throw new IndexOutOfBoundsException();
    }
    if (length == 0) {
      return;
    }
    if (in == null || buffer == null) {
      throw new NullPointerException("Null Pointer to underlying input stream");
    }

    if (offset < 0 || offset > buffer.length - length) {
      throw new IndexOutOfBoundsException();
    }
    while (length > 0) {
      int result = in.read(buffer, offset, length);
      if (result < 0) {
        throw new EOFException();
      }
      offset += result;
      length -= result;
    }
  }

  /**
   * Reads a 32-bit integer value from this stream.
   * 
   * @return the next int value from the source stream.
   * 
   * @throws IOException
   *           If a problem occurs reading from this DataInputStream.
   * 
   */
  @Override
  public final int readInt() throws IOException {
    if (readToBuff(4) < 0) {
      throw new EOFException();
    }
    return ((buff[0] & 0xff) << 24) | ((buff[1] & 0xff) << 16)
        | ((buff[2] & 0xff) << 8) | (buff[3] & 0xff);
  }

  /**
   * Answers a String representing the next line of text available
   * in this BufferedReader. A line is represented by 0 or more characters
   * followed by '\n', '\r', "\n\r" or
   * end of stream. The String does not include the newline
   * sequence.
   * 
   * @return the contents of the line or null if no characters were read before
   *         end of stream.
   * 
   * @throws IOException
   *           If the DataInputStream is already closed or some other IO error
   *           occurs.
   * 
   * @deprecated Use BufferedReader
   */
  @Deprecated
  @Override
  public final String readLine() throws IOException {
    StringBuilder line = new StringBuilder(80); // Typical line length
    boolean foundTerminator = false;
    while (true) {
      int nextByte = in.read();
      switch (nextByte) {
      case -1:
        if (line.length() == 0 && !foundTerminator) {
          return null;
        }
        return line.toString();
      case (byte) '\r':
        if (foundTerminator) {
          ((PushbackInputStream) in).unread(nextByte);
          return line.toString();
        }
        foundTerminator = true;
        /* Have to be able to peek ahead one byte */
        if (!(in.getClass() == PushbackInputStream.class)) {
          in = new PushbackInputStream(in);
        }
        break;
      case (byte) '\n':
        return line.toString();
      default:
        if (foundTerminator) {
          ((PushbackInputStream) in).unread(nextByte);
          return line.toString();
        }
        line.append((char) nextByte);
      }
    }
  }

  /**
   * Reads a 64-bit long value from this stream.
   * 
   * @return the next long value from the source stream.
   * 
   * @throws IOException
   *           If a problem occurs reading from this DataInputStream.
   * 
   */
  @Override
  public final long readLong() throws IOException {
    if (readToBuff(8) < 0) {
      throw new EOFException();
    }
    int i1 = ((buff[0] & 0xff) << 24) | ((buff[1] & 0xff) << 16)
        | ((buff[2] & 0xff) << 8) | (buff[3] & 0xff);
    int i2 = ((buff[4] & 0xff) << 24) | ((buff[5] & 0xff) << 16)
        | ((buff[6] & 0xff) << 8) | (buff[7] & 0xff);

    return ((i1 & 0xffffffffL) << 32) | (i2 & 0xffffffffL);
  }

  /**
   * Reads a 16-bit short value from this stream.
   * 
   * @return the next short value from the source stream.
   * 
   * @throws IOException
   *           If a problem occurs reading from this DataInputStream.
   * 
   */
  @Override
  public final short readShort() throws IOException {
    if (readToBuff(2) < 0) {
      throw new EOFException();
    }
    return (short) (((buff[0] & 0xff) << 8) | (buff[1] & 0xff));
  }

  /**
   * Reads an unsigned 8-bit byte value from this stream and
   * returns it as an int.
   * 
   * @return the next unsigned byte value from the source stream.
   * 
   * @throws IOException
   *           If a problem occurs reading from this DataInputStream.
   * 
   */
  @Override
  public final int readUnsignedByte() throws IOException {
    int temp = in.read();
    if (temp < 0) {
      throw new EOFException();
    }
    return temp;
  }

  /**
   * Reads a 16-bit unsigned short value from this stream and
   * returns it as an int.
   * 
   * @return the next unsigned short value from the source stream.
   * 
   * @throws IOException
   *           If a problem occurs reading from this DataInputStream.
   * 
   */
  @Override
  public final int readUnsignedShort() throws IOException {
    if (readToBuff(2) < 0) {
      throw new EOFException();
    }
    return (char) (((buff[0] & 0xff) << 8) | (buff[1] & 0xff));
  }

  /**
   * Reads a UTF format String from this Stream.
   * 
   * @return the next UTF String from the source stream.
   * 
   * @throws IOException
   *           If a problem occurs reading from this DataInputStream.
   * 
   */
  @Override
  public final String readUTF() throws IOException {
    return decodeUTF(readUnsignedShort());
  }

  String decodeUTF(int utfSize) throws IOException {
    return decodeUTF(utfSize, this);
  }

  private static String decodeUTF(int utfSize, DataInput in) throws IOException {
    byte[] buf = new byte[utfSize];
    char[] out = new char[utfSize];
    in.readFully(buf, 0, utfSize);

    return convertUTF8WithBuf(buf, out, 0, utfSize);
  }

  /**
   * Reads a UTF format String from the DataInput Stream in.
   * 
   * @param in
   *          the input stream to read from
   * @return the next UTF String from the source stream.
   * 
   * @throws IOException
   *           If a problem occurs reading from this DataInputStream.
   * 
   */
  public static final String readUTF(DataInput in) throws IOException {
    return decodeUTF(in.readUnsignedShort(), in);
  }

  /**
   * Skips count number of bytes in this stream. Subsequent
   * read()'s will not return these bytes unless
   * reset() is used.
   * 
   * @param count
   *          the number of bytes to skip.
   * @return the number of bytes actually skipped.
   * 
   * @throws IOException
   *           If the stream is already closed or another IOException occurs.
   */
  @Override
  public final int skipBytes(int count) throws IOException {
    int skipped = 0;
    long skip;
    while (skipped < count && (skip = in.skip(count - skipped)) != 0) {
      skipped += skip;
    }
    if (skipped < 0) {
      throw new EOFException();
    }
    return skipped;
  }

  public static String convertUTF8WithBuf(byte[] buf, char[] out, int offset,
      int utfSize) throws UTFDataFormatException {
    int count = 0, s = 0, a;
    while (count < utfSize) {
      if ((out[s] = (char) buf[offset + count++]) < '\u0080') {
        s++;
      } else if (((a = out[s]) & 0xe0) == 0xc0) {
        if (count >= utfSize) {
          throw new UTFDataFormatException();
        }
        int b = buf[count++];
        if ((b & 0xC0) != 0x80) {
          throw new UTFDataFormatException();
        }
        out[s++] = (char) (((a & 0x1F) << 6) | (b & 0x3F));
      } else if ((a & 0xf0) == 0xe0) {
        if (count + 1 >= utfSize) {
          throw new UTFDataFormatException();
        }
        int b = buf[count++];
        int c = buf[count++];
        if (((b & 0xC0) != 0x80) || ((c & 0xC0) != 0x80)) {
          throw new UTFDataFormatException();
        }
        out[s++] = (char) (((a & 0x0F) << 12) | ((b & 0x3F) << 6) | (c & 0x3F));
      } else {
        throw new UTFDataFormatException();
      }
    }
    return new String(out, 0, s);
  }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy