/* * @(#)LittleEndianInputStream.java 1.0.3 02/12/28 * * Copyright 1998, 1999, 2002 Elliotte Rusty Harold * */ package com.macfaq.io; import java.io.*; /** * A little endian input stream reads two's complement, * little endian integers, floating point numbers, and characters * and returns them as Java primitive types. * The standard java.io.DataInputStream class * which this class imitates reads big-endian quantities. * * @author Elliotte Rusty Harold * @version 1.0.3, 28 December 2002 * @see com.macfaq.io.LittleEndianOutputStream * @see java.io.DataInputStream */ public class LittleEndianInputStream extends FilterInputStream { /** * Creates a new little endian input stream and chains it to the * input stream specified by the in argument. * * @param in the underlying input stream. * @see java.io.FilterInputStream#out */ public LittleEndianInputStream(InputStream in) { super(in); } /** * Reads a boolean from the underlying input stream by * reading a single byte. If the byte is zero, false is returned. * If the byte is positive, true is returned. * * @return b the boolean value read. * @exception EOFException if the end of the underlying input stream * has been reached * @exception IOException if the underlying stream throws an IOException. */ public boolean readBoolean() throws IOException { int bool = in.read(); if (bool == -1) throw new EOFException(); return (bool != 0); } /** * Reads a signed byte from the underlying input stream * with value between -128 and 127 * * @return the byte value read. * @exception EOFException if the end of the underlying input stream * has been reached * @exception IOException if the underlying stream throws an IOException. */ public byte readByte(int b) throws IOException { int temp = in.read(); if (temp == -1) throw new EOFException(); return (byte) temp; } /** * Reads an unsigned byte from the underlying * input stream with value between 0 and 255 * * @return the byte value read. * @exception EOFException if the end of the underlying input * stream has been reached * @exception IOException if the underlying stream throws an IOException. */ public int readUnsignedByte() throws IOException { int temp = in.read(); if (temp == -1) throw new EOFException(); return temp; } /** * Reads a two byte signed short from the underlying * input stream in little endian order, low byte first. * * @return the short read. * @exception EOFException if the end of the underlying input stream * has been reached * @exception IOException if the underlying stream throws an IOException. */ public short readShort() throws IOException { int byte1 = in.read(); int byte2 = in.read(); // only need to test last byte read // if byte1 is -1 so is byte2 if (byte2 == -1) throw new EOFException(); return (short) (((byte2 << 24) >>> 16) + (byte1 << 24) >>> 24); } /** * Reads a two byte unsigned short from the underlying * input stream in little endian order, low byte first. * * @return the int value of the unsigned short read. * @exception EOFException if the end of the underlying input stream * has been reached * @exception IOException if the underlying stream throws an IOException. */ public int readUnsignedShort() throws IOException { int byte1 = in.read(); int byte2 = in.read(); if (byte2 == -1) throw new EOFException(); return ((byte2 << 24) >> 16) + ((byte1 << 24) >> 24); } /** * Reads a two byte Unicode char from the underlying * input stream in little endian order, low byte first. * * @return the int value of the unsigned short read. * @exception EOFException if the end of the underlying input stream * has been reached * @exception IOException if the underlying stream throws an IOException. */ public char readChar() throws IOException { int byte1 = in.read(); int byte2 = in.read(); if (byte2 == -1) throw new EOFException(); return (char) (((byte2 << 24) >>> 16) + ((byte1 << 24) >>> 24)); } /** * Reads a four byte signed int from the underlying * input stream in little endian order, low byte first. * * @return the int read. * @exception EOFException if the end of the underlying input stream * has been reached * @exception IOException if the underlying stream throws an IOException. */ public int readInt() throws IOException { int byte1, byte2, byte3, byte4; synchronized (this) { byte1 = in.read(); byte2 = in.read(); byte3 = in.read(); byte4 = in.read(); } if (byte4 == -1) { throw new EOFException(); } return (byte4 << 24) + ((byte3 << 24) >>> 8) + ((byte2 << 24) >>> 16) + ((byte1 << 24) >>> 24); } /** * Reads an eight byte signed int from the underlying * input stream in little endian order, low byte first. * * @return the int read. * @exception EOFException if the end of the underlying input stream * has been reached * @exception IOException if the underlying stream throws an IOException. */ public long readLong() throws IOException { long byte1 = in.read(); long byte2 = in.read(); long byte3 = in.read(); long byte4 = in.read(); long byte5 = in.read(); long byte6 = in.read(); long byte7 = in.read(); long byte8 = in.read(); if (byte8 == -1) { throw new EOFException(); } return (byte8 << 56) + ((byte7 << 56) >>> 8) + ((byte6 << 56) >>> 16) + ((byte5 << 56) >>> 24) + ((byte4 << 56) >>> 32) + ((byte3 << 56) >>> 40) + ((byte2 << 56) >>> 48) + ((byte1 << 56) >>> 56); } /** * Reads a string of no more than 65,535 characters * from the underlying input stream using UTF-8 * encoding. This method first reads a two byte short * in big endian order as required by the * UTF-8 specification. This gives the number of bytes in * the UTF-8 encoded version of the string. * Next this many bytes are read and decoded as UTF-8 * encoded characters. * * @return the decoded string * @exception UTFDataFormatException if the string cannot be decoded * @exception IOException if the underlying stream throws an IOException. */ public String readUTF() throws IOException { int byte1 = in.read(); int byte2 = in.read(); if (byte2 == -1) throw new EOFException(); int numbytes = (byte1 << 8) + byte2; char result[] = new char[numbytes]; int numread = 0; int numchars = 0; while (numread < numbytes) { int c1 = readUnsignedByte(); int c2, c3; // The first four bits of c1 determine how many bytes are in this char int test = c1 >> 4; if (test < 8) { // one byte numread++; result[numchars++] = (char) c1; } else if (test == 12 || test == 13) { // two bytes numread += 2; if (numread > numbytes) throw new UTFDataFormatException(); c2 = readUnsignedByte(); if ((c2 & 0xC0) != 0x80) throw new UTFDataFormatException(); result[numchars++] = (char) (((c1 & 0x1F) << 6) | (c2 & 0x3F)); } else if (test == 14) { // three bytes numread += 3; if (numread > numbytes) throw new UTFDataFormatException(); c2 = readUnsignedByte(); c3 = readUnsignedByte(); if (((c2 & 0xC0) != 0x80) || ((c3 & 0xC0) != 0x80)) { throw new UTFDataFormatException(); } result[numchars++] = (char) (((c1 & 0x0F) << 12) | ((c2 & 0x3F) << 6) | (c3 & 0x3F)); } else { // malformed throw new UTFDataFormatException(); } } // end while return new String(result, 0, numchars); } /** * * @return the next eight bytes of this input stream, interpreted as a * little endian double. * @exception EOFException if end of stream occurs before eight bytes * have been read. * @exception IOException if an I/O error occurs. */ public final double readDouble() throws IOException { return Double.longBitsToDouble(this.readLong()); } /** * * @return the next four bytes of this input stream, interpreted as a * little endian int. * @exception EOFException if end of stream occurs before four bytes * have been read. * @exception IOException if an I/O error occurs. */ public final float readFloat() throws IOException { return Float.intBitsToFloat(this.readInt()); } /** * Skip exactly n bytes of input in the underlying * input stream. This method blocks until all the bytes are skipped, * the end of the stream is detected, or an exception is thrown. * * @param n the number of bytes to skip. * @return the number of bytes skipped, generally n * @exception EOFException if this input stream reaches the end before * skipping all the bytes. * @exception IOException if the underlying stream throws an IOException. */ public final int skipBytes(int n) throws IOException { for (int i = 0; i < n; i += (int) skip(n - i)); return n; } }