This source file includes following definitions.
- bitmapFromData
 
- forwardLutFromBitmap
 
- reverseLutFromBitmap
 
- applyLut
 
- _channelData
 
- numScanLines
 
- format
 
- compress
 
- compressTile
 
- uncompress
 
- uncompressTile
 
- compress
 
- uncompress
 
#include <ImfPizCompressor.h>
#include <ImfHeader.h>
#include <ImfChannelList.h>
#include <ImfHuf.h>
#include <ImfWav.h>
#include <ImfMisc.h>
#include <ImfCheckedArithmetic.h>
#include <ImathFun.h>
#include <ImathBox.h>
#include <Iex.h>
#include <ImfIO.h>
#include <ImfXdr.h>
#include <ImfAutoArray.h>
#include <string.h>
#include <assert.h>
namespace Imf {
using Imath::divp;
using Imath::modp;
using Imath::Box2i;
using Imath::V2i;
using Iex::InputExc;
namespace {
const int USHORT_RANGE = (1 << 16);
const int BITMAP_SIZE  = (USHORT_RANGE >> 3);
void
bitmapFromData (const unsigned short data[],
        int nData,
        unsigned char bitmap[BITMAP_SIZE],
        unsigned short &minNonZero,
        unsigned short &maxNonZero)
{
    for (int i = 0; i < BITMAP_SIZE; ++i)
    bitmap[i] = 0;
    for (int i = 0; i < nData; ++i)
    bitmap[data[i] >> 3] |= (1 << (data[i] & 7));
    bitmap[0] &= ~1;                    
                    
                    
    minNonZero = BITMAP_SIZE - 1;
    maxNonZero = 0;
    for (int i = 0; i < BITMAP_SIZE; ++i)
    {
    if (bitmap[i])
    {
        if (minNonZero > i)
        minNonZero = i;
        if (maxNonZero < i)
        maxNonZero = i;
    }
    }
}
unsigned short
forwardLutFromBitmap (const unsigned char bitmap[BITMAP_SIZE],
              unsigned short lut[USHORT_RANGE])
{
    int k = 0;
    for (int i = 0; i < USHORT_RANGE; ++i)
    {
    if ((i == 0) || (bitmap[i >> 3] & (1 << (i & 7))))
        lut[i] = k++;
    else
        lut[i] = 0;
    }
    return k - 1;       
}                       
unsigned short
reverseLutFromBitmap (const unsigned char bitmap[BITMAP_SIZE],
              unsigned short lut[USHORT_RANGE])
{
    int k = 0;
    for (int i = 0; i < USHORT_RANGE; ++i)
    {
    if ((i == 0) || (bitmap[i >> 3] & (1 << (i & 7))))
        lut[k++] = i;
    }
    int n = k - 1;
    while (k < USHORT_RANGE)
    lut[k++] = 0;
    return n;           
}                       
void
applyLut (const unsigned short lut[USHORT_RANGE],
      unsigned short data[],
      int nData)
{
    for (int i = 0; i < nData; ++i)
    data[i] = lut[data[i]];
}
} 
struct PizCompressor::ChannelData
{
    unsigned short *    start;
    unsigned short *    end;
    int                 nx;
    int                 ny;
    int                 ys;
    int                 size;
};
PizCompressor::PizCompressor
    (const Header &hdr,
     size_t maxScanLineSize,
     size_t numScanLines)
:
    Compressor (hdr),
    _maxScanLineSize (maxScanLineSize),
    _format (XDR),
    _numScanLines (numScanLines),
    _tmpBuffer (0),
    _outBuffer (0),
    _numChans (0),
    _channels (hdr.channels()),
    _channelData (0)
{
    size_t tmpBufferSize =
                uiMult (maxScanLineSize, numScanLines) / 2;
    size_t outBufferSize =
                uiAdd (uiMult (maxScanLineSize, numScanLines),
                       size_t (65536 + 8192));
    _tmpBuffer = new unsigned short
            [checkArraySize (tmpBufferSize, sizeof (unsigned short))];
    _outBuffer = new char [outBufferSize];
    const ChannelList &channels = header().channels();
    bool onlyHalfChannels = true;
    for (ChannelList::ConstIterator c = channels.begin();
     c != channels.end();
     ++c)
    {
    _numChans++;
    assert (pixelTypeSize (c.channel().type) % pixelTypeSize (HALF) == 0);
    if (c.channel().type != HALF)
        onlyHalfChannels = false;
    }
    _channelData = new ChannelData[_numChans];
    const Box2i &dataWindow = hdr.dataWindow();
    _minX = dataWindow.min.x;
    _maxX = dataWindow.max.x;
    _maxY = dataWindow.max.y;
    
    
    
    
    
    if (onlyHalfChannels && (sizeof (half) == pixelTypeSize (HALF)))
    _format = NATIVE;
}
PizCompressor::~PizCompressor ()
{
    delete [] _tmpBuffer;
    delete [] _outBuffer;
    delete [] _channelData;
}
int
PizCompressor::numScanLines () const
{
    return _numScanLines;
}
Compressor::Format
PizCompressor::format () const
{
    return _format;
}
int
PizCompressor::compress (const char *inPtr,
             int inSize,
             int minY,
             const char *&outPtr)
{
    return compress (inPtr,
             inSize,
             Box2i (V2i (_minX, minY),
                V2i (_maxX, minY + numScanLines() - 1)),
             outPtr);
}
int
PizCompressor::compressTile (const char *inPtr,
                 int inSize,
                 Imath::Box2i range,
                 const char *&outPtr)
{
    return compress (inPtr, inSize, range, outPtr);
}
int
PizCompressor::uncompress (const char *inPtr,
               int inSize,
               int minY,
               const char *&outPtr)
{
    return uncompress (inPtr,
               inSize,
               Box2i (V2i (_minX, minY),
                  V2i (_maxX, minY + numScanLines() - 1)),
               outPtr);
}
int
PizCompressor::uncompressTile (const char *inPtr,
                   int inSize,
                   Imath::Box2i range,
                   const char *&outPtr)
{
    return uncompress (inPtr, inSize, range, outPtr);
}
int
PizCompressor::compress (const char *inPtr,
             int inSize,
             Imath::Box2i range,
             const char *&outPtr)
{
    
    
    
    
    
    
    
    if (inSize == 0)
    {
    outPtr = _outBuffer;
    return 0;
    }
    
    
    
    
    
    
    
    
    
    int minX = range.min.x;
    int maxX = range.max.x;
    int minY = range.min.y;
    int maxY = range.max.y;
    if (maxY > _maxY)
        maxY = _maxY;
    if (maxX > _maxX)
        maxX = _maxX;
    unsigned short *tmpBufferEnd = _tmpBuffer;
    int i = 0;
    for (ChannelList::ConstIterator c = _channels.begin();
     c != _channels.end();
     ++c, ++i)
    {
    ChannelData &cd = _channelData[i];
    cd.start = tmpBufferEnd;
    cd.end = cd.start;
    cd.nx = numSamples (c.channel().xSampling, minX, maxX);
    cd.ny = numSamples (c.channel().ySampling, minY, maxY);
    cd.ys = c.channel().ySampling;
    cd.size = pixelTypeSize (c.channel().type) / pixelTypeSize (HALF);
    tmpBufferEnd += cd.nx * cd.ny * cd.size;
    }
    if (_format == XDR)
    {
    
    
    
    for (int y = minY; y <= maxY; ++y)
    {
        for (int i = 0; i < _numChans; ++i)
        {
        ChannelData &cd = _channelData[i];
        if (modp (y, cd.ys) != 0)
            continue;
        for (int x = cd.nx * cd.size; x > 0; --x)
        {
            Xdr::read <CharPtrIO> (inPtr, *cd.end);
            ++cd.end;
        }
        }
    }
    }
    else
    {
    
    
    
    for (int y = minY; y <= maxY; ++y)
    {
        for (int i = 0; i < _numChans; ++i)
        {
        ChannelData &cd = _channelData[i];
        if (modp (y, cd.ys) != 0)
            continue;
        int n = cd.nx * cd.size;
        memcpy (cd.end, inPtr, n * sizeof (unsigned short));
        inPtr  += n * sizeof (unsigned short);
        cd.end += n;
        }
    }
    }
    #if defined (DEBUG)
    for (int i = 1; i < _numChans; ++i)
        assert (_channelData[i-1].end == _channelData[i].start);
    assert (_channelData[_numChans-1].end == tmpBufferEnd);
    #endif
    
    
    
    AutoArray <unsigned char, BITMAP_SIZE> bitmap;
    unsigned short minNonZero;
    unsigned short maxNonZero;
    bitmapFromData (_tmpBuffer,
            tmpBufferEnd - _tmpBuffer,
            bitmap,
            minNonZero, maxNonZero);
    AutoArray <unsigned short, USHORT_RANGE> lut;
    unsigned short maxValue = forwardLutFromBitmap (bitmap, lut);
    applyLut (lut, _tmpBuffer, tmpBufferEnd - _tmpBuffer);
    
    
    
    char *buf = _outBuffer;
    Xdr::write <CharPtrIO> (buf, minNonZero);
    Xdr::write <CharPtrIO> (buf, maxNonZero);
    if (minNonZero <= maxNonZero)
    {
    Xdr::write <CharPtrIO> (buf, (char *) &bitmap[0] + minNonZero,
                maxNonZero - minNonZero + 1);
    }
    
    
    
    for (int i = 0; i < _numChans; ++i)
    {
    ChannelData &cd = _channelData[i];
    for (int j = 0; j < cd.size; ++j)
    {
        wav2Encode (cd.start + j,
            cd.nx, cd.size,
            cd.ny, cd.nx * cd.size,
            maxValue);
    }
    }
    
    
    
    char *lengthPtr = buf;
    Xdr::write <CharPtrIO> (buf, int(0));
    int length = hufCompress (_tmpBuffer, tmpBufferEnd - _tmpBuffer, buf);
    Xdr::write <CharPtrIO> (lengthPtr, length);
    outPtr = _outBuffer;
    return buf - _outBuffer + length;
}
int
PizCompressor::uncompress (const char *inPtr,
               int inSize,
               Imath::Box2i range,
               const char *&outPtr)
{
    
    
    
    
    
    
    
    if (inSize == 0)
    {
    outPtr = _outBuffer;
    return 0;
    }
    
    
    
    int minX = range.min.x;
    int maxX = range.max.x;
    int minY = range.min.y;
    int maxY = range.max.y;
    if (maxY > _maxY)
        maxY = _maxY;
    if (maxX > _maxX)
        maxX = _maxX;
    unsigned short *tmpBufferEnd = _tmpBuffer;
    int i = 0;
    for (ChannelList::ConstIterator c = _channels.begin();
     c != _channels.end();
     ++c, ++i)
    {
    ChannelData &cd = _channelData[i];
    cd.start = tmpBufferEnd;
    cd.end = cd.start;
    cd.nx = numSamples (c.channel().xSampling, minX, maxX);
    cd.ny = numSamples (c.channel().ySampling, minY, maxY);
    cd.ys = c.channel().ySampling;
    cd.size = pixelTypeSize (c.channel().type) / pixelTypeSize (HALF);
    tmpBufferEnd += cd.nx * cd.ny * cd.size;
    }
    
    
    
    unsigned short minNonZero;
    unsigned short maxNonZero;
    AutoArray <unsigned char, BITMAP_SIZE> bitmap;
    memset (bitmap, 0, sizeof (unsigned char) * BITMAP_SIZE);
    Xdr::read <CharPtrIO> (inPtr, minNonZero);
    Xdr::read <CharPtrIO> (inPtr, maxNonZero);
    if (maxNonZero >= BITMAP_SIZE)
    {
    throw InputExc ("Error in header for PIZ-compressed data "
            "(invalid bitmap size).");
    }
    if (minNonZero <= maxNonZero)
    {
    Xdr::read <CharPtrIO> (inPtr, (char *) &bitmap[0] + minNonZero,
                   maxNonZero - minNonZero + 1);
    }
    AutoArray <unsigned short, USHORT_RANGE> lut;
    unsigned short maxValue = reverseLutFromBitmap (bitmap, lut);
    
    
    
    int length;
    Xdr::read <CharPtrIO> (inPtr, length);
    hufUncompress (inPtr, length, _tmpBuffer, tmpBufferEnd - _tmpBuffer);
    
    
    
    for (int i = 0; i < _numChans; ++i)
    {
    ChannelData &cd = _channelData[i];
    for (int j = 0; j < cd.size; ++j)
    {
        wav2Decode (cd.start + j,
            cd.nx, cd.size,
            cd.ny, cd.nx * cd.size,
            maxValue);
    }
    }
    
    
    
    applyLut (lut, _tmpBuffer, tmpBufferEnd - _tmpBuffer);
    
    
    
    char *outEnd = _outBuffer;
    if (_format == XDR)
    {
    
    
    
    for (int y = minY; y <= maxY; ++y)
    {
        for (int i = 0; i < _numChans; ++i)
        {
        ChannelData &cd = _channelData[i];
        if (modp (y, cd.ys) != 0)
            continue;
        for (int x = cd.nx * cd.size; x > 0; --x)
        {
            Xdr::write <CharPtrIO> (outEnd, *cd.end);
            ++cd.end;
        }
        }
    }
    }
    else
    {
    
    
    
    for (int y = minY; y <= maxY; ++y)
    {
        for (int i = 0; i < _numChans; ++i)
        {
        ChannelData &cd = _channelData[i];
        if (modp (y, cd.ys) != 0)
            continue;
        int n = cd.nx * cd.size;
        memcpy (outEnd, cd.end, n * sizeof (unsigned short));
        outEnd += n * sizeof (unsigned short);
        cd.end += n;
        }
    }
    }
    #if defined (DEBUG)
    for (int i = 1; i < _numChans; ++i)
        assert (_channelData[i-1].end == _channelData[i].start);
    assert (_channelData[_numChans-1].end == tmpBufferEnd);
    #endif
    outPtr = _outBuffer;
    return outEnd - _outBuffer;
}
}