All Downloads are FREE. Search and download functionalities are using the official Maven repository.

src.com.ibm.as400.access.DataStreamCompression Maven / Gradle / Ivy

There is a newer version: 20.0.8
Show newest version
///////////////////////////////////////////////////////////////////////////////
//                                                                             
// JTOpen (IBM Toolbox for Java - OSS version)                                 
//                                                                             
// Filename: DataStreamCompression.java
//                                                                             
// The source code contained herein is licensed under the IBM Public License   
// Version 1.0, which has been approved by the Open Source Initiative.         
// Copyright (C) 1997-2001 International Business Machines Corporation and     
// others. All rights reserved.                                                
//                                                                             
///////////////////////////////////////////////////////////////////////////////

package com.ibm.as400.access;



/**
The DataStreamCompression class provides support for RLE (Run Length Encoding)
compression and decompression.  When data is compressed using RLE, contiguous
repeating bytes in the datastream are replaced with an RLE record.  When data
is decompressed, the RLE record is expanded out into repeating bytes again.

The format of the output RLE record is as follows:

  • RLE repeater record:
    • 1 byte escape
    • 2 byte repeater
    • 2 byte repeat count
  • RLE escape record:
    • 1 byte escape
    • 1 byte escape
During compression, each escape byte will be replaced with an RLE escape record (two escape bytes). If a byte is not an escape byte, then two bytes are compared to the next two bytes to determine if the byte pair is repeated. Compression is done and an RLE repeater record built if the byte pairs are repeated more than five times (totaling 10 or more bytes repeated). For additional information on RLE, see the LIPI documentation for the Remote Command/Program Server. **/ class DataStreamCompression { private static final String copyright = "Copyright (C) 1997-2001 International Business Machines Corporation and others."; final static byte DEFAULT_ESCAPE = (byte) 0x1B; final static int ESCAPE_SIZE = 1; final static int REPEATER_SIZE = 2; final static int COUNT_SIZE = 2; final static int REPEATER_RECORD_SIZE = ESCAPE_SIZE + REPEATER_SIZE + COUNT_SIZE; final static int ESCAPE_RECORD_SIZE = ESCAPE_SIZE + ESCAPE_SIZE; /** No need to instantiate an object of this class since the class only contains two static methods. Emphasizing this fact with a private ctor. **/ private DataStreamCompression () { } /** Compress data in the source byte array and write the compressed data to the returned byte array. @param source The source (decompressed) bytes. @param sourceOffset The offset in the source bytes at which to start compressing. @param length The length of the bytes to compress. @param escape The escape character. Use DEFAULT_ESCAPE. @return The compressed bytes or null if the data was not compressed. If the compressed data length is larger than the uncompressed length, null is returned. **/ static byte[] compressRLE (byte[] source, int sourceOffset, int length, byte escape) { // Validate the input byte array. if (source == null) { throw new NullPointerException("source"); } // Validate the source offset value. if (sourceOffset >= source.length) { throw new ExtendedIllegalArgumentException("sourceOffset", ExtendedIllegalArgumentException.PARAMETER_VALUE_NOT_VALID); } // Validate the length value. if (length <= 0) { throw new ExtendedIllegalArgumentException("length", ExtendedIllegalArgumentException.PARAMETER_VALUE_NOT_VALID); } // Set flags to indicate which type of tracing, if any, should be done. boolean traceDiagnostic = Trace.isTraceOn() && Trace.isTraceDiagnosticOn(); byte[] destination = new byte[length]; int compressedCount = compressRLEInternal(source, sourceOffset, length, destination, 0, escape); // @A1A if (compressedCount >= 0) { // @A1A byte[] returnBytes = new byte[compressedCount]; // @A1A System.arraycopy(destination, 0, returnBytes, 0, compressedCount); // @A1A return returnBytes; // @A1A } // @A1A else // @A1A return null; // @A1A } // @A1A /** Compress data in the source byte array and write the compressed data to the destination byte array. @param source The source (decompressed) bytes. @param sourceOffset The offset in the source bytes at which to start compressing. @param length The length of the bytes to compress. @param destination The destination (compressed) bytes. @param destinationOffset The offset in the destination bytes at which to assign compressed bytes. @param escape The escape character. Use DEFAULT_ESCAPE. @return The number of compressed bytes, or -1 if the data was not compressed. If the compressed data length is larger than the uncompressed length, null is returned. **/ static int compressRLE (byte[] source, int sourceOffset, int length, byte[] destination, int destinationOffset, byte escape) { // Validate the input byte array. if (source == null) { throw new NullPointerException("source"); } // Validate the source offset value. if (sourceOffset >= source.length) { throw new ExtendedIllegalArgumentException("sourceOffset", ExtendedIllegalArgumentException.PARAMETER_VALUE_NOT_VALID); } // Validate the length value. if (length <= 0) { throw new ExtendedIllegalArgumentException("length", ExtendedIllegalArgumentException.PARAMETER_VALUE_NOT_VALID); } // Validate the input byte array. if (destination == null) { throw new NullPointerException("destinationOffset"); } // Validate the source offset value. if (destinationOffset >= destination.length) { throw new ExtendedIllegalArgumentException("destinationOffset", ExtendedIllegalArgumentException.PARAMETER_VALUE_NOT_VALID); } return compressRLEInternal(source, sourceOffset, length, destination, destinationOffset, escape); } // @A1A - Moved from compressRLE(byte[], int, int, byte) /** Compress data in the source byte array and write the compressed data to the destination byte array. @param source The source (decompressed) bytes. @param sourceOffset The offset in the source bytes at which to start compressing. @param length The length of the bytes to compress. @param destination The destination (compressed) bytes. @param destinationOffset The offset in the destination bytes at which to assign compressed bytes. @param escape The escape character. Use DEFAULT_ESCAPE. @return The number of compressed bytes, or -1 if the data was not compressed. If the compressed data length is larger than the uncompressed length, null is returned. **/ private static int compressRLEInternal (byte[] source, int sourceOffset, int length, byte[] destination, int destinationOffset, byte escape) { // Set flags to indicate which type of tracing, if any, should be done. boolean traceDiagnostic = Trace.isTraceOn() && Trace.isTraceDiagnosticOn(); int returnCount = -1; int i = sourceOffset; // Index into source. int j = destinationOffset; // Index into destination. boolean overflow = false; // destination array overflow indicator int sourceLength = sourceOffset + length; int destinationLength = destination.length; if (traceDiagnostic) { Trace.log(Trace.DIAGNOSTIC, "compressRLE() sourceLength: " + sourceLength); } while (i < sourceLength && overflow != true) { // Have an escape byte if (source[i] == escape) { // Bytes fit in destination array if (j + ESCAPE_RECORD_SIZE <= destinationLength) { // Write out an escape record destination[j++] = escape; destination[j++] = escape; ++i; } // Destination array overflow else { if (traceDiagnostic) Trace.log(Trace.DIAGNOSTIC, "Overflow when writing out escape record starting at dest " + j + " ..."); overflow = true; } } // Have a single, non-escape byte and end of source data. else if ((i+1) >= sourceLength) { // Bytes fit in destination array if (j < destinationLength) { // Write out the last byte. destination[j++] = source[i++]; } // Destination array overflow else { if (traceDiagnostic) Trace.log(Trace.DIAGNOSTIC, "Overflow when writing out last byte before EOD to dest " + j + " ..."); overflow = true; } } // Have a single byte, followed by an escape character. else if (source[i+1] == escape) { // Bytes fit in destination array if ((j + 1 + ESCAPE_RECORD_SIZE) <= destinationLength) { // Write out the single byte and then the escape record. destination[j++] = source[i++]; // byte before the escape byte destination[j++] = escape; destination[j++] = escape; ++i; // byte after the escape byte } // Destination array overflow else { if (traceDiagnostic) Trace.log(Trace.DIAGNOSTIC, "Overflow when writing out single byte and escape record starting at dest " + j + " ..."); overflow = true; } } // Have at least two non-escape bytes that could be a repeater. // Compare two bytes with next two bytes. else { int saveOffset = i; //@P0D int repeater = BinaryConverter.byteArrayToUnsignedShort(source, i); // @A2C // int repeater = ((source[i] & 0xFF) << 8) + (source[i+1] & 0xFF); //@P0A // @M7C byte repeaterByte1 = source[i]; byte repeaterByte2 = source[i+1]; int count = 1; // @A2C i += 2; // Calculate the number of times these two bytes are repeated. // while (((i+1) < sourceLength) && repeater == (((source[i] & 0xFF) << 8) + (source[i+1] & 0xFF)) && count < 65535) //@P0C @B1A while (((i+1) < sourceLength) && repeaterByte1 == source[i] && repeaterByte2 == source[i+1] && count < 65535) //@P0C @B1A { // @A2C count++; i += 2; } // Calculate the length of the repeating characters. int repeatLength = count * REPEATER_SIZE; // Determine if we have enough repeating bytes to merit an RLE record. if (repeatLength >= REPEATER_RECORD_SIZE * 2) { // Enough repeating data. Build RLE record. if (j + REPEATER_RECORD_SIZE <= destinationLength) { // Bytes fit in destination array; write out the repeated bytes. destination[j] = escape; //@P0C //@P0D BinaryConverter.unsignedShortToByteArray(repeater, destination, j); // @A2C // destination[++j] = (byte)(repeater >>> 8); //@P0A // destination[++j] = (byte) repeater; //@P0A destination[++j] = repeaterByte1; destination[++j] = repeaterByte2; //@P0D BinaryConverter.unsignedShortToByteArray(count, destination, j+2); // @A2C //@P0D j += 4; destination[++j] = (byte)(count >>> 8); //@P0A destination[++j] = (byte) count; //@P0A ++j; //@P0A } // Destination array overflow else { if (traceDiagnostic) Trace.log(Trace.DIAGNOSTIC, "Overflow when writing out RLE repeater record starting at dest " + j + " ..."); overflow = true; } } else { // Not enough repeating data. Just copy the data to destination array. i = saveOffset; // Bytes fit in destination array if ((j + (repeatLength - 1)) < destinationLength) { // Write out the repeated bytes. for (int n=0; n < repeatLength; n++) { destination[j++] = source[i++]; } } // Destination array overflow else { if (traceDiagnostic) Trace.log(Trace.DIAGNOSTIC, "Overflow when writing out non-repeating bytes to dest " + j + " ..."); overflow = true; } } } } returnCount = j - destinationOffset; // @A1A if (!overflow && (returnCount < length)) { // @A1C if (traceDiagnostic) Trace.log(Trace.DIAGNOSTIC, "compressRLE() length of compressed bytes returned: " + j); } else { returnCount = -1; if (traceDiagnostic) Trace.log(Trace.DIAGNOSTIC, "compressRLE() returning null (compressed size >= decompressed size)"); } return returnCount; } /** Decompress data in the source byte array and write the decompressed data to the returned byte array. @param source The source (compressed) bytes. @param sourceOffset The offset in the source bytes at which to start decompressing. @param length The length of the bytes to decompress. @param decompressedLength The length of the bytes in their decompressed state. If this length is provided, the byte array to be returned will be created using this length. If this length is not known, input 0. When set to 0, this value will be calculated and the returned byte array will be created with the calculated length. Performance is improved when the correct decompressed length is provided as input. @param escape The escape character. Use DEFAULT_ESCAPE. @return The decompressed bytes. **/ static byte[] decompressRLE (byte[] source, int sourceOffset, int length, int decompressedLength, byte escape) { // Validate the input byte array. if (source == null) { throw new NullPointerException("source"); } // Validate the source offset value. if (sourceOffset >= source.length) { throw new ExtendedIllegalArgumentException("sourceOffset", ExtendedIllegalArgumentException.PARAMETER_VALUE_NOT_VALID); } // Validate the length value. if (length <= 0) { throw new ExtendedIllegalArgumentException("length", ExtendedIllegalArgumentException.PARAMETER_VALUE_NOT_VALID); } // Validate the decompressed length value. if (decompressedLength < 0) { throw new ExtendedIllegalArgumentException("decompressedLength", ExtendedIllegalArgumentException.PARAMETER_VALUE_NOT_VALID); } // Create a temporary buffer for the decompressed data. byte[] destination; if (decompressedLength == 0) { destination = new byte[2048]; } else { destination = new byte[decompressedLength]; } return decompressRLEInternal(source, sourceOffset, length, destination, 0, escape, true, true); // @A1A } // @A1A /** Decompress data in the source byte array and write the decompressed data to the destination byte array. @param source The source (compressed) bytes. @param sourceOffset The offset in the source bytes at which to start decompressing. @param length The length of the bytes to decompress. @param destination The destination (decompressed) bytes. @param destinationOffset The offset in the destination bytes at which to assign decompressed bytes. @param escape The escape character. Use DEFAULT_ESCAPE. @param emptyDestination If set to true, assumes that the destination contains only zeros and the decompression can be optimized **/ static void decompressRLE (byte[] source, int sourceOffset, int length, byte[] destination, int destinationOffset, byte escape, boolean emptyDestination) { // Validate the input byte array. if (source == null) { throw new NullPointerException("source"); } // Validate the source offset value. if (sourceOffset >= source.length) { throw new ExtendedIllegalArgumentException("sourceOffset", ExtendedIllegalArgumentException.PARAMETER_VALUE_NOT_VALID); } // Validate the length value. if (length <= 0) { throw new ExtendedIllegalArgumentException("length", ExtendedIllegalArgumentException.PARAMETER_VALUE_NOT_VALID); } // Validate the input byte array. if (destination == null) { throw new NullPointerException("destination"); } // Validate the source offset value. if (destinationOffset >= destination.length) { throw new ExtendedIllegalArgumentException("destinationOffset", ExtendedIllegalArgumentException.PARAMETER_VALUE_NOT_VALID); } decompressRLEInternal(source, sourceOffset, length, destination, destinationOffset, escape, false, emptyDestination); } // @A1A - Moved from decompressRLE(byte[], int, int, int, byte) /** Decompress data in the source byte array and write the decompressed data to the destination byte array. @param source The source (compressed) bytes. @param sourceOffset The offset in the source bytes at which to start decompressing. @param length The length of the bytes to decompress. @param destination The destination (decompressed) bytes. @param destinationOffset The offset in the destination bytes at which to assign decompressed bytes. @param escape The escape character. Use DEFAULT_ESCAPE. @param reallocate true to reallocate the destination array, if needed, false otherwise. @parm emptyDestination true if the destination is initially empty This is is the case, then expansion of 0 bytes can be optimized. @return The decompressed bytes. **/ private static byte[] decompressRLEInternal (byte[] source, int sourceOffset, int length, byte[] destination, int destinationOffset, byte escape, boolean reallocate, boolean emptyDestination) { // Set flags to indicate which type of tracing, if any, should be done. boolean traceDiagnostic = Trace.isTraceOn() && Trace.isTraceDiagnosticOn(); boolean traceError = Trace.isTraceOn() && Trace.isTraceErrorOn(); int i = sourceOffset; // Index into source. int j = destinationOffset; // Index into destination. int saveI = -1; // keep current position of the source index int saveJ = -1; // keep current position of the destination index boolean overflow = false; // destination array overflow indicator int bytesNeeded = 0; // number of additional bytes needed int sourceLength = sourceOffset + length; int destinationLength = destination.length; if (traceDiagnostic) { Trace.log(Trace.DIAGNOSTIC, "decompressRLE() sourceLength: " + sourceLength); Trace.log(Trace.DIAGNOSTIC, "decompressRLE() destinationLength: " + destinationLength); } while (i < sourceLength) { // Have an escape byte if (source[i] == escape) { // Not end of source if ((i + ESCAPE_SIZE) < sourceLength) { // Second byte is escape; have an RLE escape record if (source[i + ESCAPE_SIZE] == escape) { // Byte fits in destination array if (j < destinationLength) { // Add escape byte to destination array destination[j] = escape; } // Destination array overflow else { if (traceDiagnostic) { Trace.log(Trace.DIAGNOSTIC, "Overflow while decompressing RLE escape record starting at " + j + " ..."); } // save each index only when overflow is encountered // the first time; otherwise, just tally up the number of // additional bytes needed if (!overflow) { saveI = i; saveJ = j; } overflow = true; bytesNeeded += ESCAPE_SIZE; } i += ESCAPE_RECORD_SIZE; j += ESCAPE_SIZE; } // Should have an RLE repeater record; // have an escape byte followed by a non-escape byte else { // Not end of source; have a complete RLE repeater record if ((i + REPEATER_SIZE + COUNT_SIZE) < sourceLength) { // Get repeater //@P0D int repeater = BinaryConverter.byteArrayToUnsignedShort(source, i + ESCAPE_SIZE); // @A2C // int repeater = ((source[i+ESCAPE_SIZE] & 0xFF) << 8) + (source[i+ESCAPE_SIZE+1] & 0xFF); //@P0A //@M7C byte repeatByte1 = source[i+ESCAPE_SIZE]; byte repeatByte2 = source[i+ESCAPE_SIZE+1]; // Get repeat count //@P0D int count = BinaryConverter.byteArrayToUnsignedShort(source, i + ESCAPE_SIZE + REPEATER_SIZE); // @A2C int count = ((source[i+ESCAPE_SIZE+REPEATER_SIZE] & 0xFF) << 8) + (source[i+ESCAPE_SIZE+REPEATER_SIZE+1] & 0xFF); //@P0A // Bytes fit in destination array if ((j + (count * REPEATER_SIZE)) <= destinationLength) { // Check to see if we can skip copying the zero'd bytes if the // destination is still empty. @M7A if (repeatByte1 == 0 && repeatByte2 == 0 && emptyDestination ) { j += count * REPEATER_SIZE; } else { // Write out the bytes to destination array for (int k = 1; k <= count; ++k) { // @A2C //@P0D BinaryConverter.unsignedShortToByteArray(repeater, destination, j); // @A2C // destination[j] = (byte)(repeater >>> 8); //@P0A // destination[j+1] = (byte) repeater; //@P0A //@M7C destination[j] = repeatByte1; destination[j+1] = repeatByte2; //@P0A j += REPEATER_SIZE; } } } // Destination array overflow else { if (traceDiagnostic) { Trace.log(Trace.DIAGNOSTIC, "Overflow while decompressing RLE repeater record starting at dest " + j + " ..."); } // save each index only when overflow is encountered // the first time; otherwise, just tally up the number of // additional bytes needed if (!overflow) { saveI = i; saveJ = j; } overflow = true; bytesNeeded += (count * REPEATER_SIZE); j += (count * REPEATER_SIZE); } i += REPEATER_RECORD_SIZE; } // Error (don't have a complete RLE repeater record before EOD) else { if (traceError) Trace.log(Trace.ERROR, "Don't have a complete RLE repeater record before EOD ..."); throw new InternalErrorException(InternalErrorException.SYNTAX_ERROR ); } } } // End of source reached and have single escape byte else { if (traceError) Trace.log(Trace.ERROR, "Don't have a complete RLE escape record before EOD ..."); throw new InternalErrorException(InternalErrorException.SYNTAX_ERROR ); } } // No RLE record found; just copy bytes from source to destination arrray else { // Byte fits in destination array if (j < destinationLength) { // Add byte to destination array destination[j++] = source[i++]; } // Destination array overflow else { if (traceDiagnostic) { Trace.log(Trace.DIAGNOSTIC, "Overflow when writing out single bytes ..."); } // save each index only when overflow is encountered // the first time; otherwise, just tally up the number of // additional bytes needed if (!overflow) { saveI = i; saveJ = j; } overflow = true; bytesNeeded += 1; ++i; ++j; } } } // Buffer for decompressed bytes that will be returned. byte[] returnBytes; if ((overflow) && (reallocate)) { // Destination array too small. Add bytes needed to length // and create a return buffer of the correct length. j = saveJ; returnBytes = new byte[(j + bytesNeeded)]; int returnBytesLength = returnBytes.length; if (traceDiagnostic) { Trace.log(Trace.DIAGNOSTIC, "Overflow. Size updated to " + returnBytesLength + " bytes."); } System.arraycopy(destination, 0, returnBytes, 0, j); i = saveI; overflow = false; while (i < sourceLength) { // Have an escape byte if (source[i] == escape) { // Not end of source if ((i + ESCAPE_SIZE) < sourceLength) { // Second byte is escape; have an RLE escape record if (source[i + ESCAPE_SIZE] == escape) { // Add escape byte to destination array returnBytes[j++] = escape; i += ESCAPE_RECORD_SIZE; } // Should have an RLE repeater record; // have an escape byte followed by a non-escape byte else { // Not end of source; have a complete RLE repeater record if ((i + REPEATER_SIZE + COUNT_SIZE) < sourceLength) { // Get repeater //@P0D int repeater = BinaryConverter.byteArrayToUnsignedShort(source, i + ESCAPE_SIZE); // @A2C int repeater = ((source[i+ESCAPE_SIZE] & 0xFF) << 8) + (source[i+ESCAPE_SIZE+1] & 0xFF); //@P0A // Get repeat count //@P0D int count = BinaryConverter.byteArrayToUnsignedShort(source, i + ESCAPE_SIZE + REPEATER_SIZE); // @A2C int count = ((source[i+ESCAPE_SIZE+REPEATER_SIZE] & 0xFF) << 8) + (source[i+ESCAPE_SIZE+REPEATER_SIZE+1] & 0xFF); //@P0A // Write out the bytes to destination array for (int k = 1; k <= count; k++) { // @A2C //@P0D BinaryConverter.unsignedShortToByteArray(repeater, returnBytes, j); // @A2C returnBytes[j] = (byte)(repeater >>> 8); //@P0A returnBytes[j+1] = (byte)repeater; //@P0A j += REPEATER_SIZE; } i += REPEATER_RECORD_SIZE; } // Error (don't have a complete RLE repeater record before EOD) else { if (traceError) Trace.log(Trace.ERROR, "Don't have a complete RLE repeater record before EOD ..."); throw new InternalErrorException(InternalErrorException.SYNTAX_ERROR ); } } } // End of source reached and have single escape byte else { if (traceError) Trace.log(Trace.ERROR, "Don't have a complete RLE escape record before EOD ..."); throw new InternalErrorException(InternalErrorException.SYNTAX_ERROR ); } } // No RLE record found; just copy bytes from source to destination array else { returnBytes[j++] = source[i++]; } } return returnBytes; } else if ((destination.length > j) && (reallocate)) { // Destination array too big. Create a return buffer of the correct length, // copy data into it, and return the decompressed bytes. returnBytes = new byte[j]; System.arraycopy(destination, destinationOffset, returnBytes, 0, returnBytes.length); return returnBytes; } else { // Destination array is correct length. Return the decompressed bytes. return destination; } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy