Blame IlmImf/ImfPizCompressor.cpp

Packit 0d464f
///////////////////////////////////////////////////////////////////////////
Packit 0d464f
//
Packit 0d464f
// Copyright (c) 2004, Industrial Light & Magic, a division of Lucas
Packit 0d464f
// Digital Ltd. LLC
Packit 0d464f
// 
Packit 0d464f
// All rights reserved.
Packit 0d464f
// 
Packit 0d464f
// Redistribution and use in source and binary forms, with or without
Packit 0d464f
// modification, are permitted provided that the following conditions are
Packit 0d464f
// met:
Packit 0d464f
// *       Redistributions of source code must retain the above copyright
Packit 0d464f
// notice, this list of conditions and the following disclaimer.
Packit 0d464f
// *       Redistributions in binary form must reproduce the above
Packit 0d464f
// copyright notice, this list of conditions and the following disclaimer
Packit 0d464f
// in the documentation and/or other materials provided with the
Packit 0d464f
// distribution.
Packit 0d464f
// *       Neither the name of Industrial Light & Magic nor the names of
Packit 0d464f
// its contributors may be used to endorse or promote products derived
Packit 0d464f
// from this software without specific prior written permission. 
Packit 0d464f
// 
Packit 0d464f
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
Packit 0d464f
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
Packit 0d464f
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
Packit 0d464f
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
Packit 0d464f
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
Packit 0d464f
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
Packit 0d464f
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
Packit 0d464f
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
Packit 0d464f
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
Packit 0d464f
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
Packit 0d464f
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
Packit 0d464f
//
Packit 0d464f
///////////////////////////////////////////////////////////////////////////
Packit 0d464f
Packit 0d464f
Packit 0d464f
//-----------------------------------------------------------------------------
Packit 0d464f
//
Packit 0d464f
//	class PizCompressor
Packit 0d464f
//
Packit 0d464f
//-----------------------------------------------------------------------------
Packit 0d464f
Packit 0d464f
#include "ImfPizCompressor.h"
Packit 0d464f
#include "ImfHeader.h"
Packit 0d464f
#include "ImfChannelList.h"
Packit 0d464f
#include "ImfHuf.h"
Packit 0d464f
#include "ImfWav.h"
Packit 0d464f
#include "ImfMisc.h"
Packit 0d464f
#include "ImfCheckedArithmetic.h"
Packit 0d464f
#include <ImathFun.h>
Packit 0d464f
#include <ImathBox.h>
Packit 0d464f
#include <Iex.h>
Packit 0d464f
#include "ImfIO.h"
Packit 0d464f
#include "ImfXdr.h"
Packit 0d464f
#include "ImfAutoArray.h"
Packit 0d464f
#include <string.h>
Packit 0d464f
#include <assert.h>
Packit 0d464f
#include "ImfNamespace.h"
Packit 0d464f
Packit 0d464f
OPENEXR_IMF_INTERNAL_NAMESPACE_SOURCE_ENTER
Packit 0d464f
Packit 0d464f
using IMATH_NAMESPACE::divp;
Packit 0d464f
using IMATH_NAMESPACE::modp;
Packit 0d464f
using IMATH_NAMESPACE::Box2i;
Packit 0d464f
using IMATH_NAMESPACE::V2i;
Packit 0d464f
using IEX_NAMESPACE::InputExc;
Packit 0d464f
Packit 0d464f
namespace {
Packit 0d464f
Packit 0d464f
//
Packit 0d464f
// Functions to compress the range of values in the pixel data
Packit 0d464f
//
Packit 0d464f
Packit 0d464f
const int USHORT_RANGE = (1 << 16);
Packit 0d464f
const int BITMAP_SIZE  = (USHORT_RANGE >> 3);
Packit 0d464f
Packit 0d464f
void
Packit 0d464f
bitmapFromData (const unsigned short data[/*nData*/],
Packit 0d464f
		int nData,
Packit 0d464f
		unsigned char bitmap[BITMAP_SIZE],
Packit 0d464f
		unsigned short &minNonZero,
Packit 0d464f
		unsigned short &maxNonZero)
Packit 0d464f
{
Packit 0d464f
    for (int i = 0; i < BITMAP_SIZE; ++i)
Packit 0d464f
	bitmap[i] = 0;
Packit 0d464f
Packit 0d464f
    for (int i = 0; i < nData; ++i)
Packit 0d464f
	bitmap[data[i] >> 3] |= (1 << (data[i] & 7));
Packit 0d464f
Packit 0d464f
    bitmap[0] &= ~1;			// zero is not explicitly stored in
Packit 0d464f
					// the bitmap; we assume that the
Packit 0d464f
					// data always contain zeroes
Packit 0d464f
    minNonZero = BITMAP_SIZE - 1;
Packit 0d464f
    maxNonZero = 0;
Packit 0d464f
Packit 0d464f
    for (int i = 0; i < BITMAP_SIZE; ++i)
Packit 0d464f
    {
Packit 0d464f
	if (bitmap[i])
Packit 0d464f
	{
Packit 0d464f
	    if (minNonZero > i)
Packit 0d464f
		minNonZero = i;
Packit 0d464f
	    if (maxNonZero < i)
Packit 0d464f
		maxNonZero = i;
Packit 0d464f
	}
Packit 0d464f
    }
Packit 0d464f
}
Packit 0d464f
Packit 0d464f
Packit 0d464f
unsigned short
Packit 0d464f
forwardLutFromBitmap (const unsigned char bitmap[BITMAP_SIZE],
Packit 0d464f
		      unsigned short lut[USHORT_RANGE])
Packit 0d464f
{
Packit 0d464f
    int k = 0;
Packit 0d464f
Packit 0d464f
    for (int i = 0; i < USHORT_RANGE; ++i)
Packit 0d464f
    {
Packit 0d464f
	if ((i == 0) || (bitmap[i >> 3] & (1 << (i & 7))))
Packit 0d464f
	    lut[i] = k++;
Packit 0d464f
	else
Packit 0d464f
	    lut[i] = 0;
Packit 0d464f
    }
Packit 0d464f
Packit 0d464f
    return k - 1;	// maximum value stored in lut[],
Packit 0d464f
}			// i.e. number of ones in bitmap minus 1
Packit 0d464f
Packit 0d464f
Packit 0d464f
unsigned short
Packit 0d464f
reverseLutFromBitmap (const unsigned char bitmap[BITMAP_SIZE],
Packit 0d464f
		      unsigned short lut[USHORT_RANGE])
Packit 0d464f
{
Packit 0d464f
    int k = 0;
Packit 0d464f
Packit 0d464f
    for (int i = 0; i < USHORT_RANGE; ++i)
Packit 0d464f
    {
Packit 0d464f
	if ((i == 0) || (bitmap[i >> 3] & (1 << (i & 7))))
Packit 0d464f
	    lut[k++] = i;
Packit 0d464f
    }
Packit 0d464f
Packit 0d464f
    int n = k - 1;
Packit 0d464f
Packit 0d464f
    while (k < USHORT_RANGE)
Packit 0d464f
	lut[k++] = 0;
Packit 0d464f
Packit 0d464f
    return n;		// maximum k where lut[k] is non-zero,
Packit 0d464f
}			// i.e. number of ones in bitmap minus 1
Packit 0d464f
Packit 0d464f
Packit 0d464f
void
Packit 0d464f
applyLut (const unsigned short lut[USHORT_RANGE],
Packit 0d464f
	  unsigned short data[/*nData*/],
Packit 0d464f
	  int nData)
Packit 0d464f
{
Packit 0d464f
    for (int i = 0; i < nData; ++i)
Packit 0d464f
	data[i] = lut[data[i]];
Packit 0d464f
}
Packit 0d464f
Packit 0d464f
Packit 0d464f
} // namespace
Packit 0d464f
Packit 0d464f
Packit 0d464f
struct PizCompressor::ChannelData
Packit 0d464f
{
Packit 0d464f
    unsigned short *	start;
Packit 0d464f
    unsigned short *	end;
Packit 0d464f
    int			nx;
Packit 0d464f
    int			ny;
Packit 0d464f
    int			ys;
Packit 0d464f
    int			size;
Packit 0d464f
};
Packit 0d464f
Packit 0d464f
Packit 0d464f
PizCompressor::PizCompressor
Packit 0d464f
    (const Header &hdr,
Packit 0d464f
     size_t maxScanLineSize,
Packit 0d464f
     size_t numScanLines)
Packit 0d464f
:
Packit 0d464f
    Compressor (hdr),
Packit 0d464f
    _maxScanLineSize (maxScanLineSize),
Packit 0d464f
    _format (XDR),
Packit 0d464f
    _numScanLines (numScanLines),
Packit 0d464f
    _tmpBuffer (0),
Packit 0d464f
    _outBuffer (0),
Packit 0d464f
    _numChans (0),
Packit 0d464f
    _channels (hdr.channels()),
Packit 0d464f
    _channelData (0)
Packit 0d464f
{
Packit 0d464f
    size_t tmpBufferSize =
Packit 0d464f
                uiMult (maxScanLineSize, numScanLines) / 2;
Packit 0d464f
Packit 0d464f
    size_t outBufferSize =
Packit 0d464f
                uiAdd (uiMult (maxScanLineSize, numScanLines),
Packit 0d464f
                       size_t (65536 + 8192));
Packit 0d464f
Packit 0d464f
    _tmpBuffer = new unsigned short
Packit 0d464f
            [checkArraySize (tmpBufferSize, sizeof (unsigned short))];
Packit 0d464f
Packit 0d464f
    _outBuffer = new char [outBufferSize];
Packit 0d464f
Packit 0d464f
    const ChannelList &channels = header().channels();
Packit 0d464f
    bool onlyHalfChannels = true;
Packit 0d464f
Packit 0d464f
    for (ChannelList::ConstIterator c = channels.begin();
Packit 0d464f
	 c != channels.end();
Packit 0d464f
	 ++c)
Packit 0d464f
    {
Packit 0d464f
	_numChans++;
Packit 0d464f
Packit 0d464f
	assert (pixelTypeSize (c.channel().type) % pixelTypeSize (HALF) == 0);
Packit 0d464f
Packit 0d464f
	if (c.channel().type != HALF)
Packit 0d464f
	    onlyHalfChannels = false;
Packit 0d464f
    }
Packit 0d464f
Packit 0d464f
    _channelData = new ChannelData[_numChans];
Packit 0d464f
Packit 0d464f
    const Box2i &dataWindow = hdr.dataWindow();
Packit 0d464f
Packit 0d464f
    _minX = dataWindow.min.x;
Packit 0d464f
    _maxX = dataWindow.max.x;
Packit 0d464f
    _maxY = dataWindow.max.y;
Packit 0d464f
Packit 0d464f
    //
Packit 0d464f
    // We can support uncompressed data in the machine's native format
Packit 0d464f
    // if all image channels are of type HALF, and if the Xdr and the
Packit 0d464f
    // native represenations of a half have the same size.
Packit 0d464f
    //
Packit 0d464f
Packit 0d464f
    if (onlyHalfChannels && (sizeof (half) == pixelTypeSize (HALF)))
Packit 0d464f
	_format = NATIVE;
Packit 0d464f
}
Packit 0d464f
Packit 0d464f
Packit 0d464f
PizCompressor::~PizCompressor ()
Packit 0d464f
{
Packit 0d464f
    delete [] _tmpBuffer;
Packit 0d464f
    delete [] _outBuffer;
Packit 0d464f
    delete [] _channelData;
Packit 0d464f
}
Packit 0d464f
Packit 0d464f
Packit 0d464f
int
Packit 0d464f
PizCompressor::numScanLines () const
Packit 0d464f
{
Packit 0d464f
    return _numScanLines;
Packit 0d464f
}
Packit 0d464f
Packit 0d464f
Packit 0d464f
Compressor::Format
Packit 0d464f
PizCompressor::format () const
Packit 0d464f
{
Packit 0d464f
    return _format;
Packit 0d464f
}
Packit 0d464f
Packit 0d464f
Packit 0d464f
int
Packit 0d464f
PizCompressor::compress (const char *inPtr,
Packit 0d464f
			 int inSize,
Packit 0d464f
			 int minY,
Packit 0d464f
			 const char *&outPtr)
Packit 0d464f
{
Packit 0d464f
    return compress (inPtr,
Packit 0d464f
		     inSize,
Packit 0d464f
		     Box2i (V2i (_minX, minY),
Packit 0d464f
			    V2i (_maxX, minY + numScanLines() - 1)),
Packit 0d464f
		     outPtr);
Packit 0d464f
}
Packit 0d464f
Packit 0d464f
Packit 0d464f
int
Packit 0d464f
PizCompressor::compressTile (const char *inPtr,
Packit 0d464f
			     int inSize,
Packit 0d464f
			     IMATH_NAMESPACE::Box2i range,
Packit 0d464f
			     const char *&outPtr)
Packit 0d464f
{
Packit 0d464f
    return compress (inPtr, inSize, range, outPtr);
Packit 0d464f
}
Packit 0d464f
Packit 0d464f
Packit 0d464f
int
Packit 0d464f
PizCompressor::uncompress (const char *inPtr,
Packit 0d464f
			   int inSize,
Packit 0d464f
			   int minY,
Packit 0d464f
			   const char *&outPtr)
Packit 0d464f
{
Packit 0d464f
    return uncompress (inPtr,
Packit 0d464f
		       inSize,
Packit 0d464f
		       Box2i (V2i (_minX, minY),
Packit 0d464f
			      V2i (_maxX, minY + numScanLines() - 1)),
Packit 0d464f
		       outPtr);
Packit 0d464f
}
Packit 0d464f
Packit 0d464f
Packit 0d464f
int
Packit 0d464f
PizCompressor::uncompressTile (const char *inPtr,
Packit 0d464f
			       int inSize,
Packit 0d464f
			       IMATH_NAMESPACE::Box2i range,
Packit 0d464f
			       const char *&outPtr)
Packit 0d464f
{
Packit 0d464f
    return uncompress (inPtr, inSize, range, outPtr);
Packit 0d464f
}
Packit 0d464f
Packit 0d464f
Packit 0d464f
int
Packit 0d464f
PizCompressor::compress (const char *inPtr,
Packit 0d464f
			 int inSize,
Packit 0d464f
			 IMATH_NAMESPACE::Box2i range,
Packit 0d464f
			 const char *&outPtr)
Packit 0d464f
{
Packit 0d464f
    //
Packit 0d464f
    // This is the compress function which is used by both the tiled and
Packit 0d464f
    // scanline compression routines.
Packit 0d464f
    //
Packit 0d464f
Packit 0d464f
    //
Packit 0d464f
    // Special case �- empty input buffer
Packit 0d464f
    //
Packit 0d464f
Packit 0d464f
    if (inSize == 0)
Packit 0d464f
    {
Packit 0d464f
	outPtr = _outBuffer;
Packit 0d464f
	return 0;
Packit 0d464f
    }
Packit 0d464f
Packit 0d464f
    //
Packit 0d464f
    // Rearrange the pixel data so that the wavelet
Packit 0d464f
    // and Huffman encoders can process them easily.
Packit 0d464f
    //
Packit 0d464f
    // The wavelet and Huffman encoders both handle only
Packit 0d464f
    // 16-bit data, so 32-bit data must be split into smaller
Packit 0d464f
    // pieces.  We treat each 32-bit channel (UINT, FLOAT) as
Packit 0d464f
    // two interleaved 16-bit channels.
Packit 0d464f
    //
Packit 0d464f
Packit 0d464f
    int minX = range.min.x;
Packit 0d464f
    int maxX = range.max.x;
Packit 0d464f
    int minY = range.min.y;
Packit 0d464f
    int maxY = range.max.y;
Packit 0d464f
    
Packit 0d464f
    if (maxY > _maxY)
Packit 0d464f
        maxY = _maxY;
Packit 0d464f
    
Packit 0d464f
    if (maxX > _maxX)
Packit 0d464f
        maxX = _maxX;
Packit 0d464f
Packit 0d464f
    unsigned short *tmpBufferEnd = _tmpBuffer;
Packit 0d464f
    int i = 0;
Packit 0d464f
Packit 0d464f
    for (ChannelList::ConstIterator c = _channels.begin();
Packit 0d464f
	 c != _channels.end();
Packit 0d464f
	 ++c, ++i)
Packit 0d464f
    {
Packit 0d464f
	ChannelData &cd = _channelData[i];
Packit 0d464f
Packit 0d464f
	cd.start = tmpBufferEnd;
Packit 0d464f
	cd.end = cd.start;
Packit 0d464f
Packit 0d464f
	cd.nx = numSamples (c.channel().xSampling, minX, maxX);
Packit 0d464f
	cd.ny = numSamples (c.channel().ySampling, minY, maxY);
Packit 0d464f
	cd.ys = c.channel().ySampling;
Packit 0d464f
Packit 0d464f
	cd.size = pixelTypeSize (c.channel().type) / pixelTypeSize (HALF);
Packit 0d464f
Packit 0d464f
	tmpBufferEnd += cd.nx * cd.ny * cd.size;
Packit 0d464f
    }
Packit 0d464f
Packit 0d464f
    if (_format == XDR)
Packit 0d464f
    {
Packit 0d464f
	//
Packit 0d464f
	// Machine-independent (Xdr) data format
Packit 0d464f
	//
Packit 0d464f
Packit 0d464f
	for (int y = minY; y <= maxY; ++y)
Packit 0d464f
	{
Packit 0d464f
	    for (int i = 0; i < _numChans; ++i)
Packit 0d464f
	    {
Packit 0d464f
		ChannelData &cd = _channelData[i];
Packit 0d464f
Packit 0d464f
		if (modp (y, cd.ys) != 0)
Packit 0d464f
		    continue;
Packit 0d464f
Packit 0d464f
		for (int x = cd.nx * cd.size; x > 0; --x)
Packit 0d464f
		{
Packit 0d464f
		    Xdr::read <CharPtrIO> (inPtr, *cd.end);
Packit 0d464f
		    ++cd.end;
Packit 0d464f
		}
Packit 0d464f
	    }
Packit 0d464f
	}
Packit 0d464f
    }
Packit 0d464f
    else
Packit 0d464f
    {
Packit 0d464f
	//
Packit 0d464f
	// Native, machine-dependent data format
Packit 0d464f
	//
Packit 0d464f
Packit 0d464f
	for (int y = minY; y <= maxY; ++y)
Packit 0d464f
	{
Packit 0d464f
	    for (int i = 0; i < _numChans; ++i)
Packit 0d464f
	    {
Packit 0d464f
		ChannelData &cd = _channelData[i];
Packit 0d464f
Packit 0d464f
		if (modp (y, cd.ys) != 0)
Packit 0d464f
		    continue;
Packit 0d464f
Packit 0d464f
		int n = cd.nx * cd.size;
Packit 0d464f
		memcpy (cd.end, inPtr, n * sizeof (unsigned short));
Packit 0d464f
		inPtr  += n * sizeof (unsigned short);
Packit 0d464f
		cd.end += n;
Packit 0d464f
	    }
Packit 0d464f
	}
Packit 0d464f
    }
Packit 0d464f
Packit 0d464f
    #if defined (DEBUG)
Packit 0d464f
Packit 0d464f
	for (int i = 1; i < _numChans; ++i)
Packit 0d464f
	    assert (_channelData[i-1].end == _channelData[i].start);
Packit 0d464f
Packit 0d464f
	assert (_channelData[_numChans-1].end == tmpBufferEnd);
Packit 0d464f
Packit 0d464f
    #endif
Packit 0d464f
Packit 0d464f
    //
Packit 0d464f
    // Compress the range of the pixel data
Packit 0d464f
    //
Packit 0d464f
Packit 0d464f
    AutoArray <unsigned char, BITMAP_SIZE> bitmap;
Packit 0d464f
    unsigned short minNonZero;
Packit 0d464f
    unsigned short maxNonZero;
Packit 0d464f
Packit 0d464f
    bitmapFromData (_tmpBuffer,
Packit 0d464f
		    tmpBufferEnd - _tmpBuffer,
Packit 0d464f
		    bitmap,
Packit 0d464f
		    minNonZero, maxNonZero);
Packit 0d464f
Packit 0d464f
    AutoArray <unsigned short, USHORT_RANGE> lut;
Packit 0d464f
    unsigned short maxValue = forwardLutFromBitmap (bitmap, lut);
Packit 0d464f
    applyLut (lut, _tmpBuffer, tmpBufferEnd - _tmpBuffer);
Packit 0d464f
Packit 0d464f
    //
Packit 0d464f
    // Store range compression info in _outBuffer
Packit 0d464f
    //
Packit 0d464f
Packit 0d464f
    char *buf = _outBuffer;
Packit 0d464f
Packit 0d464f
    Xdr::write <CharPtrIO> (buf, minNonZero);
Packit 0d464f
    Xdr::write <CharPtrIO> (buf, maxNonZero);
Packit 0d464f
Packit 0d464f
    if (minNonZero <= maxNonZero)
Packit 0d464f
    {
Packit 0d464f
	Xdr::write <CharPtrIO> (buf, (char *) &bitmap[0] + minNonZero,
Packit 0d464f
				maxNonZero - minNonZero + 1);
Packit 0d464f
    }
Packit 0d464f
Packit 0d464f
    //
Packit 0d464f
    // Apply wavelet encoding
Packit 0d464f
    //
Packit 0d464f
Packit 0d464f
    for (int i = 0; i < _numChans; ++i)
Packit 0d464f
    {
Packit 0d464f
	ChannelData &cd = _channelData[i];
Packit 0d464f
Packit 0d464f
	for (int j = 0; j < cd.size; ++j)
Packit 0d464f
	{
Packit 0d464f
	    wav2Encode (cd.start + j,
Packit 0d464f
			cd.nx, cd.size,
Packit 0d464f
			cd.ny, cd.nx * cd.size,
Packit 0d464f
			maxValue);
Packit 0d464f
	}
Packit 0d464f
    }
Packit 0d464f
Packit 0d464f
    //
Packit 0d464f
    // Apply Huffman encoding; append the result to _outBuffer
Packit 0d464f
    //
Packit 0d464f
Packit 0d464f
    char *lengthPtr = buf;
Packit 0d464f
    Xdr::write <CharPtrIO> (buf, int(0));
Packit 0d464f
Packit 0d464f
    int length = hufCompress (_tmpBuffer, tmpBufferEnd - _tmpBuffer, buf);
Packit 0d464f
    Xdr::write <CharPtrIO> (lengthPtr, length);
Packit 0d464f
Packit 0d464f
    outPtr = _outBuffer;
Packit 0d464f
    return buf - _outBuffer + length;
Packit 0d464f
}
Packit 0d464f
Packit 0d464f
Packit 0d464f
int
Packit 0d464f
PizCompressor::uncompress (const char *inPtr,
Packit 0d464f
			   int inSize,
Packit 0d464f
			   IMATH_NAMESPACE::Box2i range,
Packit 0d464f
			   const char *&outPtr)
Packit 0d464f
{
Packit 0d464f
    //
Packit 0d464f
    // This is the cunompress function which is used by both the tiled and
Packit 0d464f
    // scanline decompression routines.
Packit 0d464f
    //
Packit 0d464f
    
Packit 0d464f
    //
Packit 0d464f
    // Special case - empty input buffer
Packit 0d464f
    //
Packit 0d464f
Packit 0d464f
    if (inSize == 0)
Packit 0d464f
    {
Packit 0d464f
	outPtr = _outBuffer;
Packit 0d464f
	return 0;
Packit 0d464f
    }
Packit 0d464f
Packit 0d464f
    //
Packit 0d464f
    // Determine the layout of the compressed pixel data
Packit 0d464f
    //
Packit 0d464f
Packit 0d464f
    int minX = range.min.x;
Packit 0d464f
    int maxX = range.max.x;
Packit 0d464f
    int minY = range.min.y;
Packit 0d464f
    int maxY = range.max.y;
Packit 0d464f
    
Packit 0d464f
    if (maxY > _maxY)
Packit 0d464f
        maxY = _maxY;
Packit 0d464f
    
Packit 0d464f
    if (maxX > _maxX)
Packit 0d464f
        maxX = _maxX;
Packit 0d464f
Packit 0d464f
    unsigned short *tmpBufferEnd = _tmpBuffer;
Packit 0d464f
    int i = 0;
Packit 0d464f
Packit 0d464f
    for (ChannelList::ConstIterator c = _channels.begin();
Packit 0d464f
	 c != _channels.end();
Packit 0d464f
	 ++c, ++i)
Packit 0d464f
    {
Packit 0d464f
	ChannelData &cd = _channelData[i];
Packit 0d464f
Packit 0d464f
	cd.start = tmpBufferEnd;
Packit 0d464f
	cd.end = cd.start;
Packit 0d464f
Packit 0d464f
	cd.nx = numSamples (c.channel().xSampling, minX, maxX);
Packit 0d464f
	cd.ny = numSamples (c.channel().ySampling, minY, maxY);
Packit 0d464f
	cd.ys = c.channel().ySampling;
Packit 0d464f
Packit 0d464f
	cd.size = pixelTypeSize (c.channel().type) / pixelTypeSize (HALF);
Packit 0d464f
Packit 0d464f
	tmpBufferEnd += cd.nx * cd.ny * cd.size;
Packit 0d464f
    }
Packit 0d464f
Packit 0d464f
    //
Packit 0d464f
    // Read range compression data
Packit 0d464f
    //
Packit 0d464f
Packit 0d464f
    unsigned short minNonZero;
Packit 0d464f
    unsigned short maxNonZero;
Packit 0d464f
Packit 0d464f
    AutoArray <unsigned char, BITMAP_SIZE> bitmap;
Packit 0d464f
    memset (bitmap, 0, sizeof (unsigned char) * BITMAP_SIZE);
Packit 0d464f
Packit 0d464f
    Xdr::read <CharPtrIO> (inPtr, minNonZero);
Packit 0d464f
    Xdr::read <CharPtrIO> (inPtr, maxNonZero);
Packit 0d464f
Packit 0d464f
    if (maxNonZero >= BITMAP_SIZE)
Packit 0d464f
    {
Packit 0d464f
	throw InputExc ("Error in header for PIZ-compressed data "
Packit 0d464f
			"(invalid bitmap size).");
Packit 0d464f
    }
Packit 0d464f
Packit 0d464f
    if (minNonZero <= maxNonZero)
Packit 0d464f
    {
Packit 0d464f
	Xdr::read <CharPtrIO> (inPtr, (char *) &bitmap[0] + minNonZero,
Packit 0d464f
			       maxNonZero - minNonZero + 1);
Packit 0d464f
    }
Packit 0d464f
Packit 0d464f
    AutoArray <unsigned short, USHORT_RANGE> lut;
Packit 0d464f
    unsigned short maxValue = reverseLutFromBitmap (bitmap, lut);
Packit 0d464f
Packit 0d464f
    //
Packit 0d464f
    // Huffman decoding
Packit 0d464f
    //
Packit 0d464f
Packit 0d464f
    int length;
Packit 0d464f
    Xdr::read <CharPtrIO> (inPtr, length);
Packit 0d464f
Packit 0d464f
    hufUncompress (inPtr, length, _tmpBuffer, tmpBufferEnd - _tmpBuffer);
Packit 0d464f
Packit 0d464f
    //
Packit 0d464f
    // Wavelet decoding
Packit 0d464f
    //
Packit 0d464f
Packit 0d464f
    for (int i = 0; i < _numChans; ++i)
Packit 0d464f
    {
Packit 0d464f
	ChannelData &cd = _channelData[i];
Packit 0d464f
Packit 0d464f
	for (int j = 0; j < cd.size; ++j)
Packit 0d464f
	{
Packit 0d464f
	    wav2Decode (cd.start + j,
Packit 0d464f
			cd.nx, cd.size,
Packit 0d464f
			cd.ny, cd.nx * cd.size,
Packit 0d464f
			maxValue);
Packit 0d464f
	}
Packit 0d464f
    }
Packit 0d464f
Packit 0d464f
    //
Packit 0d464f
    // Expand the pixel data to their original range
Packit 0d464f
    //
Packit 0d464f
Packit 0d464f
    applyLut (lut, _tmpBuffer, tmpBufferEnd - _tmpBuffer);
Packit 0d464f
    
Packit 0d464f
    //
Packit 0d464f
    // Rearrange the pixel data into the format expected by the caller.
Packit 0d464f
    //
Packit 0d464f
Packit 0d464f
    char *outEnd = _outBuffer;
Packit 0d464f
Packit 0d464f
    if (_format == XDR)
Packit 0d464f
    {
Packit 0d464f
	//
Packit 0d464f
	// Machine-independent (Xdr) data format
Packit 0d464f
	//
Packit 0d464f
Packit 0d464f
	for (int y = minY; y <= maxY; ++y)
Packit 0d464f
	{
Packit 0d464f
	    for (int i = 0; i < _numChans; ++i)
Packit 0d464f
	    {
Packit 0d464f
		ChannelData &cd = _channelData[i];
Packit 0d464f
Packit 0d464f
		if (modp (y, cd.ys) != 0)
Packit 0d464f
		    continue;
Packit 0d464f
Packit 0d464f
		for (int x = cd.nx * cd.size; x > 0; --x)
Packit 0d464f
		{
Packit 0d464f
		    Xdr::write <CharPtrIO> (outEnd, *cd.end);
Packit 0d464f
		    ++cd.end;
Packit 0d464f
		}
Packit 0d464f
	    }
Packit 0d464f
	}
Packit 0d464f
    }
Packit 0d464f
    else
Packit 0d464f
    {
Packit 0d464f
	//
Packit 0d464f
	// Native, machine-dependent data format
Packit 0d464f
	//
Packit 0d464f
Packit 0d464f
	for (int y = minY; y <= maxY; ++y)
Packit 0d464f
	{
Packit 0d464f
	    for (int i = 0; i < _numChans; ++i)
Packit 0d464f
	    {
Packit 0d464f
		ChannelData &cd = _channelData[i];
Packit 0d464f
Packit 0d464f
		if (modp (y, cd.ys) != 0)
Packit 0d464f
		    continue;
Packit 0d464f
Packit 0d464f
		int n = cd.nx * cd.size;
Packit 0d464f
		memcpy (outEnd, cd.end, n * sizeof (unsigned short));
Packit 0d464f
		outEnd += n * sizeof (unsigned short);
Packit 0d464f
		cd.end += n;
Packit 0d464f
	    }
Packit 0d464f
	}
Packit 0d464f
    }
Packit 0d464f
Packit 0d464f
    #if defined (DEBUG)
Packit 0d464f
Packit 0d464f
	for (int i = 1; i < _numChans; ++i)
Packit 0d464f
	    assert (_channelData[i-1].end == _channelData[i].start);
Packit 0d464f
Packit 0d464f
	assert (_channelData[_numChans-1].end == tmpBufferEnd);
Packit 0d464f
Packit 0d464f
    #endif
Packit 0d464f
Packit 0d464f
    outPtr = _outBuffer;
Packit 0d464f
    return outEnd - _outBuffer;
Packit 0d464f
}
Packit 0d464f
Packit 0d464f
Packit 0d464f
OPENEXR_IMF_INTERNAL_NAMESPACE_SOURCE_EXIT