summaryrefslogtreecommitdiffstats
path: root/kipi-plugins/rawconverter/rawdecodingiface.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'kipi-plugins/rawconverter/rawdecodingiface.cpp')
-rw-r--r--kipi-plugins/rawconverter/rawdecodingiface.cpp668
1 files changed, 668 insertions, 0 deletions
diff --git a/kipi-plugins/rawconverter/rawdecodingiface.cpp b/kipi-plugins/rawconverter/rawdecodingiface.cpp
new file mode 100644
index 0000000..1a86bae
--- /dev/null
+++ b/kipi-plugins/rawconverter/rawdecodingiface.cpp
@@ -0,0 +1,668 @@
+/* ============================================================
+ *
+ * This file is a part of kipi-plugins project
+ * http://www.kipi-plugins.org
+ *
+ * Date : 2006-12-09
+ * Description : RAW decoding interface
+ *
+ * Copyright (C) 2006-2008 by Marcel Wiesweg <marcel.wiesweg@gmx.de>
+ * Copyright (C) 2006-2008 by Gilles Caulier <caulier dot gilles at gmail dot com>
+ *
+ * NOTE: Do not use kdDebug() in this implementation because
+ * it will be multithreaded. Use qDebug() instead.
+ * See B.K.O #133026 for details.
+ *
+ * This program is free software; you can redistribute it
+ * and/or modify it under the terms of the GNU General
+ * Public License as published by the Free Software Foundation;
+ * either version 2, or (at your option) any later version.
+ *
+ * This program 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 General Public License for more details.
+ *
+ * ============================================================ */
+
+// C++ includes.
+
+#include <cstdlib>
+#include <cstring>
+#include <iostream>
+#include <cstdio>
+
+// C Ansi includes.
+
+extern "C"
+{
+#include <unistd.h>
+#include <sys/ipc.h>
+#include <sys/shm.h>
+#include <sys/types.h>
+#include <jpeglib.h>
+#include <tiffio.h>
+#include <tiffvers.h>
+#include "iccjpeg.h"
+}
+
+// Qt Includes.
+
+#include <qcstring.h>
+#include <qfileinfo.h>
+
+// KDE includes.
+
+#include <kstandarddirs.h>
+
+// LibKExiv2 includes.
+
+#include <libkexiv2/kexiv2.h>
+
+// Local includes.
+
+#include "pluginsversion.h"
+#include "rawdecodingiface.h"
+#include "rawdecodingiface.moc"
+
+namespace KIPIRawConverterPlugin
+{
+
+RawDecodingIface::RawDecodingIface()
+ : KDcrawIface::KDcraw()
+{
+}
+
+RawDecodingIface::~RawDecodingIface()
+{
+}
+
+QByteArray RawDecodingIface::getICCProfilFromFile(KDcrawIface::RawDecodingSettings::OutputColorSpace colorSpace)
+{
+ QString filePath;
+ KGlobal::dirs()->addResourceType("profiles", KGlobal::dirs()->kde_default("data") +
+ "kipiplugin_rawconverter/profiles");
+
+ switch(colorSpace)
+ {
+ case KDcrawIface::RawDecodingSettings::SRGB:
+ {
+ filePath = KGlobal::dirs()->findResourceDir("profiles", "srgb.icm");
+ filePath.append("srgb.icm");
+ break;
+ }
+ case KDcrawIface::RawDecodingSettings::ADOBERGB:
+ {
+ filePath = KGlobal::dirs()->findResourceDir("profiles", "adobergb.icm");
+ filePath.append("adobergb.icm");
+ break;
+ }
+ case KDcrawIface::RawDecodingSettings::WIDEGAMMUT:
+ {
+ filePath = KGlobal::dirs()->findResourceDir("profiles", "widegamut.icm");
+ filePath.append("widegamut.icm");
+ break;
+ }
+ case KDcrawIface::RawDecodingSettings::PROPHOTO:
+ {
+ filePath = KGlobal::dirs()->findResourceDir("profiles", "prophoto.icm");
+ filePath.append("prophoto.icm");
+ break;
+ }
+ default:
+ break;
+ }
+
+ if ( filePath.isEmpty() )
+ return QByteArray();
+
+ QFile file(filePath);
+ if ( !file.open(IO_ReadOnly) )
+ return QByteArray();
+
+ QByteArray data(file.size());
+ QDataStream stream( &file );
+ stream.readRawBytes(data.data(), data.size());
+ file.close();
+ return data;
+}
+
+bool RawDecodingIface::decodeHalfRAWImage(const QString& filePath,
+ QString& destPath, SaveSettingsWidget::OutputFormat outputFileFormat,
+ KDcrawIface::RawDecodingSettings rawDecodingSettings)
+{
+ int width, height, rgbmax;
+ QByteArray imageData;
+ if (!KDcrawIface::KDcraw::decodeHalfRAWImage(filePath, rawDecodingSettings,
+ imageData, width, height, rgbmax))
+ return false;
+
+ return (loadedFromDcraw(filePath, destPath, outputFileFormat,
+ imageData, width, height));
+}
+
+bool RawDecodingIface::decodeRAWImage(const QString& filePath,
+ QString& destPath, SaveSettingsWidget::OutputFormat outputFileFormat,
+ KDcrawIface::RawDecodingSettings rawDecodingSettings)
+{
+ int width, height, rgbmax;
+ QByteArray imageData;
+ if (!KDcrawIface::KDcraw::decodeRAWImage(filePath, rawDecodingSettings,
+ imageData, width, height, rgbmax))
+ return false;
+
+ return (loadedFromDcraw(filePath, destPath, outputFileFormat,
+ imageData, width, height));
+}
+
+// ----------------------------------------------------------------------------------
+
+bool RawDecodingIface::loadedFromDcraw(const QString& filePath,
+ QString& destPath, SaveSettingsWidget::OutputFormat outputFileFormat,
+ const QByteArray& imageData, int width, int height)
+{
+ // -- Use a QImage instance to write IPTC preview and Exif thumbnail -------
+
+ QImage img(width, height, 32);
+ uchar* dptr = img.bits();
+ uchar* sptr = (uchar*)imageData.data();
+
+ // Set RGB color components.
+ for (int i = 0 ; i < width * height ; i++)
+ {
+ dptr[0] = sptr[2];
+ dptr[1] = sptr[1];
+ dptr[2] = sptr[0];
+ dptr[3] = 0xFF;
+ dptr += 4;
+ sptr += 3;
+ }
+
+ QImage iptcPreview = img.scale(1280, 1024, QImage::ScaleMin);
+ QImage exifThumbnail = iptcPreview.scale(160, 120, QImage::ScaleMin);
+
+ // -- Write image data into destination file -------------------------------
+
+ QByteArray ICCColorProfile = getICCProfilFromFile(m_rawDecodingSettings.outputColorSpace);
+ QString soft = QString("Kipi-plugins v.%1").arg(kipiplugins_version);
+ QFileInfo fi(filePath);
+ destPath = fi.dirPath(true) + QString("/") + ".kipi-rawconverter-tmp-"
+ + QString::number(::time(0));
+
+ // Metadata restoration and update.
+ KExiv2Iface::KExiv2 exiv2Iface;
+ exiv2Iface.load(filePath);
+ exiv2Iface.setImageProgramId(QString("Kipi-plugins"), QString(kipiplugins_version));
+ exiv2Iface.setImageDimensions(QSize(width, height));
+ exiv2Iface.setExifThumbnail(exifThumbnail);
+
+ // Update Iptc preview.
+ // NOTE: see B.K.O #130525. a JPEG segment is limited to 64K. If the IPTC byte array is
+ // bigger than 64K duing of image preview tag size, the target JPEG image will be
+ // broken. Note that IPTC image preview tag is limited to 256K!!!
+ // There is no limitation with TIFF and PNG about IPTC byte array size.
+ if (outputFileFormat != SaveSettingsWidget::OUTPUT_JPEG)
+ exiv2Iface.setImagePreview(iptcPreview);
+
+ exiv2Iface.setExifTagString("Exif.Image.DocumentName", fi.fileName());
+
+ switch(outputFileFormat)
+ {
+ case SaveSettingsWidget::OUTPUT_JPEG:
+ {
+ FILE* f = 0;
+ f = fopen(QFile::encodeName(destPath), "wb");
+
+ if (!f)
+ {
+ qDebug("Failed to open JPEG file for writing");
+ return false;
+ }
+
+ struct jpeg_compress_struct cinfo;
+ struct jpeg_error_mgr jerr;
+
+ int row_stride;
+ JSAMPROW row_pointer[1];
+
+ // Init JPEG compressor.
+ cinfo.err = jpeg_std_error(&jerr);
+ jpeg_create_compress(&cinfo);
+ jpeg_stdio_dest(&cinfo, f);
+ cinfo.image_width = width;
+ cinfo.image_height = height;
+ cinfo.input_components = 3;
+ cinfo.in_color_space = JCS_RGB;
+ jpeg_set_defaults(&cinfo);
+
+ // B.K.O #149578: set encoder horizontal and vertical chroma subsampling
+ // factor to 2x1, 1x1, 1x1 (4:2:2) : Medium subsampling.
+ // See this page for details: http://en.wikipedia.org/wiki/Chroma_subsampling
+ cinfo.comp_info[0].h_samp_factor = 2;
+ cinfo.comp_info[0].v_samp_factor = 1;
+ cinfo.comp_info[1].h_samp_factor = 1;
+ cinfo.comp_info[1].v_samp_factor = 1;
+ cinfo.comp_info[2].h_samp_factor = 1;
+ cinfo.comp_info[2].v_samp_factor = 1;
+
+ // B.K.O #154273: use 99 compresion level instead 100 to reduce output JPEG file size.
+ jpeg_set_quality(&cinfo, 99, true);
+ jpeg_start_compress(&cinfo, true);
+
+ // Write ICC color profil.
+ if (!ICCColorProfile.isEmpty())
+ write_icc_profile (&cinfo, (JOCTET *)ICCColorProfile.data(), ICCColorProfile.size());
+
+ // Write image data
+ row_stride = cinfo.image_width * 3;
+ while (!m_cancel && (cinfo.next_scanline < cinfo.image_height))
+ {
+ row_pointer[0] = (uchar*)imageData.data() + (cinfo.next_scanline * row_stride);
+ jpeg_write_scanlines(&cinfo, row_pointer, 1);
+ }
+
+ jpeg_finish_compress(&cinfo);
+ jpeg_destroy_compress(&cinfo);
+ fclose(f);
+
+ exiv2Iface.save(destPath);
+ break;
+ }
+
+ case SaveSettingsWidget::OUTPUT_PNG:
+ {
+ FILE* f = 0;
+ f = fopen(QFile::encodeName(destPath), "wb");
+
+ if (!f)
+ {
+ qDebug("Failed to open PNG file for writing");
+ return false;
+ }
+
+ png_color_8 sig_bit;
+ png_bytep row_ptr;
+ png_structp png_ptr = png_create_write_struct(PNG_LIBPNG_VER_STRING, NULL, NULL, NULL);
+ png_infop info_ptr = png_create_info_struct(png_ptr);
+ png_init_io(png_ptr, f);
+ png_set_IHDR(png_ptr, info_ptr, width, height, 8,
+ PNG_COLOR_TYPE_RGB, PNG_INTERLACE_NONE,
+ PNG_COMPRESSION_TYPE_BASE, PNG_FILTER_TYPE_BASE);
+ sig_bit.red = 8;
+ sig_bit.green = 8;
+ sig_bit.blue = 8;
+ sig_bit.alpha = 8;
+ png_set_sBIT(png_ptr, info_ptr, &sig_bit);
+ png_set_compression_level(png_ptr, 9);
+
+ // Write ICC profil.
+ if (!ICCColorProfile.isEmpty())
+ {
+ png_set_iCCP(png_ptr, info_ptr, "icc", PNG_COMPRESSION_TYPE_BASE,
+ ICCColorProfile.data(), ICCColorProfile.size());
+ }
+
+ QString libpngver(PNG_HEADER_VERSION_STRING);
+ libpngver.replace('\n', ' ');
+ soft.append(QString(" (%1)").arg(libpngver));
+ png_text text;
+ text.key = "Software";
+ text.text = (char *)soft.ascii();
+ text.compression = PNG_TEXT_COMPRESSION_zTXt;
+ png_set_text(png_ptr, info_ptr, &(text), 1);
+
+ // Store Exif data.
+ QByteArray ba = exiv2Iface.getExif();
+ const uchar ExifHeader[] = {0x45, 0x78, 0x69, 0x66, 0x00, 0x00};
+ QByteArray profile = QByteArray(ba.size() + sizeof(ExifHeader));
+ memcpy(profile.data(), ExifHeader, sizeof(ExifHeader));
+ memcpy(profile.data()+sizeof(ExifHeader), ba.data(), ba.size());
+ writeRawProfile(png_ptr, info_ptr, "exif", profile.data(), (png_uint_32) profile.size());
+
+ // Store Iptc data.
+ QByteArray ba2 = exiv2Iface.getIptc();
+ writeRawProfile(png_ptr, info_ptr, "iptc", ba2.data(), (png_uint_32) ba2.size());
+
+ png_write_info(png_ptr, info_ptr);
+ png_set_shift(png_ptr, &sig_bit);
+ png_set_packing(png_ptr);
+ unsigned char* ptr = (unsigned char*)imageData.data();
+
+ for (int y = 0; !m_cancel && (y < height); y++)
+ {
+ row_ptr = (png_bytep) ptr;
+ png_write_rows(png_ptr, &row_ptr, 1);
+ ptr += (width * 3);
+ }
+
+ png_write_end(png_ptr, info_ptr);
+ png_destroy_write_struct(&png_ptr, (png_infopp) & info_ptr);
+ png_destroy_info_struct(png_ptr, (png_infopp) & info_ptr);
+ fclose(f);
+ break;
+ }
+
+ case SaveSettingsWidget::OUTPUT_TIFF:
+ {
+ TIFF *tif=0;
+ unsigned char *data=0;
+ int y;
+ int w;
+
+ tif = TIFFOpen(QFile::encodeName(destPath), "wb");
+
+ if (!tif)
+ {
+ qDebug("Failed to open TIFF file for writing");
+ return false;
+ }
+
+ TIFFSetField(tif, TIFFTAG_IMAGEWIDTH, width);
+ TIFFSetField(tif, TIFFTAG_IMAGELENGTH, height);
+ TIFFSetField(tif, TIFFTAG_ORIENTATION, ORIENTATION_TOPLEFT);
+ TIFFSetField(tif, TIFFTAG_BITSPERSAMPLE, 8);
+ TIFFSetField(tif, TIFFTAG_PLANARCONFIG, PLANARCONFIG_CONTIG);
+ TIFFSetField(tif, TIFFTAG_COMPRESSION, COMPRESSION_ADOBE_DEFLATE);
+ TIFFSetField(tif, TIFFTAG_ZIPQUALITY, 9);
+ // NOTE : this tag values aren't defined in libtiff 3.6.1. '2' is PREDICTOR_HORIZONTAL.
+ // Use horizontal differencing for images which are
+ // likely to be continuous tone. The TIFF spec says that this
+ // usually leads to better compression.
+ // See this url for more details:
+ // http://www.awaresystems.be/imaging/tiff/tifftags/predictor.html
+ TIFFSetField(tif, TIFFTAG_PREDICTOR, 2);
+ TIFFSetField(tif, TIFFTAG_SAMPLESPERPIXEL, 3);
+ TIFFSetField(tif, TIFFTAG_PHOTOMETRIC, PHOTOMETRIC_RGB);
+ w = TIFFScanlineSize(tif);
+ TIFFSetField(tif, TIFFTAG_ROWSPERSTRIP, TIFFDefaultStripSize(tif, 0));
+
+ // Store Iptc data.
+ QByteArray ba2 = exiv2Iface.getIptc(true);
+#if defined(TIFFTAG_PHOTOSHOP)
+ TIFFSetField (tif, TIFFTAG_PHOTOSHOP, (uint32)ba2.size(), (uchar *)ba2.data());
+#endif
+
+ QString libtiffver(TIFFLIB_VERSION_STR);
+ libtiffver.replace('\n', ' ');
+ soft.append(QString(" ( %1 )").arg(libtiffver));
+ TIFFSetField(tif, TIFFTAG_SOFTWARE, (const char*)soft.ascii());
+
+ // Write ICC profil.
+ if (!ICCColorProfile.isEmpty())
+ {
+#if defined(TIFFTAG_ICCPROFILE)
+ TIFFSetField(tif, TIFFTAG_ICCPROFILE, (uint32)ICCColorProfile.size(),
+ (uchar *)ICCColorProfile.data());
+#endif
+ }
+
+ // Write full image data in tiff directory IFD0
+
+ for (y = 0; !m_cancel && (y < height); y++)
+ {
+ data = (unsigned char*)imageData.data() + (y * width * 3);
+ TIFFWriteScanline(tif, data, y, 0);
+ }
+
+ TIFFWriteDirectory(tif);
+
+ // Write thumbnail in tiff directory IFD1
+
+ QImage thumb = exiv2Iface.getExifThumbnail(false);
+ if (!thumb.isNull())
+ {
+ TIFFSetField(tif, TIFFTAG_IMAGEWIDTH, (uint32)thumb.width());
+ TIFFSetField(tif, TIFFTAG_IMAGELENGTH, (uint32)thumb.height());
+ TIFFSetField(tif, TIFFTAG_PHOTOMETRIC, PHOTOMETRIC_RGB);
+ TIFFSetField(tif, TIFFTAG_PLANARCONFIG, PLANARCONFIG_CONTIG);
+ TIFFSetField(tif, TIFFTAG_ORIENTATION, ORIENTATION_TOPLEFT);
+ TIFFSetField(tif, TIFFTAG_RESOLUTIONUNIT, RESUNIT_NONE);
+ TIFFSetField(tif, TIFFTAG_COMPRESSION, COMPRESSION_NONE);
+ TIFFSetField(tif, TIFFTAG_SAMPLESPERPIXEL, 3);
+ TIFFSetField(tif, TIFFTAG_BITSPERSAMPLE, 8);
+ TIFFSetField(tif, TIFFTAG_ROWSPERSTRIP, TIFFDefaultStripSize(tif, 0));
+
+ int i=0;
+ uint32 x, y;
+ uchar *pixelThumb;
+ uchar *dataThumb = thumb.bits();
+ uint8 *bufThumb = (uint8 *) _TIFFmalloc(TIFFScanlineSize(tif));
+
+ if (!bufThumb)
+ {
+ qDebug("Cannot allocate memory buffer for TIFF thumbnail.");
+ TIFFClose(tif);
+ return false;
+ }
+
+ for (y = 0 ; y < uint32(thumb.height()) ; y++)
+ {
+ i = 0;
+
+ for (x = 0 ; x < uint32(thumb.width()) ; x++)
+ {
+ pixelThumb = &dataThumb[((y * thumb.width()) + x) * 4];
+
+ // This might be endian dependent
+ bufThumb[i++] = (uint8)pixelThumb[2];
+ bufThumb[i++] = (uint8)pixelThumb[1];
+ bufThumb[i++] = (uint8)pixelThumb[0];
+ }
+
+ if (!TIFFWriteScanline(tif, bufThumb, y, 0))
+ {
+ qDebug("Cannot write TIFF thumbnail to target file.");
+ _TIFFfree(bufThumb);
+ TIFFClose(tif);
+ return false;
+ }
+ }
+
+ _TIFFfree(bufThumb);
+ }
+ TIFFClose(tif);
+
+ // Store metadata (Exiv2 0.18 support tiff writting mode)
+ exiv2Iface.save(destPath);
+
+ break;
+ }
+
+ case SaveSettingsWidget::OUTPUT_PPM:
+ {
+ FILE* f = fopen(QFile::encodeName(destPath), "wb");
+ if (!f)
+ {
+ qDebug("Failed to open ppm file for writing");
+ return false;
+ }
+
+ fprintf(f, "P6\n%d %d\n255\n", width, height);
+ fwrite(imageData.data(), 1, width*height*3, f);
+ fclose(f);
+ break;
+ }
+ default:
+ {
+ qDebug("Invalid output file format");
+ return false;
+ }
+ }
+
+ if (m_cancel)
+ {
+ ::remove(QFile::encodeName(destPath));
+ return false;
+ }
+
+ return true;
+}
+
+//----------------------------------------------------------------------------------
+
+void RawDecodingIface::writeRawProfile(png_struct *ping, png_info *ping_info, char *profile_type,
+ char *profile_data, png_uint_32 length)
+{
+ png_textp text;
+
+ register long i;
+
+ uchar *sp;
+
+ png_charp dp;
+
+ png_uint_32 allocated_length, description_length;
+
+ const uchar hex[16] = {'0','1','2','3','4','5','6','7','8','9','a','b','c','d','e','f'};
+
+ qDebug("Writing Raw profile: type=%s, length=%i", profile_type, (int)length);
+
+ text = (png_textp) png_malloc(ping, (png_uint_32) sizeof(png_text));
+ description_length = strlen((const char *) profile_type);
+ allocated_length = (png_uint_32) (length*2 + (length >> 5) + 20 + description_length);
+
+ text[0].text = (png_charp) png_malloc(ping, allocated_length);
+ text[0].key = (png_charp) png_malloc(ping, (png_uint_32) 80);
+ text[0].key[0] = '\0';
+
+ concatenateString(text[0].key, "Raw profile type ", 4096);
+ concatenateString(text[0].key, (const char *) profile_type, 62);
+
+ sp = (uchar*)profile_data;
+ dp = text[0].text;
+ *dp++='\n';
+
+ copyString(dp, (const char *) profile_type, allocated_length);
+
+ dp += description_length;
+ *dp++='\n';
+
+ formatString(dp, allocated_length-strlen(text[0].text), "%8lu ", length);
+
+ dp += 8;
+
+ for (i=0; i < (long) length; i++)
+ {
+ if (i%36 == 0)
+ *dp++='\n';
+
+ *(dp++)=(char) hex[((*sp >> 4) & 0x0f)];
+ *(dp++)=(char) hex[((*sp++ ) & 0x0f)];
+ }
+
+ *dp++='\n';
+ *dp='\0';
+ text[0].text_length = (png_size_t) (dp-text[0].text);
+ text[0].compression = -1;
+
+ if (text[0].text_length <= allocated_length)
+ png_set_text(ping, ping_info,text, 1);
+
+ png_free(ping, text[0].text);
+ png_free(ping, text[0].key);
+ png_free(ping, text);
+}
+
+size_t RawDecodingIface::concatenateString(char *destination, const char *source, const size_t length)
+{
+ register char *q;
+
+ register const char *p;
+
+ register size_t i;
+
+ size_t count;
+
+ if ( !destination || !source || length == 0 )
+ return 0;
+
+ p = source;
+ q = destination;
+ i = length;
+
+ while ((i-- != 0) && (*q != '\0'))
+ q++;
+
+ count = (size_t) (q-destination);
+ i = length-count;
+
+ if (i == 0)
+ return(count+strlen(p));
+
+ while (*p != '\0')
+ {
+ if (i != 1)
+ {
+ *q++=(*p);
+ i--;
+ }
+ p++;
+ }
+
+ *q='\0';
+
+ return(count+(p-source));
+}
+
+size_t RawDecodingIface::copyString(char *destination, const char *source, const size_t length)
+{
+ register char *q;
+
+ register const char *p;
+
+ register size_t i;
+
+ if ( !destination || !source || length == 0 )
+ return 0;
+
+ p = source;
+ q = destination;
+ i = length;
+
+ if ((i != 0) && (--i != 0))
+ {
+ do
+ {
+ if ((*q++=(*p++)) == '\0')
+ break;
+ }
+ while (--i != 0);
+ }
+
+ if (i == 0)
+ {
+ if (length != 0)
+ *q='\0';
+
+ do
+ {
+ }
+ while (*p++ != '\0');
+ }
+
+ return((size_t) (p-source-1));
+}
+
+long RawDecodingIface::formatString(char *string, const size_t length, const char *format,...)
+{
+ long n;
+
+ va_list operands;
+
+ va_start(operands,format);
+ n = (long) formatStringList(string, length, format, operands);
+ va_end(operands);
+ return(n);
+}
+
+long RawDecodingIface::formatStringList(char *string, const size_t length, const char *format, va_list operands)
+{
+ int n = vsnprintf(string, length, format, operands);
+
+ if (n < 0)
+ string[length-1] = '\0';
+
+ return((long) n);
+}
+
+} // namespace KIPIRawConverterPlugin