Blame IlmImf/ImfTileOffsets.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 TileOffsets
Packit 0d464f
//
Packit 0d464f
//-----------------------------------------------------------------------------
Packit 0d464f
Packit 0d464f
#include <ImfTileOffsets.h>
Packit 0d464f
#include <ImfXdr.h>
Packit 0d464f
#include <ImfIO.h>
Packit 0d464f
#include "Iex.h"
Packit 0d464f
#include "ImfNamespace.h"
Packit 0d464f
#include <algorithm>
Packit 0d464f
Packit 0d464f
OPENEXR_IMF_INTERNAL_NAMESPACE_SOURCE_ENTER
Packit 0d464f
Packit 0d464f
Packit 0d464f
TileOffsets::TileOffsets (LevelMode mode,
Packit 0d464f
			  int numXLevels, int numYLevels,
Packit 0d464f
			  const int *numXTiles, const int *numYTiles)
Packit 0d464f
:
Packit 0d464f
    _mode (mode),
Packit 0d464f
    _numXLevels (numXLevels),
Packit 0d464f
    _numYLevels (numYLevels)
Packit 0d464f
{
Packit 0d464f
    switch (_mode)
Packit 0d464f
    {
Packit 0d464f
      case ONE_LEVEL:
Packit 0d464f
      case MIPMAP_LEVELS:
Packit 0d464f
Packit 0d464f
        _offsets.resize (_numXLevels);
Packit 0d464f
Packit 0d464f
        for (unsigned int l = 0; l < _offsets.size(); ++l)
Packit 0d464f
        {
Packit 0d464f
            _offsets[l].resize (numYTiles[l]);
Packit 0d464f
Packit 0d464f
            for (unsigned int dy = 0; dy < _offsets[l].size(); ++dy)
Packit 0d464f
	    {
Packit 0d464f
                _offsets[l][dy].resize (numXTiles[l]);
Packit 0d464f
            }
Packit 0d464f
        }
Packit 0d464f
        break;
Packit 0d464f
Packit 0d464f
      case RIPMAP_LEVELS:
Packit 0d464f
Packit 0d464f
        _offsets.resize (_numXLevels * _numYLevels);
Packit 0d464f
Packit 0d464f
        for (int ly = 0; ly < _numYLevels; ++ly)
Packit 0d464f
        {
Packit 0d464f
            for (int lx = 0; lx < _numXLevels; ++lx)
Packit 0d464f
            {
Packit 0d464f
                int l = ly * _numXLevels + lx;
Packit 0d464f
                _offsets[l].resize (numYTiles[ly]);
Packit 0d464f
Packit 0d464f
                for (size_t dy = 0; dy < _offsets[l].size(); ++dy)
Packit 0d464f
                {
Packit 0d464f
                    _offsets[l][dy].resize (numXTiles[lx]);
Packit 0d464f
                }
Packit 0d464f
            }
Packit 0d464f
        }
Packit 0d464f
        break;
Packit 0d464f
Packit 0d464f
      case NUM_LEVELMODES :
Packit 0d464f
          throw IEX_NAMESPACE::ArgExc("Bad initialisation of TileOffsets object");
Packit 0d464f
    }
Packit 0d464f
}
Packit 0d464f
Packit 0d464f
Packit 0d464f
bool
Packit 0d464f
TileOffsets::anyOffsetsAreInvalid () const
Packit 0d464f
{
Packit 0d464f
    for (unsigned int l = 0; l < _offsets.size(); ++l)
Packit 0d464f
	for (unsigned int dy = 0; dy < _offsets[l].size(); ++dy)
Packit 0d464f
	    for (unsigned int dx = 0; dx < _offsets[l][dy].size(); ++dx)
Packit 0d464f
		if (_offsets[l][dy][dx] <= 0)
Packit 0d464f
		    return true;
Packit 0d464f
    
Packit 0d464f
    return false;
Packit 0d464f
}
Packit 0d464f
Packit 0d464f
Packit 0d464f
void
Packit 0d464f
TileOffsets::findTiles (OPENEXR_IMF_INTERNAL_NAMESPACE::IStream &is, bool isMultiPartFile, bool isDeep, bool skipOnly)
Packit 0d464f
{
Packit 0d464f
    for (unsigned int l = 0; l < _offsets.size(); ++l)
Packit 0d464f
    {
Packit 0d464f
	for (unsigned int dy = 0; dy < _offsets[l].size(); ++dy)
Packit 0d464f
	{
Packit 0d464f
	    for (unsigned int dx = 0; dx < _offsets[l][dy].size(); ++dx)
Packit 0d464f
	    {
Packit 0d464f
		Int64 tileOffset = is.tellg();
Packit 0d464f
Packit 0d464f
		if (isMultiPartFile)
Packit 0d464f
		{
Packit 0d464f
		    int partNumber;
Packit 0d464f
		    OPENEXR_IMF_INTERNAL_NAMESPACE::Xdr::read <OPENEXR_IMF_INTERNAL_NAMESPACE::StreamIO> (is, partNumber);
Packit 0d464f
		}
Packit 0d464f
Packit 0d464f
		int tileX;
Packit 0d464f
		OPENEXR_IMF_INTERNAL_NAMESPACE::Xdr::read <OPENEXR_IMF_INTERNAL_NAMESPACE::StreamIO> (is, tileX);
Packit 0d464f
Packit 0d464f
		int tileY;
Packit 0d464f
		OPENEXR_IMF_INTERNAL_NAMESPACE::Xdr::read <OPENEXR_IMF_INTERNAL_NAMESPACE::StreamIO> (is, tileY);
Packit 0d464f
Packit 0d464f
		int levelX;
Packit 0d464f
		OPENEXR_IMF_INTERNAL_NAMESPACE::Xdr::read <OPENEXR_IMF_INTERNAL_NAMESPACE::StreamIO> (is, levelX);
Packit 0d464f
Packit 0d464f
		int levelY;
Packit 0d464f
		OPENEXR_IMF_INTERNAL_NAMESPACE::Xdr::read <OPENEXR_IMF_INTERNAL_NAMESPACE::StreamIO> (is, levelY);
Packit 0d464f
Packit 0d464f
                if(isDeep)
Packit 0d464f
                {
Packit 0d464f
                     Int64 packed_offset_table_size;
Packit 0d464f
                     Int64 packed_sample_size;
Packit 0d464f
                     
Packit 0d464f
                     OPENEXR_IMF_INTERNAL_NAMESPACE::Xdr::read <OPENEXR_IMF_INTERNAL_NAMESPACE::StreamIO> (is, packed_offset_table_size);
Packit 0d464f
                     OPENEXR_IMF_INTERNAL_NAMESPACE::Xdr::read <OPENEXR_IMF_INTERNAL_NAMESPACE::StreamIO> (is, packed_sample_size);
Packit 0d464f
                     
Packit 0d464f
                     // next Int64 is unpacked sample size - skip that too
Packit 0d464f
                     Xdr::skip <StreamIO> (is, packed_offset_table_size+packed_sample_size+8);
Packit 0d464f
                    
Packit 0d464f
                }else{
Packit 0d464f
                    
Packit 0d464f
		     int dataSize;
Packit 0d464f
		     OPENEXR_IMF_INTERNAL_NAMESPACE::Xdr::read <OPENEXR_IMF_INTERNAL_NAMESPACE::StreamIO> (is, dataSize);
Packit 0d464f
Packit 0d464f
		     Xdr::skip <StreamIO> (is, dataSize);
Packit 0d464f
                }
Packit 0d464f
		if (skipOnly) continue;
Packit 0d464f
Packit 0d464f
		if (!isValidTile(tileX, tileY, levelX, levelY))
Packit 0d464f
		    return;
Packit 0d464f
Packit 0d464f
		operator () (tileX, tileY, levelX, levelY) = tileOffset;
Packit 0d464f
	    }
Packit 0d464f
	}
Packit 0d464f
    }
Packit 0d464f
}
Packit 0d464f
Packit 0d464f
Packit 0d464f
void
Packit 0d464f
TileOffsets::reconstructFromFile (OPENEXR_IMF_INTERNAL_NAMESPACE::IStream &is,bool isMultiPart,bool isDeep)
Packit 0d464f
{
Packit 0d464f
    //
Packit 0d464f
    // Try to reconstruct a missing tile offset table by sequentially
Packit 0d464f
    // scanning through the file, and recording the offsets in the file
Packit 0d464f
    // of the tiles we find.
Packit 0d464f
    //
Packit 0d464f
Packit 0d464f
    Int64 position = is.tellg();
Packit 0d464f
Packit 0d464f
    try
Packit 0d464f
    {
Packit 0d464f
	findTiles (is,isMultiPart,isDeep,false);
Packit 0d464f
    }
Packit 0d464f
    catch (...)
Packit 0d464f
    {
Packit 0d464f
        //
Packit 0d464f
        // Suppress all exceptions.  This function is called only to
Packit 0d464f
	// reconstruct the tile offset table for incomplete files,
Packit 0d464f
	// and exceptions are likely.
Packit 0d464f
        //
Packit 0d464f
    }
Packit 0d464f
Packit 0d464f
    is.clear();
Packit 0d464f
    is.seekg (position);
Packit 0d464f
}
Packit 0d464f
Packit 0d464f
Packit 0d464f
void
Packit 0d464f
TileOffsets::readFrom (OPENEXR_IMF_INTERNAL_NAMESPACE::IStream &is, bool &complete,bool isMultiPartFile, bool isDeep)
Packit 0d464f
{
Packit 0d464f
    //
Packit 0d464f
    // Read in the tile offsets from the file's tile offset table
Packit 0d464f
    //
Packit 0d464f
Packit 0d464f
    for (unsigned int l = 0; l < _offsets.size(); ++l)
Packit 0d464f
	for (unsigned int dy = 0; dy < _offsets[l].size(); ++dy)
Packit 0d464f
	    for (unsigned int dx = 0; dx < _offsets[l][dy].size(); ++dx)
Packit 0d464f
		OPENEXR_IMF_INTERNAL_NAMESPACE::Xdr::read <OPENEXR_IMF_INTERNAL_NAMESPACE::StreamIO> (is, _offsets[l][dy][dx]);
Packit 0d464f
Packit 0d464f
    //
Packit 0d464f
    // Check if any tile offsets are invalid.
Packit 0d464f
    //
Packit 0d464f
    // Invalid offsets mean that the file is probably incomplete
Packit 0d464f
    // (the offset table is the last thing written to the file).
Packit 0d464f
    // Either some process is still busy writing the file, or
Packit 0d464f
    // writing the file was aborted.
Packit 0d464f
    //
Packit 0d464f
    // We should still be able to read the existing parts of the
Packit 0d464f
    // file.  In order to do this, we have to make a sequential
Packit 0d464f
    // scan over the scan tile to reconstruct the tile offset
Packit 0d464f
    // table.
Packit 0d464f
    //
Packit 0d464f
Packit 0d464f
    if (anyOffsetsAreInvalid())
Packit 0d464f
    {
Packit 0d464f
	complete = false;
Packit 0d464f
	reconstructFromFile (is,isMultiPartFile,isDeep);
Packit 0d464f
    }
Packit 0d464f
    else
Packit 0d464f
    {
Packit 0d464f
	complete = true;
Packit 0d464f
    }
Packit 0d464f
Packit 0d464f
}
Packit 0d464f
Packit 0d464f
Packit 0d464f
void
Packit 0d464f
TileOffsets::readFrom (std::vector<Int64> chunkOffsets,bool &complete)
Packit 0d464f
{
Packit 0d464f
    size_t totalSize = 0;
Packit 0d464f
 
Packit 0d464f
    for (unsigned int l = 0; l < _offsets.size(); ++l)
Packit 0d464f
        for (unsigned int dy = 0; dy < _offsets[l].size(); ++dy)
Packit 0d464f
            totalSize += _offsets[l][dy].size();
Packit 0d464f
Packit 0d464f
    if (chunkOffsets.size() != totalSize)
Packit 0d464f
        throw IEX_NAMESPACE::ArgExc ("Wrong offset count, not able to read from this array");
Packit 0d464f
Packit 0d464f
Packit 0d464f
Packit 0d464f
    int pos = 0;
Packit 0d464f
    for (size_t l = 0; l < _offsets.size(); ++l)
Packit 0d464f
        for (size_t dy = 0; dy < _offsets[l].size(); ++dy)
Packit 0d464f
            for (size_t dx = 0; dx < _offsets[l][dy].size(); ++dx)
Packit 0d464f
            {
Packit 0d464f
                _offsets[l][dy][dx] = chunkOffsets[pos];
Packit 0d464f
                pos++;
Packit 0d464f
            }
Packit 0d464f
Packit 0d464f
    complete = !anyOffsetsAreInvalid();
Packit 0d464f
Packit 0d464f
}
Packit 0d464f
Packit 0d464f
Packit 0d464f
Int64
Packit 0d464f
TileOffsets::writeTo (OPENEXR_IMF_INTERNAL_NAMESPACE::OStream &os) const
Packit 0d464f
{
Packit 0d464f
    //
Packit 0d464f
    // Write the tile offset table to the file, and
Packit 0d464f
    // return the position of the start of the table
Packit 0d464f
    // in the file.
Packit 0d464f
    //
Packit 0d464f
    
Packit 0d464f
    Int64 pos = os.tellp();
Packit 0d464f
Packit 0d464f
    if (pos == -1)
Packit 0d464f
	IEX_NAMESPACE::throwErrnoExc ("Cannot determine current file position (%T).");
Packit 0d464f
Packit 0d464f
    for (unsigned int l = 0; l < _offsets.size(); ++l)
Packit 0d464f
	for (unsigned int dy = 0; dy < _offsets[l].size(); ++dy)
Packit 0d464f
	    for (unsigned int dx = 0; dx < _offsets[l][dy].size(); ++dx)
Packit 0d464f
		OPENEXR_IMF_INTERNAL_NAMESPACE::Xdr::write <OPENEXR_IMF_INTERNAL_NAMESPACE::StreamIO> (os, _offsets[l][dy][dx]);
Packit 0d464f
Packit 0d464f
    return pos;
Packit 0d464f
}
Packit 0d464f
Packit 0d464f
namespace {
Packit 0d464f
struct tilepos{
Packit 0d464f
    Int64 filePos;
Packit 0d464f
    int dx;
Packit 0d464f
    int dy;
Packit 0d464f
    int l;
Packit 0d464f
    bool operator <(const tilepos & other) const
Packit 0d464f
    {
Packit 0d464f
        return filePos < other.filePos;
Packit 0d464f
    }
Packit 0d464f
};
Packit 0d464f
}
Packit 0d464f
//-------------------------------------
Packit 0d464f
// fill array with tile coordinates in the order they appear in the file
Packit 0d464f
//
Packit 0d464f
// each input array must be of size (totalTiles)
Packit 0d464f
// 
Packit 0d464f
//
Packit 0d464f
// if the tile order is not RANDOM_Y, it is more efficient to compute the
Packit 0d464f
// tile ordering rather than using this function
Packit 0d464f
//
Packit 0d464f
//-------------------------------------
Packit 0d464f
void TileOffsets::getTileOrder(int dx_table[],int dy_table[],int lx_table[],int ly_table[]) const
Packit 0d464f
{
Packit 0d464f
    // 
Packit 0d464f
    // helper class
Packit 0d464f
    // 
Packit 0d464f
Packit 0d464f
    // how many entries?
Packit 0d464f
    size_t entries=0;
Packit 0d464f
    for (unsigned int l = 0; l < _offsets.size(); ++l)
Packit 0d464f
        for (unsigned int dy = 0; dy < _offsets[l].size(); ++dy)
Packit 0d464f
           entries+=_offsets[l][dy].size();
Packit 0d464f
        
Packit 0d464f
    std::vector<struct tilepos> table(entries);
Packit 0d464f
    
Packit 0d464f
    size_t i = 0;
Packit 0d464f
    for (unsigned int l = 0; l < _offsets.size(); ++l)
Packit 0d464f
        for (unsigned int dy = 0; dy < _offsets[l].size(); ++dy)
Packit 0d464f
            for (unsigned int dx = 0; dx < _offsets[l][dy].size(); ++dx)
Packit 0d464f
            {
Packit 0d464f
                table[i].filePos = _offsets[l][dy][dx];
Packit 0d464f
                table[i].dx = dx;
Packit 0d464f
                table[i].dy = dy;
Packit 0d464f
                table[i].l = l;
Packit 0d464f
Packit 0d464f
                ++i;
Packit 0d464f
                
Packit 0d464f
            }
Packit 0d464f
              
Packit 0d464f
    std::sort(table.begin(),table.end());
Packit 0d464f
    
Packit 0d464f
    //
Packit 0d464f
    // write out the values
Packit 0d464f
    //
Packit 0d464f
    
Packit 0d464f
    // pass 1: write out dx and dy, since these are independent of level mode
Packit 0d464f
    
Packit 0d464f
    for(size_t i=0;i
Packit 0d464f
    {
Packit 0d464f
        dx_table[i] = table[i].dx;
Packit 0d464f
        dy_table[i] = table[i].dy;
Packit 0d464f
    }
Packit 0d464f
Packit 0d464f
    // now write out the levels, which depend on the level mode
Packit 0d464f
    
Packit 0d464f
    switch (_mode)
Packit 0d464f
    {
Packit 0d464f
        case ONE_LEVEL:
Packit 0d464f
        {
Packit 0d464f
            for(size_t i=0;i
Packit 0d464f
            {
Packit 0d464f
                lx_table[i] = 0;
Packit 0d464f
                ly_table[i] = 0;               
Packit 0d464f
            }
Packit 0d464f
            break;            
Packit 0d464f
        }
Packit 0d464f
        case MIPMAP_LEVELS:
Packit 0d464f
        {
Packit 0d464f
            for(size_t i=0;i
Packit 0d464f
            {
Packit 0d464f
                lx_table[i]= table[i].l;
Packit 0d464f
                ly_table[i] =table[i].l;               
Packit 0d464f
                
Packit 0d464f
            }
Packit 0d464f
            break;
Packit 0d464f
        }
Packit 0d464f
            
Packit 0d464f
        case RIPMAP_LEVELS:
Packit 0d464f
        {
Packit 0d464f
            for(size_t i=0;i
Packit 0d464f
            {
Packit 0d464f
                lx_table[i]= table[i].l % _numXLevels;
Packit 0d464f
                ly_table[i] = table[i].l / _numXLevels; 
Packit 0d464f
                
Packit 0d464f
            }
Packit 0d464f
            break;
Packit 0d464f
        }
Packit 0d464f
        case NUM_LEVELMODES :
Packit 0d464f
            throw IEX_NAMESPACE::LogicExc("Bad level mode getting tile order");
Packit 0d464f
    }
Packit 0d464f
    
Packit 0d464f
    
Packit 0d464f
    
Packit 0d464f
}
Packit 0d464f
Packit 0d464f
Packit 0d464f
bool
Packit 0d464f
TileOffsets::isEmpty () const
Packit 0d464f
{
Packit 0d464f
    for (unsigned int l = 0; l < _offsets.size(); ++l)
Packit 0d464f
	for (unsigned int dy = 0; dy < _offsets[l].size(); ++dy)
Packit 0d464f
	    for (unsigned int dx = 0; dx < _offsets[l][dy].size(); ++dx)
Packit 0d464f
		if (_offsets[l][dy][dx] != 0)
Packit 0d464f
		    return false;
Packit 0d464f
    return true;
Packit 0d464f
}
Packit 0d464f
Packit 0d464f
Packit 0d464f
bool
Packit 0d464f
TileOffsets::isValidTile (int dx, int dy, int lx, int ly) const
Packit 0d464f
{
Packit 0d464f
    if(lx<0 || ly < 0 || dx<0 || dy < 0) return false;
Packit 0d464f
    switch (_mode)
Packit 0d464f
    {
Packit 0d464f
      case ONE_LEVEL:
Packit 0d464f
Packit 0d464f
        if (lx == 0 &&
Packit 0d464f
	    ly == 0 &&
Packit 0d464f
	    _offsets.size() > 0 &&
Packit 0d464f
            int(_offsets[0].size()) > dy &&
Packit 0d464f
            int(_offsets[0][dy].size()) > dx)
Packit 0d464f
	{
Packit 0d464f
            return true;
Packit 0d464f
	}
Packit 0d464f
Packit 0d464f
        break;
Packit 0d464f
Packit 0d464f
      case MIPMAP_LEVELS:
Packit 0d464f
Packit 0d464f
        if (lx < _numXLevels &&
Packit 0d464f
	    ly < _numYLevels &&
Packit 0d464f
            int(_offsets.size()) > lx &&
Packit 0d464f
            int(_offsets[lx].size()) > dy &&
Packit 0d464f
            int(_offsets[lx][dy].size()) > dx)
Packit 0d464f
	{
Packit 0d464f
            return true;
Packit 0d464f
	}
Packit 0d464f
Packit 0d464f
        break;
Packit 0d464f
Packit 0d464f
      case RIPMAP_LEVELS:
Packit 0d464f
Packit 0d464f
        if (lx < _numXLevels &&
Packit 0d464f
	    ly < _numYLevels &&
Packit 0d464f
	    (_offsets.size() > (size_t) lx+  ly *  (size_t) _numXLevels) &&
Packit 0d464f
            int(_offsets[lx + ly * _numXLevels].size()) > dy &&
Packit 0d464f
            int(_offsets[lx + ly * _numXLevels][dy].size()) > dx)
Packit 0d464f
	{
Packit 0d464f
            return true;
Packit 0d464f
	}
Packit 0d464f
Packit 0d464f
        break;
Packit 0d464f
Packit 0d464f
      default:
Packit 0d464f
Packit 0d464f
        return false;
Packit 0d464f
    }
Packit 0d464f
    
Packit 0d464f
    return false;
Packit 0d464f
}
Packit 0d464f
Packit 0d464f
Packit 0d464f
Int64 &
Packit 0d464f
TileOffsets::operator () (int dx, int dy, int lx, int ly)
Packit 0d464f
{
Packit 0d464f
    //
Packit 0d464f
    // Looks up the value of the tile with tile coordinate (dx, dy)
Packit 0d464f
    // and level number (lx, ly) in the _offsets array, and returns
Packit 0d464f
    // the cooresponding offset.
Packit 0d464f
    //
Packit 0d464f
Packit 0d464f
    switch (_mode)
Packit 0d464f
    {
Packit 0d464f
      case ONE_LEVEL:
Packit 0d464f
Packit 0d464f
        return _offsets[0][dy][dx];
Packit 0d464f
        break;
Packit 0d464f
Packit 0d464f
      case MIPMAP_LEVELS:
Packit 0d464f
Packit 0d464f
        return _offsets[lx][dy][dx];
Packit 0d464f
        break;
Packit 0d464f
Packit 0d464f
      case RIPMAP_LEVELS:
Packit 0d464f
Packit 0d464f
        return _offsets[lx + ly * _numXLevels][dy][dx];
Packit 0d464f
        break;
Packit 0d464f
Packit 0d464f
      default:
Packit 0d464f
Packit 0d464f
        throw IEX_NAMESPACE::ArgExc ("Unknown LevelMode format.");
Packit 0d464f
    }
Packit 0d464f
}
Packit 0d464f
Packit 0d464f
Packit 0d464f
Int64 &
Packit 0d464f
TileOffsets::operator () (int dx, int dy, int l)
Packit 0d464f
{
Packit 0d464f
    return operator () (dx, dy, l, l);
Packit 0d464f
}
Packit 0d464f
Packit 0d464f
Packit 0d464f
const Int64 &
Packit 0d464f
TileOffsets::operator () (int dx, int dy, int lx, int ly) const
Packit 0d464f
{
Packit 0d464f
    //
Packit 0d464f
    // Looks up the value of the tile with tile coordinate (dx, dy)
Packit 0d464f
    // and level number (lx, ly) in the _offsets array, and returns
Packit 0d464f
    // the cooresponding offset.
Packit 0d464f
    //
Packit 0d464f
Packit 0d464f
    switch (_mode)
Packit 0d464f
    {
Packit 0d464f
      case ONE_LEVEL:
Packit 0d464f
Packit 0d464f
        return _offsets[0][dy][dx];
Packit 0d464f
        break;
Packit 0d464f
Packit 0d464f
      case MIPMAP_LEVELS:
Packit 0d464f
Packit 0d464f
        return _offsets[lx][dy][dx];
Packit 0d464f
        break;
Packit 0d464f
Packit 0d464f
      case RIPMAP_LEVELS:
Packit 0d464f
Packit 0d464f
        return _offsets[lx + ly * _numXLevels][dy][dx];
Packit 0d464f
        break;
Packit 0d464f
Packit 0d464f
      default:
Packit 0d464f
Packit 0d464f
        throw IEX_NAMESPACE::ArgExc ("Unknown LevelMode format.");
Packit 0d464f
    }
Packit 0d464f
}
Packit 0d464f
Packit 0d464f
Packit 0d464f
const Int64 &
Packit 0d464f
TileOffsets::operator () (int dx, int dy, int l) const
Packit 0d464f
{
Packit 0d464f
    return operator () (dx, dy, l, l);
Packit 0d464f
}
Packit 0d464f
Packit 0d464f
const std::vector<std::vector<std::vector <Int64> > >&
Packit 0d464f
TileOffsets::getOffsets() const
Packit 0d464f
{
Packit 0d464f
    return _offsets;
Packit 0d464f
}
Packit 0d464f
Packit 0d464f
Packit 0d464f
OPENEXR_IMF_INTERNAL_NAMESPACE_SOURCE_EXIT