/* This file is part of ksquirrel-libs (http://ksquirrel.sf.net) Copyright (c) 2005 Dmitry Baryshev This library is free software; you can redistribute it and/or modify it under the terms of the GNU Library General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This library is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Library General Public License for more details. You should have received a copy of the GNU Library General Public License along with this library; see the file COPYING. If not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ #include #include #include #include #include #include "ksquirrel-libs/fmt_types.h" #include "ksquirrel-libs/fileio.h" #include "ksquirrel-libs/fmt_utils.h" #include #include #include #include #include #include #include #include #include #include #include #include #include "fmt_codec_openexr_defs.h" #include "fmt_codec_openexr.h" #include "ksquirrel-libs/error.h" #include "../xpm/codec_openexr.xpm" RGBA RgbaToRGBA(struct Rgba); /* * * A high-dynamic-range image format from Industrial Light & Magic * for use in digital visual effects production. * */ fmt_codec::fmt_codec() : fmt_codec_base() {} fmt_codec::~fmt_codec() {} void fmt_codec::options(codec_options *o) { o->version = "0.2.1"; o->name = "OpenEXR"; o->filter = "*.exr "; o->config = ""; o->mime = ""; o->mimetype = "image/x-exr"; o->pixmap = codec_openexr; o->readable = true; o->canbemultiple = false; o->writestatic = false; o->writeanimated = false; o->needtempfile = false; } s32 fmt_codec::read_init(const std::string &fl) { frs.open(fl.c_str(), ios::binary | ios::in); if(!frs.good()) return SQE_R_NOFILE; frs.close(); currentImage = -1; read_error = false; pixels = 0; file = fl; finfo.animated = false; return SQE_OK; } s32 fmt_codec::read_next() { currentImage++; if(currentImage) return SQE_NOTOK; fmt_image image; s32 width, height; RgbaInputFile *in = NULL; pixels = new Array2D; try { in = new RgbaInputFile(file.c_str()); Imath::Box2i dw = in->dataWindow(); width = dw.max.x - dw.min.x + 1; height = dw.max.y - dw.min.y + 1; pixels->resizeErase(height, width); in->setFrameBuffer(&(*pixels)[0][0] - dw.min.x - dw.min.y * width, 1, width); in->readPixels(dw.min.y, dw.max.y); } catch(const exception &e) { cerr << e.what() << endl; delete in; return SQE_R_BADFILE; } switch(in->compression()) { case Imf::NO_COMPRESSION: image.compression = "-"; break; case Imf::RLE_COMPRESSION: image.compression = "RLE"; break; case Imf::ZIPS_COMPRESSION: image.compression = "ZIPS"; break; case Imf::ZIP_COMPRESSION: image.compression = "ZIP"; break; case Imf::PIZ_COMPRESSION: image.compression = "PIZ"; break; case Imf::PXR24_COMPRESSION: image.compression = "PXR24"; break; case Imf::NUM_COMPRESSION_METHODS: image.compression = "Different methods"; break; default: image.compression = "Unknown"; } image.colorspace = "RGBA"; image.bpp = 32; image.w = width; image.h = height; finfo.image.push_back(image); line = -1; delete in; return SQE_OK; } s32 fmt_codec::read_next_pass() { return SQE_OK; } s32 fmt_codec::read_scanline(RGBA *scan) { RGBA rgba; fmt_image *im = image(currentImage); fmt_utils::fillAlpha(scan, im->w); line++; for(s32 x = 0; x < im->w; x++) { rgba = RgbaToRGBA((*pixels)[line][x]); memcpy(scan+x, &rgba, sizeof(RGBA)); } return SQE_OK; } void fmt_codec::read_close() { finfo.meta.clear(); finfo.image.clear(); delete pixels; pixels = 0; } /* * This function is taken from exr plugin for KImageIO, tdelibs-3.4.0: * * KImageIO Routines to read (and perhaps in the future, write) images * in the high dynamic range EXR format. * Copyright (c) 2003, Brad Hards * * This library is distributed under the conditions of the GNU LGPL. * * $Id: exr.cpp,v 1.4 2004/11/22 03:48:27 mueller Exp $ */ /* * utility function * * this does a conversion from the ILM Half (equal to Nvidia Half) * format into the normal 32 bit pixel format. Process is from the * ILM code. * */ RGBA RgbaToRGBA(struct Rgba imagePixel) { float r,g,b,a; // 1) Compensate for fogging by subtracting defog // from the raw pixel values. // Response: We work with defog of 0.0, so this is a no-op // 2) Multiply the defogged pixel values by // 2^(exposure + 2.47393). // Response: We work with exposure of 0.0. // (2^2.47393) is 5.55555 r = imagePixel.r * 5.55555; g = imagePixel.g * 5.55555; b = imagePixel.b * 5.55555; a = imagePixel.a * 5.55555; // 3) Values, which are now 1.0, are called "middle gray". // If defog and exposure are both set to 0.0, then // middle gray corresponds to a raw pixel value of 0.18. // In step 6, middle gray values will be mapped to an // intensity 3.5 f-stops below the display's maximum // intensity. // Response: no apparent content. // 4) Apply a knee function. The knee function has two // parameters, kneeLow and kneeHigh. Pixel values // below 2^kneeLow are not changed by the knee // function. Pixel values above kneeLow are lowered // according to a logarithmic curve, such that the // value 2^kneeHigh is mapped to 2^3.5 (in step 6, // this value will be mapped to the the display's // maximum intensity). // Response: kneeLow = 0.0 (2^0.0 => 1); kneeHigh = 5.0 (2^5 =>32) if (r > 1.0) r = 1.0 + Imath::Math::log ((r-1.0) * 0.184874 + 1) / 0.184874; if (g > 1.0) g = 1.0 + Imath::Math::log ((g-1.0) * 0.184874 + 1) / 0.184874; if (b > 1.0) b = 1.0 + Imath::Math::log ((b-1.0) * 0.184874 + 1) / 0.184874; if (a > 1.0) a = 1.0 + Imath::Math::log ((a-1.0) * 0.184874 + 1) / 0.184874; // // 5) Gamma-correct the pixel values, assuming that the // screen's gamma is 0.4545 (or 1/2.2). r = Imath::Math::pow (r, 0.4545); g = Imath::Math::pow (g, 0.4545); b = Imath::Math::pow (b, 0.4545); a = Imath::Math::pow (a, 0.4545); // 6) Scale the values such that pixels middle gray // pixels are mapped to 84.66 (or 3.5 f-stops below // the display's maximum intensity). // // 7) Clamp the values to [0, 255]. return RGBA( s8 (Imath::clamp ( r * 84.66f, 0.f, 255.f ) ), s8 (Imath::clamp ( g * 84.66f, 0.f, 255.f ) ), s8 (Imath::clamp ( b * 84.66f, 0.f, 255.f ) ), s8 (Imath::clamp ( a * 84.66f, 0.f, 255.f ) ) ); } #include "fmt_codec_cd_func.h"