573 lines
16 KiB
C++
573 lines
16 KiB
C++
///////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Copyright (c) 2012, Industrial Light & Magic, a division of Lucas
|
|
// Digital Ltd. LLC
|
|
//
|
|
// All rights reserved.
|
|
//
|
|
// Redistribution and use in source and binary forms, with or without
|
|
// modification, are permitted provided that the following conditions are
|
|
// met:
|
|
// * Redistributions of source code must retain the above copyright
|
|
// notice, this list of conditions and the following disclaimer.
|
|
// * Redistributions in binary form must reproduce the above
|
|
// copyright notice, this list of conditions and the following disclaimer
|
|
// in the documentation and/or other materials provided with the
|
|
// distribution.
|
|
// * Neither the name of Industrial Light & Magic nor the names of
|
|
// its contributors may be used to endorse or promote products derived
|
|
// from this software without specific prior written permission.
|
|
//
|
|
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
|
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
|
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
//
|
|
///////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
//
|
|
// Utility program to print an image file's header
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
|
|
#include "ImfNamespace.h"
|
|
#include <ImfMultiPartInputFile.h>
|
|
#include <ImfBoxAttribute.h>
|
|
#include <ImfChannelListAttribute.h>
|
|
#include <ImfChromaticitiesAttribute.h>
|
|
#include <ImfCompressionAttribute.h>
|
|
#include <ImfDoubleAttribute.h>
|
|
#include <ImfEnvmapAttribute.h>
|
|
#include <ImfFloatAttribute.h>
|
|
#include <ImfIntAttribute.h>
|
|
#include <ImfKeyCodeAttribute.h>
|
|
#include <ImfLineOrderAttribute.h>
|
|
#include <ImfMatrixAttribute.h>
|
|
#include <ImfPreviewImageAttribute.h>
|
|
#include <ImfRationalAttribute.h>
|
|
#include <ImfStringAttribute.h>
|
|
#include <ImfStringVectorAttribute.h>
|
|
#include <ImfTileDescriptionAttribute.h>
|
|
#include <ImfTimeCodeAttribute.h>
|
|
#include <ImfVecAttribute.h>
|
|
#include <ImfVersion.h>
|
|
#include <ImfHeader.h>
|
|
|
|
#include <iostream>
|
|
#include <iomanip>
|
|
|
|
|
|
using namespace OPENEXR_IMF_NAMESPACE;
|
|
using namespace std;
|
|
|
|
|
|
void
|
|
printCompression (Compression c)
|
|
{
|
|
switch (c)
|
|
{
|
|
case NO_COMPRESSION:
|
|
cout << "none";
|
|
break;
|
|
|
|
case RLE_COMPRESSION:
|
|
cout << "run-length encoding";
|
|
break;
|
|
|
|
case ZIPS_COMPRESSION:
|
|
cout << "zip, individual scanlines";
|
|
break;
|
|
|
|
case ZIP_COMPRESSION:
|
|
cout << "zip, multi-scanline blocks";
|
|
break;
|
|
|
|
case PIZ_COMPRESSION:
|
|
cout << "piz";
|
|
break;
|
|
|
|
case PXR24_COMPRESSION:
|
|
cout << "pxr24";
|
|
break;
|
|
|
|
case B44_COMPRESSION:
|
|
cout << "b44";
|
|
break;
|
|
|
|
case B44A_COMPRESSION:
|
|
cout << "b44a";
|
|
break;
|
|
|
|
case DWAA_COMPRESSION:
|
|
cout << "dwa, small scanline blocks";
|
|
break;
|
|
|
|
case DWAB_COMPRESSION:
|
|
cout << "dwa, medium scanline blocks";
|
|
break;
|
|
|
|
default:
|
|
cout << int (c);
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
printLineOrder (LineOrder lo)
|
|
{
|
|
switch (lo)
|
|
{
|
|
case INCREASING_Y:
|
|
cout << "increasing y";
|
|
break;
|
|
|
|
case DECREASING_Y:
|
|
cout << "decreasing y";
|
|
break;
|
|
|
|
case RANDOM_Y:
|
|
cout << "random y";
|
|
break;
|
|
|
|
default:
|
|
cout << int (lo);
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
printPixelType (PixelType pt)
|
|
{
|
|
switch (pt)
|
|
{
|
|
case UINT:
|
|
cout << "32-bit unsigned integer";
|
|
break;
|
|
|
|
case HALF:
|
|
cout << "16-bit floating-point";
|
|
break;
|
|
|
|
case FLOAT:
|
|
cout << "32-bit floating-point";
|
|
break;
|
|
|
|
default:
|
|
cout << "type " << int (pt);
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
printLevelMode (LevelMode lm)
|
|
{
|
|
switch (lm)
|
|
{
|
|
case ONE_LEVEL:
|
|
cout << "single level";
|
|
break;
|
|
|
|
case MIPMAP_LEVELS:
|
|
cout << "mip-map";
|
|
break;
|
|
|
|
case RIPMAP_LEVELS:
|
|
cout << "rip-map";
|
|
break;
|
|
|
|
default:
|
|
cout << "level mode " << int (lm);
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
printLevelRoundingMode (LevelRoundingMode lm)
|
|
{
|
|
switch (lm)
|
|
{
|
|
case ROUND_DOWN:
|
|
cout << "down";
|
|
break;
|
|
|
|
case ROUND_UP:
|
|
cout << "up";
|
|
break;
|
|
|
|
default:
|
|
cout << "mode " << int (lm);
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
printTimeCode (TimeCode tc)
|
|
{
|
|
cout << " "
|
|
"time " <<
|
|
setfill ('0') <<
|
|
#ifndef HAVE_COMPLETE_IOMANIP
|
|
setw (2) << tc.hours() << ":" <<
|
|
setw (2) << tc.minutes() << ":" <<
|
|
setw (2) << tc.seconds() << ":" <<
|
|
setw (2) << tc.frame() << "\n" <<
|
|
#else
|
|
setw (2) << right << tc.hours() << ":" <<
|
|
setw (2) << right << tc.minutes() << ":" <<
|
|
setw (2) << right << tc.seconds() << ":" <<
|
|
setw (2) << right << tc.frame() << "\n" <<
|
|
#endif
|
|
setfill (' ') <<
|
|
" "
|
|
"drop frame " << tc.dropFrame() << ", "
|
|
"color frame " << tc.colorFrame() << ", "
|
|
"field/phase " << tc.fieldPhase() << "\n"
|
|
" "
|
|
"bgf0 " << tc.bgf0() << ", "
|
|
"bgf1 " << tc.bgf1() << ", "
|
|
"bgf2 " << tc.bgf2() << "\n"
|
|
" "
|
|
"user data 0x" << hex << tc.userData() << dec;
|
|
}
|
|
|
|
|
|
void
|
|
printEnvmap (Envmap e)
|
|
{
|
|
switch (e)
|
|
{
|
|
case ENVMAP_LATLONG:
|
|
cout << "latitude-longitude map";
|
|
break;
|
|
|
|
case ENVMAP_CUBE:
|
|
cout << "cube-face map";
|
|
break;
|
|
|
|
default:
|
|
cout << "map type " << int (e);
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
printChannelList (const ChannelList &cl)
|
|
{
|
|
for (ChannelList::ConstIterator i = cl.begin(); i != cl.end(); ++i)
|
|
{
|
|
cout << "\n " << i.name() << ", ";
|
|
|
|
printPixelType (i.channel().type);
|
|
|
|
cout << ", sampling " <<
|
|
i.channel().xSampling << " " <<
|
|
i.channel().ySampling;
|
|
|
|
if (i.channel().pLinear)
|
|
cout << ", plinear";
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
printInfo (const char fileName[])
|
|
{
|
|
MultiPartInputFile in (fileName);
|
|
int parts = in.parts();
|
|
|
|
//
|
|
// Check to see if any parts are incomplete
|
|
//
|
|
|
|
bool fileComplete = true;
|
|
|
|
for (int i = 0; i < parts && fileComplete; ++i)
|
|
if (!in.partComplete (i))
|
|
fileComplete = false;
|
|
|
|
//
|
|
// Print file name and file format version
|
|
//
|
|
|
|
cout << "\nfile " << fileName <<
|
|
(fileComplete? "": " (incomplete)") <<
|
|
":\n\n";
|
|
|
|
cout << "file format version: " <<
|
|
getVersion (in.version()) << ", "
|
|
"flags 0x" <<
|
|
setbase (16) << getFlags (in.version()) << setbase (10) << "\n";
|
|
|
|
//
|
|
// Print the header of every part in the file
|
|
//
|
|
|
|
for (int p = 0; p < parts ; ++p)
|
|
{
|
|
const Header & h = in.header (p);
|
|
|
|
if (parts != 1)
|
|
{
|
|
cout << "\n\n part " << p <<
|
|
(in.partComplete (p)? "": " (incomplete)") <<
|
|
":\n";
|
|
|
|
}
|
|
|
|
for (Header::ConstIterator i = h.begin(); i != h.end(); ++i)
|
|
{
|
|
const Attribute *a = &i.attribute();
|
|
cout << i.name() << " (type " << a->typeName() << ")";
|
|
|
|
if (const Box2iAttribute *ta =
|
|
dynamic_cast <const Box2iAttribute *> (a))
|
|
{
|
|
cout << ": " << ta->value().min << " - " << ta->value().max;
|
|
}
|
|
|
|
else if (const Box2fAttribute *ta =
|
|
dynamic_cast <const Box2fAttribute *> (a))
|
|
{
|
|
cout << ": " << ta->value().min << " - " << ta->value().max;
|
|
}
|
|
else if (const ChannelListAttribute *ta =
|
|
dynamic_cast <const ChannelListAttribute *> (a))
|
|
{
|
|
cout << ":";
|
|
printChannelList (ta->value());
|
|
}
|
|
else if (const ChromaticitiesAttribute *ta =
|
|
dynamic_cast <const ChromaticitiesAttribute *> (a))
|
|
{
|
|
cout << ":\n"
|
|
" red " << ta->value().red << "\n"
|
|
" green " << ta->value().green << "\n"
|
|
" blue " << ta->value().blue << "\n"
|
|
" white " << ta->value().white;
|
|
}
|
|
else if (const CompressionAttribute *ta =
|
|
dynamic_cast <const CompressionAttribute *> (a))
|
|
{
|
|
cout << ": ";
|
|
printCompression (ta->value());
|
|
}
|
|
else if (const DoubleAttribute *ta =
|
|
dynamic_cast <const DoubleAttribute *> (a))
|
|
{
|
|
cout << ": " << ta->value();
|
|
}
|
|
else if (const EnvmapAttribute *ta =
|
|
dynamic_cast <const EnvmapAttribute *> (a))
|
|
{
|
|
cout << ": ";
|
|
printEnvmap (ta->value());
|
|
}
|
|
else if (const FloatAttribute *ta =
|
|
dynamic_cast <const FloatAttribute *> (a))
|
|
{
|
|
cout << ": " << ta->value();
|
|
}
|
|
else if (const IntAttribute *ta =
|
|
dynamic_cast <const IntAttribute *> (a))
|
|
{
|
|
cout << ": " << ta->value();
|
|
}
|
|
else if (const KeyCodeAttribute *ta =
|
|
dynamic_cast <const KeyCodeAttribute *> (a))
|
|
{
|
|
cout << ":\n"
|
|
" film manufacturer code " <<
|
|
ta->value().filmMfcCode() << "\n"
|
|
" film type code " <<
|
|
ta->value().filmType() << "\n"
|
|
" prefix " <<
|
|
ta->value().prefix() << "\n"
|
|
" count " <<
|
|
ta->value().count() << "\n"
|
|
" perf offset " <<
|
|
ta->value().perfOffset() << "\n"
|
|
" perfs per frame " <<
|
|
ta->value().perfsPerFrame() << "\n"
|
|
" perfs per count " <<
|
|
ta->value().perfsPerCount();
|
|
}
|
|
else if (const LineOrderAttribute *ta =
|
|
dynamic_cast <const LineOrderAttribute *> (a))
|
|
{
|
|
cout << ": ";
|
|
printLineOrder (ta->value());
|
|
}
|
|
else if (const M33fAttribute *ta =
|
|
dynamic_cast <const M33fAttribute *> (a))
|
|
{
|
|
cout << ":\n"
|
|
" (" <<
|
|
ta->value()[0][0] << " " <<
|
|
ta->value()[0][1] << " " <<
|
|
ta->value()[0][2] << "\n " <<
|
|
ta->value()[1][0] << " " <<
|
|
ta->value()[1][1] << " " <<
|
|
ta->value()[1][2] << "\n " <<
|
|
ta->value()[2][0] << " " <<
|
|
ta->value()[2][1] << " " <<
|
|
ta->value()[2][2] << ")";
|
|
}
|
|
else if (const M44fAttribute *ta =
|
|
dynamic_cast <const M44fAttribute *> (a))
|
|
{
|
|
cout << ":\n"
|
|
" (" <<
|
|
ta->value()[0][0] << " " <<
|
|
ta->value()[0][1] << " " <<
|
|
ta->value()[0][2] << " " <<
|
|
ta->value()[0][3] << "\n " <<
|
|
ta->value()[1][0] << " " <<
|
|
ta->value()[1][1] << " " <<
|
|
ta->value()[1][2] << " " <<
|
|
ta->value()[1][3] << "\n " <<
|
|
ta->value()[2][0] << " " <<
|
|
ta->value()[2][1] << " " <<
|
|
ta->value()[2][2] << " " <<
|
|
ta->value()[2][3] << "\n " <<
|
|
ta->value()[3][0] << " " <<
|
|
ta->value()[3][1] << " " <<
|
|
ta->value()[3][2] << " " <<
|
|
ta->value()[3][3] << ")";
|
|
}
|
|
else if (const PreviewImageAttribute *ta =
|
|
dynamic_cast <const PreviewImageAttribute *> (a))
|
|
{
|
|
cout << ": " <<
|
|
ta->value().width() << " by " <<
|
|
ta->value().height() << " pixels";
|
|
}
|
|
else if (const StringAttribute *ta =
|
|
dynamic_cast <const StringAttribute *> (a))
|
|
{
|
|
cout << ": \"" << ta->value() << "\"";
|
|
}
|
|
else if (const StringVectorAttribute * ta =
|
|
dynamic_cast<const StringVectorAttribute *>(a))
|
|
{
|
|
cout << ":";
|
|
|
|
for (StringVector::const_iterator i = ta->value().begin();
|
|
i != ta->value().end();
|
|
++i)
|
|
{
|
|
cout << "\n \"" << *i << "\"";
|
|
}
|
|
}
|
|
else if (const RationalAttribute *ta =
|
|
dynamic_cast <const RationalAttribute *> (a))
|
|
{
|
|
cout << ": " << ta->value().n << "/" << ta->value().d <<
|
|
" (" << double (ta->value()) << ")";
|
|
}
|
|
else if (const TileDescriptionAttribute *ta =
|
|
dynamic_cast <const TileDescriptionAttribute *> (a))
|
|
{
|
|
cout << ":\n ";
|
|
|
|
printLevelMode (ta->value().mode);
|
|
|
|
cout << "\n tile size " <<
|
|
ta->value().xSize << " by " <<
|
|
ta->value().ySize << " pixels";
|
|
|
|
if (ta->value().mode != ONE_LEVEL)
|
|
{
|
|
cout << "\n level sizes rounded ";
|
|
printLevelRoundingMode (ta->value().roundingMode);
|
|
}
|
|
}
|
|
else if (const TimeCodeAttribute *ta =
|
|
dynamic_cast <const TimeCodeAttribute *> (a))
|
|
{
|
|
cout << ":\n";
|
|
printTimeCode (ta->value());
|
|
}
|
|
else if (const V2iAttribute *ta =
|
|
dynamic_cast <const V2iAttribute *> (a))
|
|
{
|
|
cout << ": " << ta->value();
|
|
}
|
|
else if (const V2fAttribute *ta =
|
|
dynamic_cast <const V2fAttribute *> (a))
|
|
{
|
|
cout << ": " << ta->value();
|
|
}
|
|
else if (const V3iAttribute *ta =
|
|
dynamic_cast <const V3iAttribute *> (a))
|
|
{
|
|
cout << ": " << ta->value();
|
|
}
|
|
else if (const V3fAttribute *ta =
|
|
dynamic_cast <const V3fAttribute *> (a))
|
|
{
|
|
cout << ": " << ta->value();
|
|
}
|
|
|
|
cout << '\n';
|
|
}
|
|
}
|
|
|
|
cout << endl;
|
|
}
|
|
|
|
|
|
void
|
|
usageMessage (const char argv0[])
|
|
{
|
|
std::cerr << "usage: " << argv0 << " imagefile [imagefile ...]\n";
|
|
}
|
|
|
|
|
|
int
|
|
main(int argc, char **argv)
|
|
{
|
|
if (argc < 2)
|
|
{
|
|
usageMessage (argv[0]);
|
|
return 1;
|
|
}
|
|
|
|
for (int i = 1; i < argc; ++i)
|
|
{
|
|
if (!strcmp (argv[i], "-h"))
|
|
{
|
|
usageMessage (argv[0]);
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
try
|
|
{
|
|
for (int i = 1; i < argc; ++i)
|
|
printInfo (argv[i]);
|
|
|
|
return 0;
|
|
}
|
|
catch (const std::exception &e)
|
|
{
|
|
std::cerr << e.what() << std::endl;
|
|
return 1;
|
|
}
|
|
|
|
}
|