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