| 1 | /*
|
|---|
| 2 | * SeekableInputStream
|
|---|
| 3 | *
|
|---|
| 4 | * Author: Lasse Collin <lasse.collin@tukaani.org>
|
|---|
| 5 | *
|
|---|
| 6 | * This file has been put into the public domain.
|
|---|
| 7 | * You can do whatever you want with this file.
|
|---|
| 8 | */
|
|---|
| 9 |
|
|---|
| 10 | package org.tukaani.xz;
|
|---|
| 11 |
|
|---|
| 12 | import java.io.InputStream;
|
|---|
| 13 | import java.io.IOException;
|
|---|
| 14 |
|
|---|
| 15 | /**
|
|---|
| 16 | * Input stream with random access support.
|
|---|
| 17 | */
|
|---|
| 18 | public abstract class SeekableInputStream extends InputStream {
|
|---|
| 19 | /**
|
|---|
| 20 | * Seeks <code>n</code> bytes forward in this stream.
|
|---|
| 21 | * <p>
|
|---|
| 22 | * This will not seek past the end of the file. If the current position
|
|---|
| 23 | * is already at or past the end of the file, this doesn't seek at all
|
|---|
| 24 | * and returns <code>0</code>. Otherwise, if skipping <code>n</code> bytes
|
|---|
| 25 | * would cause the position to exceed the stream size, this will do
|
|---|
| 26 | * equivalent of <code>seek(length())</code> and the return value will
|
|---|
| 27 | * be adjusted accordingly.
|
|---|
| 28 | * <p>
|
|---|
| 29 | * If <code>n</code> is negative, the position isn't changed and
|
|---|
| 30 | * the return value is <code>0</code>. It doesn't seek backward
|
|---|
| 31 | * because it would conflict with the specification of
|
|---|
| 32 | * {@link java.io.InputStream#skip(long) InputStream.skip}.
|
|---|
| 33 | *
|
|---|
| 34 | * @return <code>0</code> if <code>n</code> is negative,
|
|---|
| 35 | * less than <code>n</code> if skipping <code>n</code>
|
|---|
| 36 | * bytes would seek past the end of the file,
|
|---|
| 37 | * <code>n</code> otherwise
|
|---|
| 38 | *
|
|---|
| 39 | * @throws IOException might be thrown by {@link #seek(long)}
|
|---|
| 40 | */
|
|---|
| 41 | public long skip(long n) throws IOException {
|
|---|
| 42 | if (n <= 0)
|
|---|
| 43 | return 0;
|
|---|
| 44 |
|
|---|
| 45 | long size = length();
|
|---|
| 46 | long pos = position();
|
|---|
| 47 | if (pos >= size)
|
|---|
| 48 | return 0;
|
|---|
| 49 |
|
|---|
| 50 | if (size - pos < n)
|
|---|
| 51 | n = size - pos;
|
|---|
| 52 |
|
|---|
| 53 | seek(pos + n);
|
|---|
| 54 | return n;
|
|---|
| 55 | }
|
|---|
| 56 |
|
|---|
| 57 | /**
|
|---|
| 58 | * Gets the size of the stream.
|
|---|
| 59 | */
|
|---|
| 60 | public abstract long length() throws IOException;
|
|---|
| 61 |
|
|---|
| 62 | /**
|
|---|
| 63 | * Gets the current position in the stream.
|
|---|
| 64 | */
|
|---|
| 65 | public abstract long position() throws IOException;
|
|---|
| 66 |
|
|---|
| 67 | /**
|
|---|
| 68 | * Seeks to the specified absolute position in the stream.
|
|---|
| 69 | * <p>
|
|---|
| 70 | * Seeking past the end of the file should be supported by the subclasses
|
|---|
| 71 | * unless there is a good reason to do otherwise. If one has seeked
|
|---|
| 72 | * past the end of the stream, <code>read</code> will return
|
|---|
| 73 | * <code>-1</code> to indicate end of stream.
|
|---|
| 74 | *
|
|---|
| 75 | * @param pos new read position in the stream
|
|---|
| 76 | *
|
|---|
| 77 | * @throws IOException if <code>pos</code> is negative or if
|
|---|
| 78 | * a stream-specific I/O error occurs
|
|---|
| 79 | */
|
|---|
| 80 | public abstract void seek(long pos) throws IOException;
|
|---|
| 81 | }
|
|---|