3 * Copyright (C) 2008 The Geeqie Team
5 * Author: Vladimir Nadvornik
7 * This software is released under the GNU General Public License (GNU GPL).
8 * Please read the included file COPYING for more information.
9 * This software comes with no warranty of any kind, use at your own risk!
16 #include <exiv2/image.hpp>
17 #include <exiv2/exif.hpp>
20 // EXIV2_TEST_VERSION is defined in Exiv2 0.15 and newer.
21 #ifndef EXIV2_TEST_VERSION
22 # define EXIV2_TEST_VERSION(major,minor,patch) \
23 ( EXIV2_VERSION >= EXIV2_MAKE_VERSION(major,minor,patch) )
27 #include <sys/types.h>
33 #if !EXIV2_TEST_VERSION(0,17,90)
34 #include <exiv2/tiffparser.hpp>
35 #include <exiv2/tiffcomposite.hpp>
36 #include <exiv2/tiffvisitor.hpp>
37 #include <exiv2/tiffimage.hpp>
38 #include <exiv2/cr2image.hpp>
39 #include <exiv2/crwimage.hpp>
40 #if EXIV2_TEST_VERSION(0,16,0)
41 #include <exiv2/orfimage.hpp>
43 #if EXIV2_TEST_VERSION(0,13,0)
44 #include <exiv2/rafimage.hpp>
46 #include <exiv2/futils.hpp>
48 #include <exiv2/preview.hpp>
58 #include "filefilter.h"
59 #include "ui_fileops.h"
64 Exiv2::ExifData::const_iterator exifIter; /* for exif_get_next_item */
65 Exiv2::IptcData::const_iterator iptcIter; /* for exif_get_next_item */
66 #if EXIV2_TEST_VERSION(0,16,0)
67 Exiv2::XmpData::const_iterator xmpIter; /* for exif_get_next_item */
74 virtual void writeMetadata()
76 g_critical("Unsupported method of writing metadata");
79 virtual ExifData *original()
84 virtual Exiv2::Image *image() = 0;
86 virtual Exiv2::ExifData &exifData() = 0;
88 virtual Exiv2::IptcData &iptcData() = 0;
90 #if EXIV2_TEST_VERSION(0,16,0)
91 virtual Exiv2::XmpData &xmpData() = 0;
94 virtual void add_jpeg_color_profile(unsigned char *cp_data, guint cp_length) = 0;
96 virtual guchar *get_jpeg_color_profile(guint *data_len) = 0;
99 // This allows read-only access to the original metadata
100 struct _ExifDataOriginal : public _ExifData
103 Exiv2::Image::AutoPtr image_;
105 /* the icc profile in jpeg is not technically exif - store it here */
106 unsigned char *cp_data_;
111 _ExifDataOriginal(gchar *path)
115 gchar *pathl = path_from_utf8(path);
116 image_ = Exiv2::ImageFactory::open(pathl);
118 // g_assert (image.get() != 0);
119 image_->readMetadata();
121 #if EXIV2_TEST_VERSION(0,16,0)
122 if (image_->mimeType() == "application/rdf+xml")
124 //Exiv2 sidecar converts xmp to exif and iptc, we don't want it.
125 image_->clearExifData();
126 image_->clearIptcData();
130 #if EXIV2_TEST_VERSION(0,14,0)
131 if (image_->mimeType() == "image/jpeg")
133 /* try to get jpeg color profile */
134 Exiv2::BasicIo &io = image_->io();
135 gint open = io.isopen();
136 if (!open) io.open();
137 unsigned char *mapped = (unsigned char*)io.mmap();
138 if (mapped) exif_jpeg_parse_color(this, mapped, io.size());
140 if (!open) io.close();
145 virtual ~_ExifDataOriginal()
147 if (cp_data_) g_free(cp_data_);
150 virtual Exiv2::Image *image()
155 virtual Exiv2::ExifData &exifData ()
157 return image_->exifData();
160 virtual Exiv2::IptcData &iptcData ()
162 return image_->iptcData();
165 #if EXIV2_TEST_VERSION(0,16,0)
166 virtual Exiv2::XmpData &xmpData ()
168 return image_->xmpData();
172 virtual void add_jpeg_color_profile(unsigned char *cp_data, guint cp_length)
174 if (cp_data_) g_free(cp_data_);
176 cp_length_ = cp_length;
179 virtual guchar *get_jpeg_color_profile(guint *data_len)
183 if (data_len) *data_len = cp_length_;
184 return (unsigned char *) g_memdup(cp_data_, cp_length_);
190 // This allows read-write access to the metadata
191 struct _ExifDataProcessed : public _ExifData
194 _ExifDataOriginal *imageData_;
195 _ExifDataOriginal *sidecarData_;
197 Exiv2::ExifData exifData_;
198 Exiv2::IptcData iptcData_;
199 #if EXIV2_TEST_VERSION(0,16,0)
200 Exiv2::XmpData xmpData_;
204 _ExifDataProcessed(gchar *path, gchar *sidecar_path)
206 imageData_ = new _ExifDataOriginal(path);
208 #if EXIV2_TEST_VERSION(0,16,0)
209 xmpData_ = imageData_->xmpData();
210 DEBUG_2("xmp count %li", xmpData_.count());
211 if (sidecar_path && xmpData_.empty())
213 sidecarData_ = new _ExifDataOriginal(sidecar_path);
214 xmpData_ = sidecarData_->xmpData();
217 exifData_ = imageData_->exifData();
218 iptcData_ = imageData_->iptcData();
221 virtual ~_ExifDataProcessed()
223 if (imageData_) delete imageData_;
224 if (sidecarData_) delete sidecarData_;
227 virtual ExifData *original()
232 virtual void writeMetadata()
237 sidecarData_->image()->setXmpData(xmpData_);
238 //Exiv2 sidecar converts xmp to exif and iptc, we don't want it.
239 sidecarData_->image()->clearExifData();
240 sidecarData_->image()->clearIptcData();
241 sidecarData_->image()->writeMetadata();
245 imageData_->image()->setExifData(exifData_);
246 imageData_->image()->setIptcData(iptcData_);
247 imageData_->image()->setXmpData(xmpData_);
248 imageData_->image()->writeMetadata();
252 virtual Exiv2::Image *image()
254 return imageData_->image();
257 virtual Exiv2::ExifData &exifData ()
262 virtual Exiv2::IptcData &iptcData ()
267 #if EXIV2_TEST_VERSION(0,16,0)
268 virtual Exiv2::XmpData &xmpData ()
274 virtual void add_jpeg_color_profile(unsigned char *cp_data, guint cp_length)
276 imageData_->add_jpeg_color_profile(cp_data, cp_length);
279 virtual guchar *get_jpeg_color_profile(guint *data_len)
281 return imageData_->get_jpeg_color_profile(data_len);
290 ExifData *exif_read(gchar *path, gchar *sidecar_path)
292 DEBUG_1("exif read %s, sidecar: %s", path, sidecar_path ? sidecar_path : "-");
294 return new _ExifDataProcessed(path, sidecar_path);
296 catch (Exiv2::AnyError& e) {
297 std::cout << "Caught Exiv2 exception '" << e << "'\n";
303 int exif_write(ExifData *exif)
306 exif->writeMetadata();
309 catch (Exiv2::AnyError& e) {
310 std::cout << "Caught Exiv2 exception '" << e << "'\n";
317 void exif_free(ExifData *exif)
319 g_assert(dynamic_cast<_ExifDataProcessed *>(exif)); // this should not be called on ExifDataOriginal
323 ExifData *exif_get_original(ExifData *exif)
325 return exif->original();
329 ExifItem *exif_get_item(ExifData *exif, const gchar *key)
332 Exiv2::Metadatum *item = NULL;
334 Exiv2::ExifKey ekey(key);
335 Exiv2::ExifData::iterator pos = exif->exifData().findKey(ekey);
336 if (pos == exif->exifData().end()) return NULL;
339 catch (Exiv2::AnyError& e) {
341 Exiv2::IptcKey ekey(key);
342 Exiv2::IptcData::iterator pos = exif->iptcData().findKey(ekey);
343 if (pos == exif->iptcData().end()) return NULL;
346 catch (Exiv2::AnyError& e) {
347 #if EXIV2_TEST_VERSION(0,16,0)
348 Exiv2::XmpKey ekey(key);
349 Exiv2::XmpData::iterator pos = exif->xmpData().findKey(ekey);
350 if (pos == exif->xmpData().end()) return NULL;
355 return (ExifItem *)item;
357 catch (Exiv2::AnyError& e) {
358 std::cout << "Caught Exiv2 exception '" << e << "'\n";
363 ExifItem *exif_add_item(ExifData *exif, const gchar *key)
366 Exiv2::Metadatum *item = NULL;
368 Exiv2::ExifKey ekey(key);
369 exif->exifData().add(ekey, NULL);
370 Exiv2::ExifData::iterator pos = exif->exifData().end(); // a hack, there should be a better way to get the currently added item
374 catch (Exiv2::AnyError& e) {
376 Exiv2::IptcKey ekey(key);
377 exif->iptcData().add(ekey, NULL);
378 Exiv2::IptcData::iterator pos = exif->iptcData().end();
382 catch (Exiv2::AnyError& e) {
383 #if EXIV2_TEST_VERSION(0,16,0)
384 Exiv2::XmpKey ekey(key);
385 exif->xmpData().add(ekey, NULL);
386 Exiv2::XmpData::iterator pos = exif->xmpData().end();
392 return (ExifItem *)item;
394 catch (Exiv2::AnyError& e) {
395 std::cout << "Caught Exiv2 exception '" << e << "'\n";
401 ExifItem *exif_get_first_item(ExifData *exif)
404 exif->exifIter = exif->exifData().begin();
405 exif->iptcIter = exif->iptcData().begin();
406 #if EXIV2_TEST_VERSION(0,16,0)
407 exif->xmpIter = exif->xmpData().begin();
409 if (exif->exifIter != exif->exifData().end())
411 const Exiv2::Metadatum *item = &*exif->exifIter;
413 return (ExifItem *)item;
415 if (exif->iptcIter != exif->iptcData().end())
417 const Exiv2::Metadatum *item = &*exif->iptcIter;
419 return (ExifItem *)item;
421 #if EXIV2_TEST_VERSION(0,16,0)
422 if (exif->xmpIter != exif->xmpData().end())
424 const Exiv2::Metadatum *item = &*exif->xmpIter;
426 return (ExifItem *)item;
432 catch (Exiv2::AnyError& e) {
433 std::cout << "Caught Exiv2 exception '" << e << "'\n";
438 ExifItem *exif_get_next_item(ExifData *exif)
441 if (exif->exifIter != exif->exifData().end())
443 const Exiv2::Metadatum *item = &*exif->exifIter;
445 return (ExifItem *)item;
447 if (exif->iptcIter != exif->iptcData().end())
449 const Exiv2::Metadatum *item = &*exif->iptcIter;
451 return (ExifItem *)item;
453 #if EXIV2_TEST_VERSION(0,16,0)
454 if (exif->xmpIter != exif->xmpData().end())
456 const Exiv2::Metadatum *item = &*exif->xmpIter;
458 return (ExifItem *)item;
463 catch (Exiv2::AnyError& e) {
464 std::cout << "Caught Exiv2 exception '" << e << "'\n";
469 char *exif_item_get_tag_name(ExifItem *item)
472 if (!item) return NULL;
473 return g_strdup(((Exiv2::Metadatum *)item)->key().c_str());
475 catch (Exiv2::AnyError& e) {
476 std::cout << "Caught Exiv2 exception '" << e << "'\n";
481 guint exif_item_get_tag_id(ExifItem *item)
485 return ((Exiv2::Metadatum *)item)->tag();
487 catch (Exiv2::AnyError& e) {
488 std::cout << "Caught Exiv2 exception '" << e << "'\n";
493 guint exif_item_get_elements(ExifItem *item)
497 return ((Exiv2::Metadatum *)item)->count();
499 catch (Exiv2::AnyError& e) {
500 std::cout << "Caught Exiv2 exception '" << e << "'\n";
505 char *exif_item_get_data(ExifItem *item, guint *data_len)
509 Exiv2::Metadatum *md = (Exiv2::Metadatum *)item;
510 if (data_len) *data_len = md->size();
511 char *data = (char *)g_malloc(md->size());
512 long res = md->copy((Exiv2::byte *)data, Exiv2::littleEndian /* should not matter */);
513 g_assert(res == md->size());
516 catch (Exiv2::AnyError& e) {
517 std::cout << "Caught Exiv2 exception '" << e << "'\n";
522 char *exif_item_get_description(ExifItem *item)
525 if (!item) return NULL;
526 return g_locale_to_utf8(((Exiv2::Metadatum *)item)->tagLabel().c_str(), -1, NULL, NULL, NULL);
528 catch (std::exception& e) {
529 // std::cout << "Caught Exiv2 exception '" << e << "'\n";
535 invalidTypeId, unsignedByte, asciiString, unsignedShort,
536 unsignedLong, unsignedRational, signedByte, undefined,
537 signedShort, signedLong, signedRational, string,
538 date, time, comment, directory,
539 xmpText, xmpAlt, xmpBag, xmpSeq,
543 static guint format_id_trans_tbl [] = {
545 EXIF_FORMAT_BYTE_UNSIGNED,
547 EXIF_FORMAT_SHORT_UNSIGNED,
548 EXIF_FORMAT_LONG_UNSIGNED,
549 EXIF_FORMAT_RATIONAL_UNSIGNED,
551 EXIF_FORMAT_UNDEFINED,
554 EXIF_FORMAT_RATIONAL,
558 EXIF_FORMAT_UNDEFINED,
567 guint exif_item_get_format_id(ExifItem *item)
570 if (!item) return EXIF_FORMAT_UNKNOWN;
571 guint id = ((Exiv2::Metadatum *)item)->typeId();
572 if (id >= (sizeof(format_id_trans_tbl) / sizeof(format_id_trans_tbl[0])) ) return EXIF_FORMAT_UNKNOWN;
573 return format_id_trans_tbl[id];
575 catch (Exiv2::AnyError& e) {
576 std::cout << "Caught Exiv2 exception '" << e << "'\n";
577 return EXIF_FORMAT_UNKNOWN;
581 const char *exif_item_get_format_name(ExifItem *item, gint brief)
584 if (!item) return NULL;
585 return ((Exiv2::Metadatum *)item)->typeName();
587 catch (Exiv2::AnyError& e) {
588 std::cout << "Caught Exiv2 exception '" << e << "'\n";
594 gchar *exif_item_get_data_as_text(ExifItem *item)
597 if (!item) return NULL;
598 Exiv2::Metadatum *metadatum = (Exiv2::Metadatum *)item;
599 #if EXIV2_TEST_VERSION(0,17,0)
600 return g_locale_to_utf8(metadatum->print().c_str(), -1, NULL, NULL, NULL);
602 std::stringstream str;
603 Exiv2::Exifdatum *exifdatum;
604 Exiv2::Iptcdatum *iptcdatum;
605 #if EXIV2_TEST_VERSION(0,16,0)
606 Exiv2::Xmpdatum *xmpdatum;
608 if ((exifdatum = dynamic_cast<Exiv2::Exifdatum *>(metadatum)))
610 else if ((iptcdatum = dynamic_cast<Exiv2::Iptcdatum *>(metadatum)))
612 #if EXIV2_TEST_VERSION(0,16,0)
613 else if ((xmpdatum = dynamic_cast<Exiv2::Xmpdatum *>(metadatum)))
617 return g_locale_to_utf8(str.str().c_str(), -1, NULL, NULL, NULL);
620 catch (Exiv2::AnyError& e) {
625 gchar *exif_item_get_string(ExifItem *item, int idx)
628 if (!item) return NULL;
629 Exiv2::Metadatum *em = (Exiv2::Metadatum *)item;
630 #if EXIV2_TEST_VERSION(0,16,0)
631 std::string str = em->toString(idx);
633 std::string str = em->toString(); // FIXME
635 if (idx == 0 && str == "") str = em->toString();
636 if (str.length() > 5 && str.substr(0, 5) == "lang=")
638 std::string::size_type pos = str.find_first_of(' ');
639 if (pos != std::string::npos) str = str.substr(pos+1);
642 // return g_locale_to_utf8(str.c_str(), -1, NULL, NULL, NULL); // FIXME
643 return g_strdup(str.c_str());
645 catch (Exiv2::AnyError& e) {
651 gint exif_item_get_integer(ExifItem *item, gint *value)
655 *value = ((Exiv2::Metadatum *)item)->toLong();
658 catch (Exiv2::AnyError& e) {
659 std::cout << "Caught Exiv2 exception '" << e << "'\n";
664 ExifRational *exif_item_get_rational(ExifItem *item, gint *sign, guint n)
667 if (!item) return NULL;
668 if (n >= exif_item_get_elements(item)) return NULL;
669 Exiv2::Rational v = ((Exiv2::Metadatum *)item)->toRational(n);
670 static ExifRational ret;
673 if (sign) *sign = (((Exiv2::Metadatum *)item)->typeId() == Exiv2::signedRational);
676 catch (Exiv2::AnyError& e) {
677 std::cout << "Caught Exiv2 exception '" << e << "'\n";
682 gchar *exif_get_tag_description_by_key(const gchar *key)
685 Exiv2::ExifKey ekey(key);
686 return g_locale_to_utf8(Exiv2::ExifTags::tagLabel(ekey.tag(), ekey.ifdId ()), -1, NULL, NULL, NULL);
688 catch (Exiv2::AnyError& e) {
689 std::cout << "Caught Exiv2 exception '" << e << "'\n";
694 int exif_item_set_string(ExifItem *item, const char *str)
698 ((Exiv2::Metadatum *)item)->setValue(std::string(str));
701 catch (Exiv2::AnyError& e) {
706 int exif_item_delete(ExifData *exif, ExifItem *item)
710 for (Exiv2::ExifData::iterator i = exif->exifData().begin(); i != exif->exifData().end(); ++i) {
711 if (((Exiv2::Metadatum *)item) == &*i) {
712 i = exif->exifData().erase(i);
716 for (Exiv2::IptcData::iterator i = exif->iptcData().begin(); i != exif->iptcData().end(); ++i) {
717 if (((Exiv2::Metadatum *)item) == &*i) {
718 i = exif->iptcData().erase(i);
722 #if EXIV2_TEST_VERSION(0,16,0)
723 for (Exiv2::XmpData::iterator i = exif->xmpData().begin(); i != exif->xmpData().end(); ++i) {
724 if (((Exiv2::Metadatum *)item) == &*i) {
725 i = exif->xmpData().erase(i);
732 catch (Exiv2::AnyError& e) {
737 void exif_add_jpeg_color_profile(ExifData *exif, unsigned char *cp_data, guint cp_length)
739 exif->add_jpeg_color_profile(cp_data, cp_length);
742 guchar *exif_get_color_profile(ExifData *exif, guint *data_len)
744 guchar *ret = exif->get_jpeg_color_profile(data_len);
747 ExifItem *prof_item = exif_get_item(exif, "Exif.Image.InterColorProfile");
748 if (prof_item && exif_item_get_format_id(prof_item) == EXIF_FORMAT_UNDEFINED)
749 ret = (guchar *)exif_item_get_data(prof_item, data_len);
753 #if EXIV2_TEST_VERSION(0,17,90)
755 guchar *exif_get_preview(ExifData *exif, guint *data_len, gint requested_width, gint requested_height)
757 if (!exif) return NULL;
759 const char* path = exif->image()->io().path().c_str();
760 /* given image pathname, first do simple (and fast) file extension test */
761 gboolean is_raw = filter_file_class(path, FORMAT_CLASS_RAWIMAGE);
763 if (!is_raw && requested_width == 0) return NULL;
767 Exiv2::PreviewManager pm(*exif->image());
769 Exiv2::PreviewPropertiesList list = pm.getPreviewProperties();
773 Exiv2::PreviewPropertiesList::iterator pos;
774 Exiv2::PreviewPropertiesList::iterator last = --list.end();
776 if (requested_width == 0)
778 pos = last; // the largest
785 if (pos->width_ >= (uint32_t)requested_width &&
786 pos->height_ >= (uint32_t)requested_height) break;
790 // we are not interested in smaller thumbnails in normal image formats - we can use full image instead
793 if (pos->width_ < (uint32_t)requested_width || pos->height_ < (uint32_t)requested_height) return NULL;
797 Exiv2::PreviewImage image = pm.getPreviewImage(*pos);
799 Exiv2::DataBuf buf = image.copy();
800 std::pair<Exiv2::byte*, long> p = buf.release();
802 *data_len = p.second;
807 catch (Exiv2::AnyError& e) {
808 std::cout << "Caught Exiv2 exception '" << e << "'\n";
813 void exif_free_preview(guchar *buf)
815 delete[] (Exiv2::byte*)buf;
820 #if !EXIV2_TEST_VERSION(0,17,90)
822 /* This is a dirty hack to support raw file preview, bassed on
823 tiffparse.cpp from Exiv2 examples */
828 RawFile(Exiv2::BasicIo &io);
831 const Exiv2::Value *find(uint16_t tag, uint16_t group);
833 unsigned long preview_offset();
837 Exiv2::TiffComponent::AutoPtr rootDir;
839 const Exiv2::byte *map_data;
841 unsigned long offset;
844 typedef struct _UnmapData UnmapData;
852 static GList *exif_unmap_list = 0;
854 extern "C" guchar *exif_get_preview(ExifData *exif, guint *data_len, gint requested_width, gint requested_height)
856 unsigned long offset;
858 if (!exif) return NULL;
859 const char* path = exif->image()->io().path().c_str();
861 /* given image pathname, first do simple (and fast) file extension test */
862 if (!filter_file_class(path, FORMAT_CLASS_RAWIMAGE)) return NULL;
871 RawFile rf(exif->image()->io());
872 offset = rf.preview_offset();
873 DEBUG_1("%s: offset %lu", path, offset);
875 fd = open(path, O_RDONLY);
881 if (fstat(fd, &st) == -1)
886 map_len = st.st_size;
887 map_data = (guchar *) mmap(0, map_len, PROT_READ, MAP_PRIVATE, fd, 0);
889 if (map_data == MAP_FAILED)
893 *data_len = map_len - offset;
894 ud = g_new(UnmapData, 1);
895 ud->ptr = map_data + offset;
896 ud->map_data = map_data;
897 ud->map_len = map_len;
899 exif_unmap_list = g_list_prepend(exif_unmap_list, ud);
903 catch (Exiv2::AnyError& e) {
904 std::cout << "Caught Exiv2 exception '" << e << "'\n";
910 void exif_free_preview(guchar *buf)
912 GList *work = exif_unmap_list;
916 UnmapData *ud = (UnmapData *)work->data;
919 munmap(ud->map_data, ud->map_len);
920 exif_unmap_list = g_list_remove_link(exif_unmap_list, work);
926 g_assert_not_reached();
929 using namespace Exiv2;
931 RawFile::RawFile(BasicIo &io) : io_(io), map_data(NULL), map_len(0), offset(0)
935 if (fstat(fd, &st) == -1)
939 map_len = st.st_size;
940 map_data = (Exiv2::byte *) mmap(0, map_len, PROT_READ, MAP_PRIVATE, fd, 0);
941 if (map_data == MAP_FAILED)
946 if (io.open() != 0) {
947 throw Error(9, io.path(), strError());
950 map_data = io.mmap();
954 type = Exiv2::ImageFactory::getType(map_data, map_len);
956 #if EXIV2_TEST_VERSION(0,16,0)
957 TiffHeaderBase *tiffHeader = NULL;
959 TiffHeade2 *tiffHeader = NULL;
961 Cr2Header *cr2Header = NULL;
964 case Exiv2::ImageType::tiff:
965 tiffHeader = new TiffHeade2();
967 case Exiv2::ImageType::cr2:
968 cr2Header = new Cr2Header();
970 #if EXIV2_TEST_VERSION(0,16,0)
971 case Exiv2::ImageType::orf:
972 tiffHeader = new OrfHeader();
975 #if EXIV2_TEST_VERSION(0,13,0)
976 case Exiv2::ImageType::raf:
977 if (map_len < 84 + 4) throw Error(14);
978 offset = getULong(map_data + 84, bigEndian);
981 case Exiv2::ImageType::crw:
983 // Parse the image, starting with a CIFF header component
984 Exiv2::CiffHeader::AutoPtr parseTree(new Exiv2::CiffHeader);
985 parseTree->read(map_data, map_len);
986 CiffComponent *entry = parseTree->findComponent(0x2007, 0);
987 if (entry) offset = entry->pData() - map_data;
992 throw Error(3, "RAW");
995 // process tiff-like formats
997 TiffCompFactoryFct createFct = TiffCreator::create;
999 rootDir = createFct(Tag::root, Group::none);
1000 if (0 == rootDir.get()) {
1001 throw Error(1, "No root element defined in TIFF structure");
1006 if (!tiffHeader->read(map_data, map_len)) throw Error(3, "TIFF");
1007 #if EXIV2_TEST_VERSION(0,16,0)
1008 rootDir->setStart(map_data + tiffHeader->offset());
1010 rootDir->setStart(map_data + tiffHeader->ifdOffset());
1016 rootDir->setStart(map_data + cr2Header->offset());
1019 TiffRwState::AutoPtr state(new TiffRwState(tiffHeader ? tiffHeader->byteOrder() : littleEndian, 0, createFct));
1021 TiffReader reader(map_data,
1026 rootDir->accept(reader);
1034 RawFile::~RawFile(void)
1040 const Value * RawFile::find(uint16_t tag, uint16_t group)
1042 TiffFinder finder(tag, group);
1043 rootDir->accept(finder);
1044 TiffEntryBase* te = dynamic_cast<TiffEntryBase*>(finder.result());
1047 DEBUG_1("(tag: %04x %04x) ", tag, group);
1048 return te->pValue();
1054 unsigned long RawFile::preview_offset(void)
1057 if (offset) return offset;
1059 if (type == Exiv2::ImageType::cr2)
1061 val = find(0x111, Group::ifd0);
1062 if (val) return val->toLong();
1067 val = find(0x201, Group::sub0_0);
1068 if (val) return val->toLong();
1070 val = find(0x201, Group::ifd0);
1071 if (val) return val->toLong();
1073 val = find(0x201, Group::ignr); // for PEF files, originally it was probably ifd2
1074 if (val) return val->toLong();
1076 val = find(0x111, Group::sub0_1); // dng
1077 if (val) return val->toLong();
1088 /* vim: set shiftwidth=8 softtabstop=0 cindent cinoptions={1s: */