Blame IlmImfTest/testOptimizedInterleavePatterns.cpp

Packit Service 6754ca
///////////////////////////////////////////////////////////////////////////
Packit Service 6754ca
//
Packit Service 6754ca
// Copyright (c) 2013, Weta Digital Ltd
Packit Service 6754ca
//
Packit Service 6754ca
// All rights reserved.
Packit Service 6754ca
//
Packit Service 6754ca
// Redistribution and use in source and binary forms, with or without
Packit Service 6754ca
// modification, are permitted provided that the following conditions are
Packit Service 6754ca
// met:
Packit Service 6754ca
// *       Redistributions of source code must retain the above copyright
Packit Service 6754ca
// notice, this list of conditions and the following disclaimer.
Packit Service 6754ca
// *       Redistributions in binary form must reproduce the above
Packit Service 6754ca
// copyright notice, this list of conditions and the following disclaimer
Packit Service 6754ca
// in the documentation and/or other materials provided with the
Packit Service 6754ca
// distribution.
Packit Service 6754ca
// *       Neither the name of Industrial Light & Magic nor the names of
Packit Service 6754ca
// its contributors may be used to endorse or promote products derived
Packit Service 6754ca
// from this software without specific prior written permission.
Packit Service 6754ca
//
Packit Service 6754ca
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
Packit Service 6754ca
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
Packit Service 6754ca
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
Packit Service 6754ca
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
Packit Service 6754ca
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
Packit Service 6754ca
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
Packit Service 6754ca
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
Packit Service 6754ca
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
Packit Service 6754ca
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
Packit Service 6754ca
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
Packit Service 6754ca
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
Packit Service 6754ca
//
Packit Service 6754ca
///////////////////////////////////////////////////////////////////////////
Packit Service 6754ca
Packit Service 6754ca
#include "ImfInputFile.h"
Packit Service 6754ca
#include <stdlib.h>
Packit Service 6754ca
#include <vector>
Packit Service 6754ca
#include "ImfChannelList.h"
Packit Service 6754ca
#include "ImfOutputFile.h"
Packit Service 6754ca
#include "ImfCompression.h"
Packit Service 6754ca
#include "ImfStandardAttributes.h"
Packit Service 6754ca
#include <algorithm>
Packit Service 6754ca
#include <iostream>
Packit Service 6754ca
#include <assert.h>
Packit Service 6754ca
#include <IlmThread.h>
Packit Service 6754ca
#include <ImathBox.h>
Packit Service 6754ca
Packit Service 6754ca
#include "tmpDir.h"
Packit Service 6754ca
Packit Service 6754ca
namespace IMF = OPENEXR_IMF_NAMESPACE;
Packit Service 6754ca
using namespace IMF;
Packit Service 6754ca
using namespace std;
Packit Service 6754ca
using namespace IMATH_NAMESPACE;
Packit Service 6754ca
using namespace ILMTHREAD_NAMESPACE;
Packit Service 6754ca
Packit Service 6754ca
Packit Service 6754ca
namespace
Packit Service 6754ca
{
Packit Service 6754ca
Packit Service 6754ca
using OPENEXR_IMF_NAMESPACE::UINT;
Packit Service 6754ca
using OPENEXR_IMF_NAMESPACE::FLOAT;
Packit Service 6754ca
Packit Service 6754ca
std::string filename;
Packit Service 6754ca
Packit Service 6754ca
vector<char> writingBuffer; // buffer as file was written
Packit Service 6754ca
vector<char> readingBuffer; // buffer containing new image (and filled channels?)
Packit Service 6754ca
vector<char> preReadBuffer; // buffer as it was before reading - unread, unfilled channels should be unchanged
Packit Service 6754ca
Packit Service 6754ca
int gOptimisedReads = 0;
Packit Service 6754ca
int gSuccesses = 0;
Packit Service 6754ca
int gFailures = 0;
Packit Service 6754ca
Packit Service 6754ca
Packit Service 6754ca
//
Packit Service 6754ca
// @todo Needs a description of what this is used for.
Packit Service 6754ca
//
Packit Service 6754ca
//
Packit Service 6754ca
struct Schema
Packit Service 6754ca
{
Packit Service 6754ca
    const char*         _name;     // name of this scheme
Packit Service 6754ca
    const char* const*  _active;   // channels to be read
Packit Service 6754ca
    const char* const * _passive;  // channels to be ignored (keep in buffer passed to inputfile, should not be overwritten)
Packit Service 6754ca
    int                 _banks;
Packit Service 6754ca
    const char* const * _views;    // list of views to write, or NULL
Packit Service 6754ca
    const PixelType*    _types;    // NULL for all HALF, otherwise per-channel type
Packit Service 6754ca
    
Packit Service 6754ca
    vector<string> views() const
Packit Service 6754ca
    {
Packit Service 6754ca
        const char * const* v = _views;
Packit Service 6754ca
        vector<string> svec;
Packit Service 6754ca
        while(*v!=NULL)
Packit Service 6754ca
        {
Packit Service 6754ca
            svec.push_back (*v);
Packit Service 6754ca
            v++;
Packit Service 6754ca
        }
Packit Service 6754ca
        return svec;
Packit Service 6754ca
    } 
Packit Service 6754ca
};
Packit Service 6754ca
Packit Service 6754ca
Packit Service 6754ca
const char * rgb[] = {"R","G","B",NULL};
Packit Service 6754ca
const char * rgba[] = {"R","G","B","A",NULL};
Packit Service 6754ca
const char * bgr[] = {"B","G","R",NULL};
Packit Service 6754ca
const char * abgr[] = {"A","B","G","R",NULL};
Packit Service 6754ca
const char * alpha[] = {"A",NULL};
Packit Service 6754ca
const char * redalpha[] = {"R","A",NULL};
Packit Service 6754ca
const char * rgbrightrgb[] = {"R","G","B","right.R","right.G","right.B",NULL};
Packit Service 6754ca
const char * rgbleftrgb[] = {"R","G","B","left.R","left.G","left.B",NULL};
Packit Service 6754ca
const char * rgbarightrgba[] = {"R","G","B","A","right.R","right.G","right.B","right.A",NULL};
Packit Service 6754ca
const char * rgbaleftrgba[] = {"R","G","B","A","left.R","left.G","left.B","left.A",NULL};
Packit Service 6754ca
const char * rgbrightrgba[] = {"R","G","B","right.R","right.G","right.B","right.A",NULL};
Packit Service 6754ca
const char * rgbleftrgba[] = {"R","G","B","left.R","left.G","left.B","left.A",NULL};
Packit Service 6754ca
const char * rgbarightrgb[] = {"R","G","B","A","right.R","right.G","right.B",NULL};
Packit Service 6754ca
const char * rgbaleftrgb[] = {"R","G","B","A","left.R","left.G","left.B",NULL};
Packit Service 6754ca
const char * rightrgba[] = {"right.R","right.G","right.B","right.A",NULL};
Packit Service 6754ca
const char * leftrgba[] = {"left.R","left.G","left.B","left.A",NULL};
Packit Service 6754ca
const char * rightrgb[] = {"right.R","right.G","right.B",NULL};
Packit Service 6754ca
const char * leftrgb[] = {"left.R","left.G","left.B",NULL};
Packit Service 6754ca
const char * threeview[] ={"R","G","B","A","left.R","left.G","left.B","left.A","right.R","right.G","right.B","right.A",NULL};
Packit Service 6754ca
const char * trees[] = {"rimu","pohutukawa","manuka","kauri",NULL};
Packit Service 6754ca
const char * treesandbirds[]= {"kiwi","rimu","pohutukawa","kakapu","kauri","manuka","moa","fantail",NULL};
Packit Service 6754ca
Packit Service 6754ca
const char * lefthero[] = {"left","right",NULL};
Packit Service 6754ca
const char * righthero[] = {"right","left",NULL};
Packit Service 6754ca
const char * centrehero[] = {"centre","left","right",NULL};
Packit Service 6754ca
Packit Service 6754ca
const PixelType four_floats[] = {IMF::FLOAT,IMF::FLOAT,IMF::FLOAT,IMF::FLOAT};
Packit Service 6754ca
const PixelType hhhfff[] = {IMF::HALF,IMF::HALF,IMF::HALF,IMF::FLOAT,IMF::FLOAT,IMF::FLOAT};
Packit Service 6754ca
const PixelType hhhhffff[] = {IMF::HALF,IMF::HALF,IMF::HALF,IMF::HALF,IMF::FLOAT,IMF::FLOAT,IMF::FLOAT,IMF::FLOAT};
Packit Service 6754ca
Packit Service 6754ca
Schema Schemes[] =
Packit Service 6754ca
{
Packit Service 6754ca
        {"RGBHalf"              ,rgb           ,NULL      ,1 ,NULL       ,NULL       },
Packit Service 6754ca
        {"RGBAHalf"             ,rgba          ,NULL      ,1 ,NULL       ,NULL       },
Packit Service 6754ca
        {"ABGRHalf"             ,abgr          ,NULL      ,1 ,NULL       ,NULL       },
Packit Service 6754ca
        {"RGBFloat"             ,rgb           ,NULL      ,1 ,NULL       ,four_floats},
Packit Service 6754ca
        {"BGRHalf"              ,bgr           ,NULL      ,1 ,NULL       ,NULL       },
Packit Service 6754ca
        {"RGBLeftRGB"           ,rgbleftrgb    ,NULL      ,1 ,righthero  ,NULL       },
Packit Service 6754ca
        {"RGBRightRGB"          ,rgbrightrgb   ,NULL      ,1 ,lefthero   ,NULL       },
Packit Service 6754ca
        {"RGBALeftRGBA"         ,rgbaleftrgba  ,NULL      ,1 ,righthero  ,NULL       },
Packit Service 6754ca
        {"RGBARightRGBA"        ,rgbarightrgba ,NULL      ,1 ,lefthero   ,NULL       },
Packit Service 6754ca
        {"LeftRGB"              ,leftrgb       ,NULL      ,1 ,NULL       ,NULL       },
Packit Service 6754ca
        {"RightRGB"             ,rightrgb      ,NULL      ,1 ,NULL       ,NULL       },
Packit Service 6754ca
        {"LeftRGBA"             ,leftrgba      ,NULL      ,1 ,NULL       ,NULL       },
Packit Service 6754ca
        {"RightRGBA"            ,rightrgba     ,NULL      ,1 ,NULL       ,NULL       },
Packit Service 6754ca
        {"TripleView"           ,threeview     ,NULL      ,1 ,centrehero ,NULL       },
Packit Service 6754ca
        {"Trees"                ,trees         ,NULL      ,1 ,NULL       ,NULL       },
Packit Service 6754ca
        {"TreesAndBirds"        ,treesandbirds ,NULL      ,1 ,NULL       ,NULL       },
Packit Service 6754ca
        {"RGBLeftRGBA"          ,rgbleftrgba   ,NULL      ,1 ,righthero  ,NULL       },
Packit Service 6754ca
        {"RGBRightRGBA"         ,rgbrightrgba  ,NULL      ,1 ,lefthero   ,NULL       },
Packit Service 6754ca
        {"RGBALeftRGB"          ,rgbaleftrgb   ,NULL      ,1 ,righthero  ,NULL       },
Packit Service 6754ca
        {"RGBARightRGB"         ,rgbarightrgb  ,NULL      ,1 ,lefthero   ,NULL       },
Packit Service 6754ca
        {"TwinRGBLeftRGB"       ,rgbleftrgb    ,NULL      ,2 ,righthero  ,NULL       },
Packit Service 6754ca
        {"TwinRGBRightRGB"      ,rgbrightrgb   ,NULL      ,2 ,lefthero   ,NULL       },
Packit Service 6754ca
        {"TwinRGBALeftRGBA"     ,rgbaleftrgba  ,NULL      ,2 ,righthero  ,NULL       },
Packit Service 6754ca
        {"TwinRGBARightRGBA"    ,rgbarightrgba ,NULL     , 2 ,lefthero   ,NULL       },
Packit Service 6754ca
        {"TripleTripleView"     ,threeview     ,NULL      ,3 ,centrehero ,NULL       },
Packit Service 6754ca
        {"Alpha"                ,alpha         ,NULL      ,1 ,NULL       ,NULL       },
Packit Service 6754ca
        {"RedAlpha"             ,redalpha      ,NULL      ,1 ,NULL       ,NULL       },
Packit Service 6754ca
        {"RG+BA"                ,rgba          ,NULL      ,2 ,NULL       ,NULL       },//interleave only RG, then BA
Packit Service 6754ca
        {"RGBpassiveA"          ,rgb           ,alpha     ,1 ,NULL       ,NULL       },//interleave only RG, then BA
Packit Service 6754ca
        {"RGBpassiveleftRGB"    ,rgb           ,leftrgb   ,1 ,NULL       ,NULL       },
Packit Service 6754ca
        {"RGBFloatA"            ,rgba          ,NULL      ,1 ,NULL       ,hhhfff     },
Packit Service 6754ca
        {"RGBFloatLeftRGB"      ,rgbleftrgb    ,NULL      ,1 ,righthero  ,hhhfff     },
Packit Service 6754ca
        {"RGBAFloatLeftRGBA"    ,rgbaleftrgba  ,NULL      ,1 ,righthero  ,hhhhffff   },
Packit Service 6754ca
        {"RGBApassiverightRGBA" ,rgba          ,rightrgba ,1 ,NULL       ,NULL       },
Packit Service 6754ca
        {"BanksOfTreesAndBirds" ,treesandbirds ,NULL      ,2 ,NULL       ,NULL       },
Packit Service 6754ca
        {NULL,NULL,NULL,0,NULL,NULL}
Packit Service 6754ca
};
Packit Service 6754ca
Packit Service 6754ca
Packit Service 6754ca
Packit Service 6754ca
bool compare(const FrameBuffer& asRead,
Packit Service 6754ca
             const FrameBuffer& asWritten,
Packit Service 6754ca
             const Box2i& dataWindow,
Packit Service 6754ca
             bool nonfatal
Packit Service 6754ca
            )
Packit Service 6754ca
{
Packit Service 6754ca
    for (FrameBuffer::ConstIterator i =asRead.begin();i!=asRead.end();i++)
Packit Service 6754ca
    {
Packit Service 6754ca
        FrameBuffer::ConstIterator p = asWritten.find(i.name());
Packit Service 6754ca
        for (int y=dataWindow.min.y; y<= dataWindow.max.y; y++)
Packit Service 6754ca
        {
Packit Service 6754ca
            for (int x = dataWindow.min.x; x <= dataWindow.max.x; x++)
Packit Service 6754ca
                 
Packit Service 6754ca
            {
Packit Service 6754ca
                char * ptr = (i.slice().base+i.slice().yStride*y +i.slice().xStride*x);
Packit Service 6754ca
                half readHalf;
Packit Service 6754ca
                switch (i.slice().type)
Packit Service 6754ca
                {
Packit Service 6754ca
                    case IMF::FLOAT :
Packit Service 6754ca
                        readHalf =  half(*(float*) ptr);
Packit Service 6754ca
                        break;
Packit Service 6754ca
                    case IMF::HALF :
Packit Service 6754ca
                        readHalf = half(*(half*) ptr);
Packit Service 6754ca
                        break;
Packit Service 6754ca
                    case IMF::UINT :
Packit Service 6754ca
                        continue; // can't very well check this
Packit Service 6754ca
                    default :
Packit Service 6754ca
                        cout << "don't know about that\n";
Packit Service 6754ca
                        exit(1);
Packit Service 6754ca
                }
Packit Service 6754ca
                
Packit Service 6754ca
                half writtenHalf;
Packit Service 6754ca
Packit Service 6754ca
                if (p!=asWritten.end())
Packit Service 6754ca
                {
Packit Service 6754ca
                    char * ptr = p.slice().base+p.slice().yStride*y +
Packit Service 6754ca
                                 p.slice().xStride*x;
Packit Service 6754ca
                    switch (p.slice().type)
Packit Service 6754ca
                    {
Packit Service 6754ca
                    case IMF::FLOAT :
Packit Service 6754ca
                        writtenHalf = half(*(float*) ptr);
Packit Service 6754ca
                        break;
Packit Service 6754ca
                    case IMF::HALF :
Packit Service 6754ca
                        writtenHalf = half(*(half*) ptr);
Packit Service 6754ca
                        break;
Packit Service 6754ca
                    case IMF::UINT :
Packit Service 6754ca
                        continue;
Packit Service 6754ca
                    default :
Packit Service 6754ca
                        cout << "don't know about that\n";
Packit Service 6754ca
                        exit(1);
Packit Service 6754ca
                    }
Packit Service 6754ca
                }
Packit Service 6754ca
                else
Packit Service 6754ca
                {
Packit Service 6754ca
                    writtenHalf=half(i.slice().fillValue);
Packit Service 6754ca
                }
Packit Service 6754ca
Packit Service 6754ca
                if (writtenHalf.bits()!=readHalf.bits())
Packit Service 6754ca
                {
Packit Service 6754ca
                    if (nonfatal)
Packit Service 6754ca
                    {
Packit Service 6754ca
                        return false;
Packit Service 6754ca
                    }
Packit Service 6754ca
                    else
Packit Service 6754ca
                    {
Packit Service 6754ca
                        cout << "\n\nerror reading back channel " << i.name() << " pixel " << x << ',' << y << " got " << readHalf << " expected " << writtenHalf << endl;
Packit Service 6754ca
                        assert(writtenHalf.bits()==readHalf.bits());
Packit Service 6754ca
                        exit(1);
Packit Service 6754ca
                    }
Packit Service 6754ca
                }             
Packit Service 6754ca
            }
Packit Service 6754ca
Packit Service 6754ca
        }
Packit Service 6754ca
    }
Packit Service 6754ca
    return true;
Packit Service 6754ca
}
Packit Service 6754ca
Packit Service 6754ca
//
Packit Service 6754ca
// allocate readingBuffer or writingBuffer, setting up a framebuffer to point to the right thing
Packit Service 6754ca
//
Packit Service 6754ca
ChannelList
Packit Service 6754ca
setupBuffer (const Header& hdr,       // header to grab datawindow from
Packit Service 6754ca
             const char * const *channels, // NULL terminated list of channels to write
Packit Service 6754ca
             const char * const *passivechannels, // NULL terminated list of channels to write
Packit Service 6754ca
             const PixelType* pt,     // type of each channel, or NULL for all HALF
Packit Service 6754ca
             FrameBuffer& buf,        // buffer to fill with pointers to channel
Packit Service 6754ca
             FrameBuffer& prereadbuf, // channels which aren't being read - indexes into the preread buffer
Packit Service 6754ca
             FrameBuffer& postreadbuf, // channels which aren't being read - indexes into the postread buffer
Packit Service 6754ca
             int banks,                    // number of banks - channels within each bank are interleaved, banks are scanline interleaved
Packit Service 6754ca
             bool writing                  // true if should allocate
Packit Service 6754ca
            )
Packit Service 6754ca
{
Packit Service 6754ca
    Box2i dw = hdr.dataWindow();
Packit Service 6754ca
Packit Service 6754ca
    //
Packit Service 6754ca
    // how many channels in total
Packit Service 6754ca
    //
Packit Service 6754ca
    int activechans = 0;
Packit Service 6754ca
    int bytes_per_pixel =0;
Packit Service 6754ca
    
Packit Service 6754ca
    while (channels[activechans]!=NULL)
Packit Service 6754ca
    {
Packit Service 6754ca
        if (pt==NULL)
Packit Service 6754ca
        {
Packit Service 6754ca
            bytes_per_pixel+=2;
Packit Service 6754ca
        }
Packit Service 6754ca
        else
Packit Service 6754ca
        {
Packit Service 6754ca
            switch (pt[activechans])
Packit Service 6754ca
            {
Packit Service 6754ca
                case IMF::HALF : bytes_per_pixel+=2;break;
Packit Service 6754ca
                case IMF::FLOAT : case IMF::UINT : bytes_per_pixel+=4;break;
Packit Service 6754ca
                default :
Packit Service 6754ca
                    cout << "Unexpected PixelType?\n";
Packit Service 6754ca
                    exit(1);
Packit Service 6754ca
            }
Packit Service 6754ca
        }
Packit Service 6754ca
        activechans++;
Packit Service 6754ca
    }
Packit Service 6754ca
Packit Service 6754ca
    
Packit Service 6754ca
    int passivechans=0;
Packit Service 6754ca
    while (passivechannels!=NULL && passivechannels[passivechans]!=NULL)
Packit Service 6754ca
    {
Packit Service 6754ca
        if (pt==NULL)
Packit Service 6754ca
        {
Packit Service 6754ca
            bytes_per_pixel+=2;
Packit Service 6754ca
        }
Packit Service 6754ca
        else
Packit Service 6754ca
        {
Packit Service 6754ca
            switch (pt[passivechans+activechans])
Packit Service 6754ca
            {
Packit Service 6754ca
                case IMF::HALF : bytes_per_pixel+=2;break;
Packit Service 6754ca
                case IMF::FLOAT : case IMF::UINT : bytes_per_pixel+=4;break;
Packit Service 6754ca
                default :
Packit Service 6754ca
                    cout << "Unexpected PixelType?\n";
Packit Service 6754ca
                    exit(1);
Packit Service 6754ca
            }
Packit Service 6754ca
        }
Packit Service 6754ca
        passivechans++;
Packit Service 6754ca
    }
Packit Service 6754ca
Packit Service 6754ca
   int chans = activechans+passivechans;
Packit Service 6754ca
Packit Service 6754ca
    
Packit Service 6754ca
    int bytes_per_bank = bytes_per_pixel/banks;
Packit Service 6754ca
    
Packit Service 6754ca
    int samples = (hdr.dataWindow().max.x+1-hdr.dataWindow().min.x)*
Packit Service 6754ca
                  (hdr.dataWindow().max.y+1-hdr.dataWindow().min.y)*chans;
Packit Service 6754ca
Packit Service 6754ca
    int size = samples*bytes_per_pixel;
Packit Service 6754ca
Packit Service 6754ca
    
Packit Service 6754ca
    if (writing)
Packit Service 6754ca
    {
Packit Service 6754ca
        writingBuffer.resize(size);
Packit Service 6754ca
    }
Packit Service 6754ca
    else
Packit Service 6754ca
    {
Packit Service 6754ca
        readingBuffer.resize(size);
Packit Service 6754ca
    }
Packit Service 6754ca
   
Packit Service 6754ca
     const char * write_ptr = writing ? &writingBuffer[0] : &readingBuffer[0];
Packit Service 6754ca
     // fill with random halfs, casting to floats for float channels
Packit Service 6754ca
     int chan=0;
Packit Service 6754ca
     for (int i=0;i
Packit Service 6754ca
     {
Packit Service 6754ca
         unsigned short int values = (unsigned short int) floor((double(rand())/double(RAND_MAX))*65535.0);
Packit Service 6754ca
         half v;
Packit Service 6754ca
         v.setBits(values);
Packit Service 6754ca
         if (pt==NULL || pt[chan]==IMF::HALF)
Packit Service 6754ca
         {
Packit Service 6754ca
             *(half*)write_ptr = half(v);
Packit Service 6754ca
             write_ptr+=2;
Packit Service 6754ca
         }
Packit Service 6754ca
         else
Packit Service 6754ca
         {
Packit Service 6754ca
             *(float*)write_ptr = float(v);
Packit Service 6754ca
             write_ptr+=4;
Packit Service 6754ca
         }
Packit Service 6754ca
         chan++;
Packit Service 6754ca
         if (chan==chans)
Packit Service 6754ca
         {
Packit Service 6754ca
             chan=0;
Packit Service 6754ca
         }
Packit Service 6754ca
     
Packit Service 6754ca
     }
Packit Service 6754ca
Packit Service 6754ca
     if (!writing)
Packit Service 6754ca
     {
Packit Service 6754ca
         //take a copy of the buffer as it was before being read
Packit Service 6754ca
         preReadBuffer = readingBuffer;
Packit Service 6754ca
     }
Packit Service 6754ca
     
Packit Service 6754ca
    char* offset=NULL;
Packit Service 6754ca
Packit Service 6754ca
    ChannelList chanlist;
Packit Service 6754ca
Packit Service 6754ca
    int bytes_per_row = bytes_per_pixel*(dw.max.x+1-dw.min.x);
Packit Service 6754ca
    int bytes_per_bank_row = bytes_per_row/banks;
Packit Service 6754ca
    
Packit Service 6754ca
    int first_pixel_index = bytes_per_row*dw.min.y+bytes_per_bank*dw.min.x;
Packit Service 6754ca
    
Packit Service 6754ca
    for (int i=0;i
Packit Service 6754ca
    {
Packit Service 6754ca
        PixelType type = pt==NULL ? IMF::HALF : pt[i];
Packit Service 6754ca
        if (i
Packit Service 6754ca
        {
Packit Service 6754ca
            chanlist.insert(channels[i],type);
Packit Service 6754ca
        }
Packit Service 6754ca
Packit Service 6754ca
Packit Service 6754ca
        if (i % (chans/banks) ==0)
Packit Service 6754ca
        {
Packit Service 6754ca
            //
Packit Service 6754ca
            // set offset pointer to beginning of bank
Packit Service 6754ca
            //
Packit Service 6754ca
Packit Service 6754ca
            int bank = i / (chans/banks);
Packit Service 6754ca
            offset = (writing ? &writingBuffer[0] :
Packit Service 6754ca
                                &readingBuffer[0]) + bank*bytes_per_bank_row - first_pixel_index;
Packit Service 6754ca
        }
Packit Service 6754ca
        
Packit Service 6754ca
        if (i
Packit Service 6754ca
        {
Packit Service 6754ca
            
Packit Service 6754ca
            buf.insert (channels[i],
Packit Service 6754ca
                        Slice (type,
Packit Service 6754ca
                               offset,
Packit Service 6754ca
                               bytes_per_bank,
Packit Service 6754ca
                               bytes_per_row,
Packit Service 6754ca
                               1,1,100+i));
Packit Service 6754ca
        }
Packit Service 6754ca
        else
Packit Service 6754ca
        {
Packit Service 6754ca
            if (!writing)
Packit Service 6754ca
            {
Packit Service 6754ca
                
Packit Service 6754ca
                postreadbuf.insert (passivechannels[i-activechans],
Packit Service 6754ca
                                    Slice (type,
Packit Service 6754ca
                                           offset,
Packit Service 6754ca
                                           bytes_per_bank,
Packit Service 6754ca
                                           bytes_per_row,
Packit Service 6754ca
                                           1,1,0.4));
Packit Service 6754ca
Packit Service 6754ca
                char * pre_offset = offset-&readingBuffer[0]+&preReadBuffer[0];
Packit Service 6754ca
                prereadbuf.insert (passivechannels[i-activechans],
Packit Service 6754ca
                                   Slice (type,
Packit Service 6754ca
                                          pre_offset,
Packit Service 6754ca
                                          bytes_per_bank,
Packit Service 6754ca
                                          bytes_per_row,
Packit Service 6754ca
                                          1,1,0.4));
Packit Service 6754ca
            }
Packit Service 6754ca
        }
Packit Service 6754ca
        switch (type)
Packit Service 6754ca
        {
Packit Service 6754ca
            case IMF::HALF :
Packit Service 6754ca
                offset+=2;
Packit Service 6754ca
                break;
Packit Service 6754ca
            case IMF::FLOAT :
Packit Service 6754ca
                offset+=4;
Packit Service 6754ca
                break;
Packit Service 6754ca
            default :
Packit Service 6754ca
                cout << "Unexpected Pixel Type\n";
Packit Service 6754ca
                exit(1);
Packit Service 6754ca
        
Packit Service 6754ca
        }
Packit Service 6754ca
    }
Packit Service 6754ca
Packit Service 6754ca
    return chanlist;
Packit Service 6754ca
}
Packit Service 6754ca
Packit Service 6754ca
Packit Service 6754ca
Packit Service 6754ca
Box2i writefile(Schema & scheme,FrameBuffer& buf,bool tiny)
Packit Service 6754ca
{
Packit Service 6754ca
    const int height = 128;
Packit Service 6754ca
    const int width  = 128;
Packit Service 6754ca
Packit Service 6754ca
    Header hdr(width,height,1);
Packit Service 6754ca
    
Packit Service 6754ca
    
Packit Service 6754ca
    //min values in range (-100,100)
Packit Service 6754ca
    hdr.dataWindow().min.x = int(200.0*double(rand())/double(RAND_MAX)-100.0);
Packit Service 6754ca
    hdr.dataWindow().min.y = int(200.0*double(rand())/double(RAND_MAX)-100.0);
Packit Service 6754ca
    
Packit Service 6754ca
    
Packit Service 6754ca
    // in tiny mode, make image up to 14*14 pixels (less than two SSE instructions)
Packit Service 6754ca
    if (tiny)
Packit Service 6754ca
    {
Packit Service 6754ca
        hdr.dataWindow().max.x = hdr.dataWindow().min.x + 1+int(13*double(rand())/double(RAND_MAX));
Packit Service 6754ca
        hdr.dataWindow().max.y = hdr.dataWindow().min.y + 1+int(13*double(rand())/double(RAND_MAX));
Packit Service 6754ca
    }
Packit Service 6754ca
    else
Packit Service 6754ca
    {
Packit Service 6754ca
        // in normal mode, make chunky images
Packit Service 6754ca
        hdr.dataWindow().max.x = hdr.dataWindow().min.x + 64+int(400*double(rand())/double(RAND_MAX));
Packit Service 6754ca
        hdr.dataWindow().max.y = hdr.dataWindow().min.y + 64+int(400*double(rand())/double(RAND_MAX));
Packit Service 6754ca
    }
Packit Service 6754ca
    
Packit Service 6754ca
    hdr.compression()=ZIPS_COMPRESSION;
Packit Service 6754ca
    
Packit Service 6754ca
    FrameBuffer dummy1,dummy2;
Packit Service 6754ca
    
Packit Service 6754ca
    hdr.channels() = setupBuffer (hdr,
Packit Service 6754ca
                                  scheme._active,
Packit Service 6754ca
                                  scheme._passive,
Packit Service 6754ca
                                  scheme._types,
Packit Service 6754ca
                                  buf,
Packit Service 6754ca
                                  dummy1,
Packit Service 6754ca
                                  dummy2,
Packit Service 6754ca
                                  scheme._banks,
Packit Service 6754ca
                                  true);
Packit Service 6754ca
    
Packit Service 6754ca
    if (scheme._views)
Packit Service 6754ca
    {
Packit Service 6754ca
        addMultiView(hdr,scheme.views());
Packit Service 6754ca
    }
Packit Service 6754ca
    
Packit Service 6754ca
    remove (filename.c_str());
Packit Service 6754ca
    OutputFile f(filename.c_str(), hdr);
Packit Service 6754ca
    f.setFrameBuffer(buf);
Packit Service 6754ca
    f.writePixels(hdr.dataWindow().max.y-hdr.dataWindow().min.y+1);
Packit Service 6754ca
Packit Service 6754ca
    return hdr.dataWindow();
Packit Service 6754ca
}
Packit Service 6754ca
Packit Service 6754ca
bool
Packit Service 6754ca
readfile (Schema scheme,
Packit Service 6754ca
          FrameBuffer & buf,      ///< list of channels to read: index to readingBuffer
Packit Service 6754ca
          FrameBuffer & preread,  ///< list of channels to skip: index to preReadBuffer
Packit Service 6754ca
          FrameBuffer & postread) ///< list of channels to skip: index to readingBuffer)
Packit Service 6754ca
{
Packit Service 6754ca
    InputFile infile (filename.c_str());
Packit Service 6754ca
    setupBuffer(infile.header(),
Packit Service 6754ca
                scheme._active,
Packit Service 6754ca
                scheme._passive,
Packit Service 6754ca
                scheme._types,
Packit Service 6754ca
                buf,
Packit Service 6754ca
                preread,
Packit Service 6754ca
                postread,
Packit Service 6754ca
                scheme._banks,false);
Packit Service 6754ca
    infile.setFrameBuffer(buf);
Packit Service 6754ca
    
Packit Service 6754ca
    cout.flush();
Packit Service 6754ca
    infile.readPixels (infile.header().dataWindow().min.y,
Packit Service 6754ca
                       infile.header().dataWindow().max.y);
Packit Service 6754ca
Packit Service 6754ca
    return infile.isOptimizationEnabled();
Packit Service 6754ca
}
Packit Service 6754ca
Packit Service 6754ca
void
Packit Service 6754ca
test (Schema writeScheme, Schema readScheme, bool nonfatal, bool tiny)
Packit Service 6754ca
{
Packit Service 6754ca
    ostringstream q;
Packit Service 6754ca
    q << writeScheme._name << " read as " << readScheme._name << "...";
Packit Service 6754ca
    cout << left << setw(53) << q.str();
Packit Service 6754ca
Packit Service 6754ca
    FrameBuffer writeFrameBuf;
Packit Service 6754ca
    Box2i dw = writefile(writeScheme,writeFrameBuf,tiny);
Packit Service 6754ca
    FrameBuffer readFrameBuf;
Packit Service 6754ca
    FrameBuffer preReadFrameBuf;
Packit Service 6754ca
    FrameBuffer postReadFrameBuf;
Packit Service 6754ca
    cout.flush();
Packit Service 6754ca
    bool opt = readfile (readScheme,
Packit Service 6754ca
                         readFrameBuf,
Packit Service 6754ca
                         preReadFrameBuf,
Packit Service 6754ca
                         postReadFrameBuf);
Packit Service 6754ca
    if (compare(readFrameBuf, writeFrameBuf, dw, nonfatal) &&
Packit Service 6754ca
        compare(preReadFrameBuf, postReadFrameBuf, dw, nonfatal)
Packit Service 6754ca
    )
Packit Service 6754ca
    {
Packit Service 6754ca
        cout <<  " OK ";
Packit Service 6754ca
        if (opt)
Packit Service 6754ca
        {
Packit Service 6754ca
            cout << "OPTIMISED ";
Packit Service 6754ca
            gOptimisedReads++;
Packit Service 6754ca
        }
Packit Service 6754ca
        cout << "\n";
Packit Service 6754ca
        gSuccesses++;
Packit Service 6754ca
    }
Packit Service 6754ca
    else
Packit Service 6754ca
    {
Packit Service 6754ca
        cout <<  " FAIL" << endl;
Packit Service 6754ca
        gFailures++;
Packit Service 6754ca
    }
Packit Service 6754ca
    remove (filename.c_str());
Packit Service 6754ca
}
Packit Service 6754ca
Packit Service 6754ca
Packit Service 6754ca
void runtests(bool nonfatal,bool tiny)
Packit Service 6754ca
{
Packit Service 6754ca
    srand(1);
Packit Service 6754ca
    int i=0;
Packit Service 6754ca
    int skipped=0;
Packit Service 6754ca
    
Packit Service 6754ca
    gFailures=0;
Packit Service 6754ca
    gSuccesses=0;
Packit Service 6754ca
    gOptimisedReads=0;
Packit Service 6754ca
    
Packit Service 6754ca
    
Packit Service 6754ca
    while(Schemes[i]._name!=NULL)
Packit Service 6754ca
    {
Packit Service 6754ca
        int j=0;
Packit Service 6754ca
        while(Schemes[j]._name!=NULL)
Packit Service 6754ca
        {
Packit Service 6754ca
           cout << right << setw(2) << i << ',' << right << setw(2) << j << ": ";
Packit Service 6754ca
           cout.flush();
Packit Service 6754ca
Packit Service 6754ca
           if (nonfatal)
Packit Service 6754ca
           {
Packit Service 6754ca
               cout << " skipping " << Schemes[i]._name << ',' << Schemes[j]._name << ": known to crash\n";
Packit Service 6754ca
               skipped++;
Packit Service 6754ca
           }
Packit Service 6754ca
           else
Packit Service 6754ca
           {
Packit Service 6754ca
               test(Schemes[i],Schemes[j],nonfatal,tiny);
Packit Service 6754ca
           }
Packit Service 6754ca
           j++;
Packit Service 6754ca
        }
Packit Service 6754ca
        i++;
Packit Service 6754ca
    }
Packit Service 6754ca
Packit Service 6754ca
    cout << gFailures << '/' << (gSuccesses+gFailures) << " runs failed\n";
Packit Service 6754ca
    cout << skipped << " tests skipped (assumed to be bad)\n";
Packit Service 6754ca
    cout << gOptimisedReads << '/' << gSuccesses << " optimised\n";
Packit Service 6754ca
        
Packit Service 6754ca
    if (gFailures>0 )
Packit Service 6754ca
    {
Packit Service 6754ca
        cout << " TESTS FAILED\n";
Packit Service 6754ca
        assert(false);
Packit Service 6754ca
    }
Packit Service 6754ca
}
Packit Service 6754ca
Packit Service 6754ca
Packit Service 6754ca
} // namespace anon
Packit Service 6754ca
Packit Service 6754ca
Packit Service 6754ca
void 
Packit Service 6754ca
testOptimizedInterleavePatterns (const std::string & tempDir)
Packit Service 6754ca
{
Packit Service 6754ca
    filename = tempDir + "imf_test_interleave_patterns.exr";
Packit Service 6754ca
Packit Service 6754ca
Packit Service 6754ca
    cout << "Testing SSE optimisation with different interleave patterns (large images) ... " << endl;
Packit Service 6754ca
    runtests (false,false);
Packit Service 6754ca
    
Packit Service 6754ca
    cout << "Testing SSE optimisation with different interleave patterns (tiny images) ... " << endl;
Packit Service 6754ca
    runtests (false,true);
Packit Service 6754ca
    
Packit Service 6754ca
    cout << "ok\n" << endl;
Packit Service 6754ca
}
Packit Service 6754ca