/* * 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. */ /* * Imported by CG 20081213 based on Apache Harmony ("enhanced") revision 721075. */ package java.io; /** * <code>BufferedInputStream</code> is a class which takes an input stream and * <em>buffers</em> the input. In this way, costly interaction with the original * input stream can be minimized by reading buffered amounts of data * infrequently. The drawback is that extra space is required to hold the buffer * and that copying takes place when reading that buffer. * * @see BufferedOutputStream */ public class BufferedInputStream extends FilterInputStream { /** * The buffer containing the current bytes read from the target InputStream. */ protected volatile byte[] buf; /** * The total number of bytes inside the byte array <code>buf</code>. */ protected int count; /** * The current limit, which when passed, invalidates the current mark. */ protected int marklimit; /** * The currently marked position. -1 indicates no mark has been set or the * mark has been invalidated. */ protected int markpos = -1; /** * The current position within the byte array <code>buf</code>. */ protected int pos; /* fib0 and fib1 are fibonacci numbers multiplied by 8192. In general */ /* fib1 holds the size of the current buffer and fib0 holds the previous */ /* number in the fibonacci sequence; however the size of the current */ /* buffer will be less than fib1 if the buffer has been "trimmed". */ private int fib0 = 8192; private int fib1 = 8192; /** * Constructs a new <code>BufferedInputStream</code> on the InputStream * <code>in</code>. The default buffer size (8Kb) is allocated and all reads * can now be filtered through this stream. * * @param in * the InputStream to buffer reads on. */ public BufferedInputStream(InputStream in) { super(in); buf = new byte[8192]; } /** * Constructs a new BufferedInputStream on the InputStream <code>in</code>. * The buffer size is specified by the parameter <code>size</code> and all * reads can now be filtered through this BufferedInputStream. * * @param in * the InputStream to buffer reads on. * @param size * the size of buffer to allocate. */ public BufferedInputStream(InputStream in, int size) { super(in); if (size <= 0) { throw new IllegalArgumentException("size must be > 0"); } buf = new byte[size]; } /** * Answers an int representing the number of bytes that are available before * this BufferedInputStream will block. This method returns the number of * bytes available in the buffer plus those available in the target stream. * * @return the number of bytes available before blocking. * * @throws IOException * If an error occurs in this stream. */ public synchronized int available() throws IOException { InputStream localIn = in; // 'in' could be invalidated by close() if (buf == null || localIn == null) { throw new IOException("stream is closed"); } return count - pos + localIn.available(); } /** * Close this BufferedInputStream. This implementation closes the target * stream and releases any resources associated with it. * * @throws IOException * If an error occurs attempting to close this stream. */ public void close() throws IOException { buf = null; InputStream localIn = in; in = null; if (localIn != null) { localIn.close(); } } private int fillbuf(InputStream localIn, byte[] localBuf) throws IOException { if (markpos == -1 || (pos - markpos >= marklimit)) { /* Mark position not set or exceeded readlimit */ int result = localIn.read(localBuf); if (result > 0) { markpos = -1; pos = 0; count = result == -1 ? 0 : result; } return result; } if (markpos == 0 && marklimit > localBuf.length) { /* Increase buffer size to accommodate the readlimit */ /* [CG 20090226] This calculation is intended to reduce heap */ /* fragmentation by favouring multiples of 8192 bytes. */ int newLength = (count + 8192) & -8192; if (newLength > localBuf.length) { newLength += fib0 - 8192; fib1 = newLength; //System.out.println("old length = " + localBuf.length + ", previous = " + fib0 + ", new = " + newLength); fib0 = localBuf.length < 8192 ? 8192 : localBuf.length; if (newLength > marklimit) { newLength = marklimit; } byte[] newbuf = new byte[newLength]; System.arraycopy(localBuf, 0, newbuf, 0, localBuf.length); // System.out.println(this + " expanded from " + localBuf.length + " to " + newLength); // Reassign buf, which will invalidate any local references // FIXME: what if buf was null? localBuf = buf = newbuf; } } else if (markpos > 0) { System.arraycopy(localBuf, markpos, localBuf, 0, localBuf.length - markpos); // System.out.println(this + " shuffled down " + localBuf.length + " bytes from offset " + markpos + " to offset 0"); } /* Set the new position and mark position */ pos -= markpos; count = markpos = 0; int bytesread = localIn.read(localBuf, pos, localBuf.length - pos); /* [CG 20090226] */ /* WAS: count = bytesread <= 0 ? pos : pos + bytesread; */ if (bytesread <= 0) { // probably end of strem, so trim buffer count = pos; byte[] newbuf = new byte[pos]; System.arraycopy(localBuf, 0, newbuf, 0, pos); // System.out.println(this + " trimmed from " + localBuf.length + " to " + pos); localBuf = buf = newbuf; } else { count = pos + bytesread; } return bytesread; } private void emptybuf(byte[] localBuf) throws IOException { if (markpos == -1 || (pos - markpos >= marklimit)) { if (count - pos < fib0 && fib0 < localBuf.length) { // System.out.println(this + " shrinking from " + localBuf.length + " to " + fib0); byte[] newbuf = new byte[fib0]; System.arraycopy(localBuf, pos, newbuf, 0, count - pos); count -= pos; pos = 0; fib0 = fib1 - fib0; fib1 = fib1 - fib0; localBuf = buf = newbuf; } } } /** * Set a Mark position in this BufferedInputStream. The parameter * <code>readLimit</code> indicates how many bytes can be read before a mark * is invalidated. Sending reset() will reposition the Stream back to the * marked position provided <code>readLimit</code> has not been surpassed. * The underlying buffer may be increased in size to allow * <code>readlimit</code> number of bytes to be supported. * * @param readlimit * the number of bytes to be able to read before invalidating the * mark. */ public synchronized void mark(int readlimit) { marklimit = readlimit; markpos = pos; } /** * Answers a boolean indicating whether or not this BufferedInputStream * supports mark() and reset(). This implementation answers * <code>true</code>. * * @return <code>true</code> for BufferedInputStreams. */ public boolean markSupported() { return true; } /** * Reads a single byte from this BufferedInputStream and returns the result * as an int. The low-order byte is returned or -1 of the end of stream was * encountered. If the underlying buffer does not contain any available * bytes then it is filled and the first byte is returned. * * @return the byte read or -1 if end of stream. * * @throws IOException * If the stream is already closed or another IOException * occurs. */ public synchronized int read() throws IOException { // Use local refs since buf and in may be invalidated by an // unsynchronized close() byte[] localBuf = buf; InputStream localIn = in; if (localBuf == null || localIn == null) { throw new IOException("stream is closed"); } /* Are there buffered bytes available? */ if (pos >= count && fillbuf(localIn, localBuf) == -1) { return -1; /* no, fill buffer */ } // localBuf may have been invalidated by fillbuf if (localBuf != buf) { localBuf = buf; if (localBuf == null) { throw new IOException("stream is closed"); } } /* Did filling the buffer fail with -1 (EOF)? */ if (count - pos > 0) { int result = localBuf[pos++] & 0xFF; emptybuf(localBuf); return result; } return -1; } /** * Reads at most <code>length</code> bytes from this BufferedInputStream and * stores them in byte array <code>buffer</code> starting at offset * <code>offset</code>. Answer the number of bytes actually read or -1 if no * bytes were read and end of stream was encountered. If all the buffered * bytes have been used, a mark has not been set, and the requested number * of bytes is larger than the receiver's buffer size, this implementation * bypasses the buffer and simply places the results directly into * <code>buffer</code>. * * @param buffer * the byte array in which to store the read bytes. * @param offset * the offset in <code>buffer</code> to store the read bytes. * @param length * the maximum number of bytes to store in <code>buffer</code>. * @return the number of bytes actually read or -1 if end of stream. * * @throws IOException * If the stream is already closed or another IOException * occurs. */ public synchronized int read(byte[] buffer, int offset, int length) throws IOException { // Use local ref since buf may be invalidated by an unsynchronized // close() byte[] localBuf = buf; if (localBuf == null) { throw new IOException("stream is closed"); } // avoid int overflow if (offset > buffer.length - length || offset < 0 || length < 0) { throw new IndexOutOfBoundsException(); } if (length == 0) { return 0; } InputStream localIn = in; if (localIn == null) { throw new IOException("stream is closed"); } int required; if (pos < count) { /* There are bytes available in the buffer. */ int copylength = count - pos >= length ? length : count - pos; System.arraycopy(localBuf, pos, buffer, offset, copylength); pos += copylength; if (copylength == length || localIn.available() == 0) { emptybuf(localBuf); return copylength; } offset += copylength; required = length - copylength; } else { required = length; } while (true) { int read; /* * If we're not marked and the required size is greater than the * buffer, simply read the bytes directly bypassing the buffer. */ if (markpos == -1 && required >= localBuf.length) { read = localIn.read(buffer, offset, required); if (read == -1) { return required == length ? -1 : length - required; } } else { if (fillbuf(localIn, localBuf) == -1) { return required == length ? -1 : length - required; } // localBuf may have been invalidated by fillbuf if (localBuf != buf) { localBuf = buf; if (localBuf == null) { throw new IOException("stream is closed"); } } read = count - pos >= required ? required : count - pos; System.arraycopy(localBuf, pos, buffer, offset, read); pos += read; } required -= read; if (required == 0) { emptybuf(localBuf); return length; } if (localIn.available() == 0) { emptybuf(localBuf); return length - required; } offset += read; } } /** * Reset this BufferedInputStream to the last marked location. If the * <code>readlimit</code> has been passed or no <code>mark</code> has been * set, throw IOException. This implementation resets the target stream. * * @throws IOException * If the stream is already closed or another IOException * occurs. */ public synchronized void reset() throws IOException { if (buf == null) { throw new IOException("stream is closed"); } if (-1 == markpos) { throw new IOException("mark has been invalidated"); } pos = markpos; } /** * Skips <code>amount</code> number of bytes in this BufferedInputStream. * Subsequent <code>read()</code>'s will not return these bytes unless * <code>reset()</code> is used. * * @param amount * 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. */ public synchronized long skip(long amount) throws IOException { // Use local refs since buf and in may be invalidated by an // unsynchronized close() byte[] localBuf = buf; InputStream localIn = in; if (localBuf == null || localIn == null) { throw new IOException("stream is closed"); } if (amount < 1) { return 0; } if (count - pos >= amount) { pos += amount; return amount; } long read = count - pos; pos = count; if (markpos != -1) { if (amount <= marklimit) { if (fillbuf(localIn, localBuf) == -1) { return read; } if (count - pos >= amount - read) { pos += amount - read; return amount; } // Couldn't get all the bytes, skip what we read read += (count - pos); pos = count; return read; } markpos = -1; } return read + localIn.skip(amount - read); } }