package android.content.pm;
import java.io.FilterInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.util.Arrays;
/**
* A class that limits the amount of data that is read from an InputStream. When
* the specified length is reached, the stream returns an EOF even if the
* underlying stream still has more data.
*
* @hide
*/
public class LimitedLengthInputStream extends FilterInputStream {
/**
* The end of the stream where we don't want to allow more data to be read.
*/
private final long mEnd;
/**
* Current offset in the stream.
*/
private long mOffset;
/**
* @param in underlying stream to wrap
* @param offset offset into stream where data starts
* @param length length of data at offset
* @throws IOException if an error occurred with the underlying stream
*/
public LimitedLengthInputStream(InputStream in, long offset, long length) throws IOException {
super(in);
if (in == null) {
throw new IOException("in == null");
}
if (offset < 0) {
throw new IOException("offset < 0");
}
if (length < 0) {
throw new IOException("length < 0");
}
if (length > Long.MAX_VALUE - offset) {
throw new IOException("offset + length > Long.MAX_VALUE");
}
mEnd = offset + length;
skip(offset);
mOffset = offset;
}
@Override
public synchronized int read() throws IOException {
if (mOffset >= mEnd) {
return -1;
}
mOffset++;
return super.read();
}
@Override
public int read(byte[] buffer, int offset, int byteCount) throws IOException {
if (mOffset >= mEnd) {
return -1;
}
final int arrayLength = buffer.length;
Arrays.checkOffsetAndCount(arrayLength, offset, byteCount);
if (mOffset > Long.MAX_VALUE - byteCount) {
throw new IOException("offset out of bounds: " + mOffset + " + " + byteCount);
}
if (mOffset + byteCount > mEnd) {
byteCount = (int) (mEnd - mOffset);
}
final int numRead = super.read(buffer, offset, byteCount);
mOffset += numRead;
return numRead;
}
@Override
public int read(byte[] buffer) throws IOException {
return read(buffer, 0, buffer.length);
}
}