io.airlift.compress.lzo.LzopHadoopInputStream Maven / Gradle / Ivy
/*
* Licensed 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.
*/
package io.airlift.compress.lzo;
import io.airlift.compress.hadoop.HadoopInputStream;
import java.io.ByteArrayInputStream;
import java.io.EOFException;
import java.io.IOException;
import java.io.InputStream;
import java.util.Arrays;
import java.util.zip.Adler32;
import java.util.zip.CRC32;
import java.util.zip.Checksum;
import static io.airlift.compress.lzo.LzoConstants.LZOP_MAGIC;
import static io.airlift.compress.lzo.LzoConstants.LZO_1X_VARIANT;
import static io.airlift.compress.lzo.LzoConstants.SIZE_OF_LONG;
import static java.lang.String.format;
import static java.util.Objects.requireNonNull;
class LzopHadoopInputStream
extends HadoopInputStream
{
private static final int LZO_VERSION_MAX = 0x20A0;
private static final int LZOP_FILE_VERSION_MIN = 0x0940;
private static final int LZOP_FORMAT_VERSION_MAX = 0x1010;
private static final int LZOP_FLAG_ADLER32_DECOMPRESSED = 0x0000_0001;
private static final int LZOP_FLAG_ADLER32_COMPRESSED = 0x0000_0002;
private static final int LZOP_FLAG_CRC32_DECOMPRESSED = 0x0000_0100;
private static final int LZOP_FLAG_CRC32_COMPRESSED = 0x0000_0200;
private static final int LZOP_FLAG_CRC32_HEADER = 0x0000_1000;
private static final int LZOP_FLAG_IO_MASK = 0x0000_000c;
private static final int LZOP_FLAG_OPERATING_SYSTEM_MASK = 0xff00_0000;
private static final int LZOP_FLAG_CHARACTER_SET_MASK = 0x00f0_0000;
private final LzoDecompressor decompressor = new LzoDecompressor();
private final InputStream in;
private final byte[] uncompressedChunk;
private int uncompressedLength;
private int uncompressedOffset;
private boolean finished;
private byte[] compressed = new byte[0];
private final boolean adler32Decompressed;
private final boolean adler32Compressed;
private final boolean crc32Decompressed;
private final boolean crc32Compressed;
public LzopHadoopInputStream(InputStream in, int maxUncompressedLength)
throws IOException
{
this.in = requireNonNull(in, "in is null");
// over allocate buffer which makes decompression easier
uncompressedChunk = new byte[maxUncompressedLength + SIZE_OF_LONG];
byte[] magic = new byte[LZOP_MAGIC.length];
readInput(magic, 0, magic.length);
if (!Arrays.equals(magic, LZOP_MAGIC)) {
throw new IOException("Not an LZOP file");
}
byte[] header = new byte[25];
readInput(header, 0, header.length);
ByteArrayInputStream headerStream = new ByteArrayInputStream(header);
// lzop version: ignored
int lzopFileVersion = readBigEndianShort(headerStream);
if (lzopFileVersion < LZOP_FILE_VERSION_MIN) {
throw new IOException(format("Unsupported LZOP file version 0x%08X", lzopFileVersion));
}
// lzo version
int lzoVersion = readBigEndianShort(headerStream);
if (lzoVersion > LZO_VERSION_MAX) {
throw new IOException(format("Unsupported LZO version 0x%08X", lzoVersion));
}
// lzop version of the format
int lzopFormatVersion = readBigEndianShort(headerStream);
if (lzopFormatVersion > LZOP_FORMAT_VERSION_MAX) {
throw new IOException(format("Unsupported LZOP format version 0x%08X", lzopFormatVersion));
}
// variant: must be LZO 1X
int variant = headerStream.read();
if (variant != LZO_1X_VARIANT) {
throw new IOException(format("Unsupported LZO variant %s", variant));
}
// level: ignored
headerStream.read();
// flags
int flags = readBigEndianInt(headerStream);
// ignore flags about the compression environment
flags &= ~LZOP_FLAG_IO_MASK;
flags &= ~LZOP_FLAG_OPERATING_SYSTEM_MASK;
flags &= ~LZOP_FLAG_CHARACTER_SET_MASK;
// checksum flags
adler32Decompressed = (flags & LZOP_FLAG_ADLER32_DECOMPRESSED) != 0;
adler32Compressed = (flags & LZOP_FLAG_ADLER32_COMPRESSED) != 0;
crc32Decompressed = (flags & LZOP_FLAG_CRC32_DECOMPRESSED) != 0;
crc32Compressed = (flags & LZOP_FLAG_CRC32_COMPRESSED) != 0;
boolean crc32Header = (flags & LZOP_FLAG_CRC32_HEADER) != 0;
flags &= ~LZOP_FLAG_ADLER32_DECOMPRESSED;
flags &= ~LZOP_FLAG_ADLER32_COMPRESSED;
flags &= ~LZOP_FLAG_CRC32_DECOMPRESSED;
flags &= ~LZOP_FLAG_CRC32_COMPRESSED;
flags &= ~LZOP_FLAG_CRC32_HEADER;
// no other flags are supported
if (flags != 0) {
throw new IOException(format("Unsupported LZO flags 0x%08X", flags));
}
// output file mode: ignored
readBigEndianInt(headerStream);
// output file modified time: ignored
readBigEndianInt(headerStream);
// output file time zone offset: ignored
readBigEndianInt(headerStream);
// output file name: ignored
int fileNameLength = headerStream.read();
byte[] fileName = new byte[fileNameLength];
readInput(fileName, 0, fileName.length);
// verify header checksum
int headerChecksumValue = readBigEndianInt(in);
Checksum headerChecksum = crc32Header ? new CRC32() : new Adler32();
headerChecksum.update(header, 0, header.length);
headerChecksum.update(fileName, 0, fileName.length);
if (headerChecksumValue != (int) headerChecksum.getValue()) {
throw new IOException("Invalid header checksum");
}
}
@Override
public int read()
throws IOException
{
if (finished) {
return -1;
}
while (uncompressedOffset >= uncompressedLength) {
int compressedLength = bufferCompressedData();
if (finished) {
return -1;
}
decompress(compressedLength, uncompressedChunk, 0, uncompressedChunk.length);
}
return uncompressedChunk[uncompressedOffset++] & 0xFF;
}
@Override
public int read(byte[] output, int offset, int length)
throws IOException
{
if (finished) {
return -1;
}
while (uncompressedOffset >= uncompressedLength) {
int compressedLength = bufferCompressedData();
if (finished) {
return -1;
}
// favor writing directly to user buffer to avoid extra copy
if (length >= uncompressedLength) {
decompress(compressedLength, output, offset, length);
uncompressedOffset = uncompressedLength;
return uncompressedLength;
}
decompress(compressedLength, uncompressedChunk, 0, uncompressedChunk.length);
}
int size = Math.min(length, uncompressedLength - uncompressedOffset);
System.arraycopy(uncompressedChunk, uncompressedOffset, output, offset, size);
uncompressedOffset += size;
return size;
}
@Override
public void resetState()
{
uncompressedLength = 0;
uncompressedOffset = 0;
finished = false;
}
@Override
public void close()
throws IOException
{
in.close();
}
private int bufferCompressedData()
throws IOException
{
uncompressedOffset = 0;
uncompressedLength = readBigEndianInt(in);
if (uncompressedLength == -1) {
// LZOP file MUST end with uncompressedLength == 0
throw new EOFException("encountered EOF while reading block data");
}
if (uncompressedLength == 0) {
finished = true;
return -1;
}
int compressedLength = readBigEndianInt(in);
if (compressedLength == -1) {
throw new EOFException("encountered EOF while reading block data");
}
skipChecksums(compressedLength < uncompressedLength);
return compressedLength;
}
private void skipChecksums(boolean compressed)
throws IOException
{
if (adler32Decompressed) {
readBigEndianInt(in);
}
if (crc32Decompressed) {
readBigEndianInt(in);
}
if (compressed && adler32Compressed) {
readBigEndianInt(in);
}
if (compressed && crc32Compressed) {
readBigEndianInt(in);
}
}
private void decompress(int compressedLength, byte[] output, int outputOffset, int outputLength)
throws IOException
{
if (uncompressedLength == compressedLength) {
readInput(output, outputOffset, compressedLength);
}
else {
if (compressed.length < compressedLength) {
// over allocate buffer which makes decompression easier
compressed = new byte[compressedLength + SIZE_OF_LONG];
}
readInput(compressed, 0, compressedLength);
int actualUncompressedLength = decompressor.decompress(compressed, 0, compressedLength, output, outputOffset, outputLength);
if (actualUncompressedLength != uncompressedLength) {
throw new IOException("Decompressor did not decompress the entire block");
}
}
}
private void readInput(byte[] buffer, int offset, int length)
throws IOException
{
while (length > 0) {
int size = in.read(buffer, offset, length);
if (size == -1) {
throw new EOFException("encountered EOF while reading block data");
}
offset += size;
length -= size;
}
}
private static int readBigEndianShort(InputStream in)
throws IOException
{
int b1 = in.read();
if (b1 < 0) {
return -1;
}
int b2 = in.read();
// If second byte is negative, the stream it truncated
if ((b2) < 0) {
throw new IOException("Stream is truncated");
}
return (b1 << 8) + (b2);
}
private static int readBigEndianInt(InputStream in)
throws IOException
{
int b1 = in.read();
if (b1 < 0) {
return -1;
}
int b2 = in.read();
int b3 = in.read();
int b4 = in.read();
// If any of the other bits are negative, the stream it truncated
if ((b2 | b3 | b4) < 0) {
throw new IOException("Stream is truncated");
}
return ((b1 << 24) + (b2 << 16) + (b3 << 8) + (b4));
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy