| 1 | /*
|
|---|
| 2 | * DeltaInputStream
|
|---|
| 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 | import org.tukaani.xz.delta.DeltaDecoder;
|
|---|
| 15 |
|
|---|
| 16 | /**
|
|---|
| 17 | * Decodes raw Delta-filtered data (no XZ headers).
|
|---|
| 18 | * <p>
|
|---|
| 19 | * The delta filter doesn't change the size of the data and thus it
|
|---|
| 20 | * cannot have an end-of-payload marker. It will simply decode until
|
|---|
| 21 | * its input stream indicates end of input.
|
|---|
| 22 | */
|
|---|
| 23 | public class DeltaInputStream extends InputStream {
|
|---|
| 24 | /**
|
|---|
| 25 | * Smallest supported delta calculation distance.
|
|---|
| 26 | */
|
|---|
| 27 | public static final int DISTANCE_MIN = 1;
|
|---|
| 28 |
|
|---|
| 29 | /**
|
|---|
| 30 | * Largest supported delta calculation distance.
|
|---|
| 31 | */
|
|---|
| 32 | public static final int DISTANCE_MAX = 256;
|
|---|
| 33 |
|
|---|
| 34 | private InputStream in;
|
|---|
| 35 | private final DeltaDecoder delta;
|
|---|
| 36 |
|
|---|
| 37 | private IOException exception = null;
|
|---|
| 38 |
|
|---|
| 39 | private final byte[] tempBuf = new byte[1];
|
|---|
| 40 |
|
|---|
| 41 | /**
|
|---|
| 42 | * Creates a new Delta decoder with the given delta calculation distance.
|
|---|
| 43 | *
|
|---|
| 44 | * @param in input stream from which Delta filtered data
|
|---|
| 45 | * is read
|
|---|
| 46 | *
|
|---|
| 47 | * @param distance delta calculation distance, must be in the
|
|---|
| 48 | * range [<code>DISTANCE_MIN</code>,
|
|---|
| 49 | * <code>DISTANCE_MAX</code>]
|
|---|
| 50 | */
|
|---|
| 51 | public DeltaInputStream(InputStream in, int distance) {
|
|---|
| 52 | // Check for null because otherwise null isn't detect
|
|---|
| 53 | // in this constructor.
|
|---|
| 54 | if (in == null)
|
|---|
| 55 | throw new NullPointerException();
|
|---|
| 56 |
|
|---|
| 57 | this.in = in;
|
|---|
| 58 | this.delta = new DeltaDecoder(distance);
|
|---|
| 59 | }
|
|---|
| 60 |
|
|---|
| 61 | /**
|
|---|
| 62 | * Decode the next byte from this input stream.
|
|---|
| 63 | *
|
|---|
| 64 | * @return the next decoded byte, or <code>-1</code> to indicate
|
|---|
| 65 | * the end of input on the input stream <code>in</code>
|
|---|
| 66 | *
|
|---|
| 67 | * @throws IOException may be thrown by <code>in</code>
|
|---|
| 68 | */
|
|---|
| 69 | public int read() throws IOException {
|
|---|
| 70 | return read(tempBuf, 0, 1) == -1 ? -1 : (tempBuf[0] & 0xFF);
|
|---|
| 71 | }
|
|---|
| 72 |
|
|---|
| 73 | /**
|
|---|
| 74 | * Decode into an array of bytes.
|
|---|
| 75 | * <p>
|
|---|
| 76 | * This calls <code>in.read(buf, off, len)</code> and defilters the
|
|---|
| 77 | * returned data.
|
|---|
| 78 | *
|
|---|
| 79 | * @param buf target buffer for decoded data
|
|---|
| 80 | * @param off start offset in <code>buf</code>
|
|---|
| 81 | * @param len maximum number of bytes to read
|
|---|
| 82 | *
|
|---|
| 83 | * @return number of bytes read, or <code>-1</code> to indicate
|
|---|
| 84 | * the end of the input stream <code>in</code>
|
|---|
| 85 | *
|
|---|
| 86 | * @throws XZIOException if the stream has been closed
|
|---|
| 87 | *
|
|---|
| 88 | * @throws IOException may be thrown by underlaying input
|
|---|
| 89 | * stream <code>in</code>
|
|---|
| 90 | */
|
|---|
| 91 | public int read(byte[] buf, int off, int len) throws IOException {
|
|---|
| 92 | if (len == 0)
|
|---|
| 93 | return 0;
|
|---|
| 94 |
|
|---|
| 95 | if (in == null)
|
|---|
| 96 | throw new XZIOException("Stream closed");
|
|---|
| 97 |
|
|---|
| 98 | if (exception != null)
|
|---|
| 99 | throw exception;
|
|---|
| 100 |
|
|---|
| 101 | int size;
|
|---|
| 102 | try {
|
|---|
| 103 | size = in.read(buf, off, len);
|
|---|
| 104 | } catch (IOException e) {
|
|---|
| 105 | exception = e;
|
|---|
| 106 | throw e;
|
|---|
| 107 | }
|
|---|
| 108 |
|
|---|
| 109 | if (size == -1)
|
|---|
| 110 | return -1;
|
|---|
| 111 |
|
|---|
| 112 | delta.decode(buf, off, size);
|
|---|
| 113 | return size;
|
|---|
| 114 | }
|
|---|
| 115 |
|
|---|
| 116 | /**
|
|---|
| 117 | * Calls <code>in.available()</code>.
|
|---|
| 118 | *
|
|---|
| 119 | * @return the value returned by <code>in.available()</code>
|
|---|
| 120 | */
|
|---|
| 121 | public int available() throws IOException {
|
|---|
| 122 | if (in == null)
|
|---|
| 123 | throw new XZIOException("Stream closed");
|
|---|
| 124 |
|
|---|
| 125 | if (exception != null)
|
|---|
| 126 | throw exception;
|
|---|
| 127 |
|
|---|
| 128 | return in.available();
|
|---|
| 129 | }
|
|---|
| 130 |
|
|---|
| 131 | /**
|
|---|
| 132 | * Closes the stream and calls <code>in.close()</code>.
|
|---|
| 133 | * If the stream was already closed, this does nothing.
|
|---|
| 134 | *
|
|---|
| 135 | * @throws IOException if thrown by <code>in.close()</code>
|
|---|
| 136 | */
|
|---|
| 137 | public void close() throws IOException {
|
|---|
| 138 | if (in != null) {
|
|---|
| 139 | try {
|
|---|
| 140 | in.close();
|
|---|
| 141 | } finally {
|
|---|
| 142 | in = null;
|
|---|
| 143 | }
|
|---|
| 144 | }
|
|---|
| 145 | }
|
|---|
| 146 | }
|
|---|