diff options
Diffstat (limited to 'src/tracker-extract/tracker-extract-png.c')
-rw-r--r-- | src/tracker-extract/tracker-extract-png.c | 729 |
1 files changed, 0 insertions, 729 deletions
diff --git a/src/tracker-extract/tracker-extract-png.c b/src/tracker-extract/tracker-extract-png.c deleted file mode 100644 index 0e1976541..000000000 --- a/src/tracker-extract/tracker-extract-png.c +++ /dev/null @@ -1,729 +0,0 @@ -/* - * Copyright (C) 2006, Jamie McCracken <jamiemcc@gnome.org> - * Copyright (C) 2008, Nokia <ivan.frade@nokia.com> - * - * This library is free software; you can redistribute it and/or - * modify it under the terms of the GNU Lesser General Public - * License as published by the Free Software Foundation; either - * version 2.1 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 - * Lesser General Public License for more details. - * - * You should have received a copy of the GNU Lesser General Public - * License along with this library; if not, write to the - * Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, - * Boston, MA 02110-1301, USA. - */ - -#include "config.h" - -#include <png.h> - -#include <libtracker-common/tracker-file-utils.h> -#include <libtracker-common/tracker-date-time.h> -#include <libtracker-extract/tracker-extract.h> - -#define RFC1123_DATE_FORMAT "%d %B %Y %H:%M:%S %z" -#define CM_TO_INCH 0.393700787 - -typedef struct { - const gchar *title; - const gchar *copyright; - const gchar *creator; - const gchar *description; - const gchar *date; - const gchar *license; - const gchar *artist; - const gchar *make; - const gchar *model; - const gchar *orientation; - const gchar *white_balance; - const gchar *fnumber; - const gchar *flash; - const gchar *focal_length; - const gchar *exposure_time; - const gchar *iso_speed_ratings; - const gchar *metering_mode; - const gchar *comment; - const gchar *city; - const gchar *state; - const gchar *address; - const gchar *country; - const gchar *gps_direction; -} MergeData; - -typedef struct { - const gchar *author; - const gchar *creator; - const gchar *description; - const gchar *comment; - const gchar *copyright; - gchar *creation_time; - const gchar *title; - const gchar *disclaimer; - const gchar *software; -} PngData; - -static gchar * -rfc1123_to_iso8601_date (const gchar *date) -{ - /* From: ex. RFC1123 date: "22 May 1997 18:07:10 -0600" - * To : ex. ISO8601 date: "2007-05-22T18:07:10-0600" - */ - return tracker_date_format_to_iso8601 (date, RFC1123_DATE_FORMAT); -} - -#if defined(PNG_iTXt_SUPPORTED) && (defined(HAVE_EXEMPI) || defined(HAVE_LIBEXIF)) - -/* Handle raw profiles by Imagemagick (at least). Hex encoded with - * line-changes and other (undocumented/unofficial) twists. - */ -static gchar * -raw_profile_new (const gchar *input, - const guint input_length, - guint *output_length) -{ - static const gchar* const lut = "0123456789abcdef"; - gchar *output; - const gchar *ptr; - const gchar *length_ptr; - gsize size; - gchar *length_str; - guint length; - - size_t len; - size_t i; - size_t o; - char *p; - char *q; - - ptr = input; - - if (*ptr != '\n') { - return NULL; - } - - ptr++; - - if (!g_ascii_isalpha (*ptr)) { - return NULL; - } - - /* Skip the type string */ - do { - ptr++; - } while (g_ascii_isalpha (*ptr)); - - if (*ptr != '\n') { - return NULL; - } - - /* Hop over whitespaces */ - do { - ptr++; - } while (*ptr == ' '); - - if (!g_ascii_isdigit (*ptr)) { - return NULL; - } - - /* Get the length string */ - length_ptr = ptr; - size = 1; - - do { - ptr++; - size++; - } while (g_ascii_isdigit (*ptr)); - - length_str = g_strndup (length_ptr, size - 1); - - if (*ptr != '\n') { - return NULL; - } - - ptr++; - - length = atoi (length_str); - g_free (length_str); - - len = length; - i = 0; - o = 0; - - output = malloc (length + 1); /* A bit less with non-valid */ - - o = 0; - while (o < len) { - do { - gchar a = ptr[i]; - p = strchr (lut, a); - i++; - } while (p == 0); - - do { - gchar b = ptr[i]; - q = strchr (lut, b); - i++; - } while (q == 0); - - output[o] = (((p - lut) << 4) | (q - lut)); - o++; - } - - output[o] = '\0'; - *output_length = o; - - return output; -} - -#endif /* defined(PNG_iTXt_SUPPORTED) && (defined(HAVE_EXEMPI) || defined(HAVE_LIBEXIF)) */ - -static void -read_metadata (TrackerResource *metadata, - png_structp png_ptr, - png_infop info_ptr, - png_infop end_ptr, - const gchar *uri) -{ - MergeData md = { 0 }; - PngData pd = { 0 }; - TrackerExifData *ed = NULL; - TrackerXmpData *xd = NULL; - png_infop info_ptrs[2]; - png_textp text_ptr; - gint info_index; - gint num_text; - gint i; - gint found; - GPtrArray *keywords; - - info_ptrs[0] = info_ptr; - info_ptrs[1] = end_ptr; - - for (info_index = 0; info_index < 2; info_index++) { - if ((found = png_get_text (png_ptr, info_ptrs[info_index], &text_ptr, &num_text)) < 1) { - g_debug ("Calling png_get_text() returned %d (< 1)", found); - continue; - } - - for (i = 0; i < num_text; i++) { - if (!text_ptr[i].key || !text_ptr[i].text || text_ptr[i].text[0] == '\0') { - continue; - } - -#if defined(HAVE_EXEMPI) && defined(PNG_iTXt_SUPPORTED) - if (g_strcmp0 ("XML:com.adobe.xmp", text_ptr[i].key) == 0) { - /* ATM tracker_extract_xmp_read supports setting xd - * multiple times, keep it that way as here it's - * theoretically possible that the function gets - * called multiple times - */ - xd = tracker_xmp_new (text_ptr[i].text, - text_ptr[i].itxt_length, - uri); - - continue; - } - - if (g_strcmp0 ("Raw profile type xmp", text_ptr[i].key) == 0) { - gchar *xmp_buffer; - guint xmp_buffer_length = 0; - guint input_len; - - if (text_ptr[i].text_length) { - input_len = text_ptr[i].text_length; - } else { - input_len = text_ptr[i].itxt_length; - } - - xmp_buffer = raw_profile_new (text_ptr[i].text, - input_len, - &xmp_buffer_length); - - if (xmp_buffer) { - xd = tracker_xmp_new (xmp_buffer, - xmp_buffer_length, - uri); - } - - g_free (xmp_buffer); - - continue; - } -#endif /*HAVE_EXEMPI && PNG_iTXt_SUPPORTED */ - -#if defined(HAVE_LIBEXIF) && defined(PNG_iTXt_SUPPORTED) - if (g_strcmp0 ("Raw profile type exif", text_ptr[i].key) == 0) { - gchar *exif_buffer; - guint exif_buffer_length = 0; - guint input_len; - - if (text_ptr[i].text_length) { - input_len = text_ptr[i].text_length; - } else { - input_len = text_ptr[i].itxt_length; - } - - exif_buffer = raw_profile_new (text_ptr[i].text, - input_len, - &exif_buffer_length); - - if (exif_buffer) { - ed = tracker_exif_new (exif_buffer, - exif_buffer_length, - uri); - } - - g_free (exif_buffer); - - continue; - } -#endif /* HAVE_LIBEXIF && PNG_iTXt_SUPPORTED */ - - if (g_strcmp0 (text_ptr[i].key, "Author") == 0) { - pd.author = text_ptr[i].text; - continue; - } - - if (g_strcmp0 (text_ptr[i].key, "Creator") == 0) { - pd.creator = text_ptr[i].text; - continue; - } - - if (g_strcmp0 (text_ptr[i].key, "Description") == 0) { - pd.description = text_ptr[i].text; - continue; - } - - if (g_strcmp0 (text_ptr[i].key, "Comment") == 0) { - pd.comment = text_ptr[i].text; - continue; - } - - if (g_strcmp0 (text_ptr[i].key, "Copyright") == 0) { - pd.copyright = text_ptr[i].text; - continue; - } - - if (g_strcmp0 (text_ptr[i].key, "Creation Time") == 0) { - pd.creation_time = rfc1123_to_iso8601_date (text_ptr[i].text); - continue; - } - - if (g_strcmp0 (text_ptr[i].key, "Title") == 0) { - pd.title = text_ptr[i].text; - continue; - } - - if (g_strcmp0 (text_ptr[i].key, "Disclaimer") == 0) { - pd.disclaimer = text_ptr[i].text; - continue; - } - - if (g_strcmp0(text_ptr[i].key, "Software") == 0) { - pd.software = text_ptr[i].text; - continue; - } - } - } - - if (!ed) { - ed = g_new0 (TrackerExifData, 1); - } - - if (!xd) { - xd = g_new0 (TrackerXmpData, 1); - } - - md.creator = tracker_coalesce_strip (3, xd->creator, pd.creator, pd.author); - md.title = tracker_coalesce_strip (5, xd->title, pd.title, ed->document_name, xd->title2, xd->pdf_title); - md.copyright = tracker_coalesce_strip (3, xd->rights, pd.copyright, ed->copyright); - md.license = tracker_coalesce_strip (2, xd->license, pd.disclaimer); - md.description = tracker_coalesce_strip (3, xd->description, pd.description, ed->description); - md.date = tracker_coalesce_strip (5, xd->date, xd->time_original, pd.creation_time, ed->time, ed->time_original); - md.comment = tracker_coalesce_strip (2, pd.comment, ed->user_comment); - md.artist = tracker_coalesce_strip (3, xd->artist, ed->artist, xd->contributor); - md.orientation = tracker_coalesce_strip (2, xd->orientation, ed->orientation); - md.exposure_time = tracker_coalesce_strip (2, xd->exposure_time, ed->exposure_time); - md.iso_speed_ratings = tracker_coalesce_strip (2, xd->iso_speed_ratings, ed->iso_speed_ratings); - md.fnumber = tracker_coalesce_strip (2, xd->fnumber, ed->fnumber); - md.flash = tracker_coalesce_strip (2, xd->flash, ed->flash); - md.focal_length = tracker_coalesce_strip (2, xd->focal_length, ed->focal_length); - md.metering_mode = tracker_coalesce_strip (2, xd->metering_mode, ed->metering_mode); - md.white_balance = tracker_coalesce_strip (2, xd->white_balance, ed->white_balance); - md.make = tracker_coalesce_strip (2, xd->make, ed->make); - md.model = tracker_coalesce_strip (2, xd->model, ed->model); - - keywords = g_ptr_array_new_with_free_func ((GDestroyNotify) g_free); - - if (md.comment) { - tracker_resource_set_string (metadata, "nie:comment", md.comment); - } - - if (md.license) { - tracker_resource_set_string (metadata, "nie:license", md.license); - } - - /* TODO: add ontology and store this ed->software */ - - if (md.creator) { - TrackerResource *creator = tracker_extract_new_contact (md.creator); - - tracker_resource_set_relation (metadata, "nco:creator", creator); - - g_object_unref (creator); - } - - tracker_guarantee_resource_date_from_file_mtime (metadata, - "nie:contentCreated", - md.date, - uri); - - if (md.description) { - tracker_resource_set_string (metadata, "nie:description", md.description); - } - - if (md.copyright) { - tracker_resource_set_string (metadata, "nie:copyright", md.copyright); - } - - tracker_guarantee_resource_title_from_file (metadata, - "nie:title", - md.title, - uri, - NULL); - - if (md.make || md.model) { - TrackerResource *equipment = tracker_extract_new_equipment (md.make, md.model); - - tracker_resource_set_relation (metadata, "nfo:equipment", equipment); - - g_object_unref (equipment); - } - - if (md.artist) { - TrackerResource *artist = tracker_extract_new_contact (md.artist); - - tracker_resource_set_relation (metadata, "nco:contributor", artist); - - g_object_unref (artist); - } - - if (md.orientation) { - TrackerResource *orientation; - - orientation = tracker_resource_new (md.orientation); - tracker_resource_set_relation (metadata, "nfo:orientation", orientation); - g_object_unref (orientation); - } - - if (md.exposure_time) { - tracker_resource_set_string (metadata, "nmm:exposureTime", md.exposure_time); - } - - if (md.iso_speed_ratings) { - tracker_resource_set_string (metadata, "nmm:isoSpeed", md.iso_speed_ratings); - } - - if (md.white_balance) { - TrackerResource *white_balance; - - white_balance = tracker_resource_new (md.white_balance); - tracker_resource_set_relation (metadata, "nmm:meteringMode", white_balance); - g_object_unref (white_balance); - } - - if (md.fnumber) { - tracker_resource_set_string (metadata, "nmm:fnumber", md.fnumber); - } - - if (md.flash) { - TrackerResource *flash; - - flash = tracker_resource_new (md.flash); - tracker_resource_set_relation (metadata, "nmm:flash", flash); - g_object_unref (flash); - } - - if (md.focal_length) { - tracker_resource_set_string (metadata, "nmm:focalLength", md.focal_length); - } - - if (md.metering_mode) { - TrackerResource *metering; - - metering = tracker_resource_new (md.metering_mode); - tracker_resource_set_relation (metadata, "nmm:meteringMode", metering); - g_object_unref (metering); - } - - if (xd->keywords) { - tracker_keywords_parse (keywords, xd->keywords); - } - - if (xd->pdf_keywords) { - tracker_keywords_parse (keywords, xd->pdf_keywords); - } - - if (xd->rating) { - tracker_resource_set_string (metadata, "nao:numericRating", xd->rating); - } - - if (xd->subject) { - tracker_keywords_parse (keywords, xd->subject); - } - - if (xd->publisher) { - TrackerResource *publisher = tracker_extract_new_contact (xd->publisher); - - tracker_resource_set_relation (metadata, "nco:creator", publisher); - - g_object_unref (publisher); - } - - if (xd->type) { - tracker_resource_set_string (metadata, "dc:type", xd->type); - } - - if (xd->format) { - tracker_resource_set_string (metadata, "dc:format", xd->format); - } - - if (xd->identifier) { - tracker_resource_set_string (metadata, "dc:identifier", xd->identifier); - } - - if (xd->source) { - tracker_resource_set_string (metadata, "dc:source", xd->source); - } - - if (xd->language) { - tracker_resource_set_string (metadata, "dc:language", xd->language); - } - - if (xd->relation) { - tracker_resource_set_string (metadata, "dc:relation", xd->relation); - } - - if (xd->coverage) { - tracker_resource_set_string (metadata, "dc:coverage", xd->coverage); - } - - if (xd->address || xd->state || xd->country || xd->city || - xd->gps_altitude || xd->gps_latitude || xd-> gps_longitude) { - - TrackerResource *location = tracker_extract_new_location (xd->address, - xd->state, xd->city, xd->country, xd->gps_altitude, - xd->gps_latitude, xd-> gps_longitude); - - tracker_resource_set_relation (metadata, "slo:location", location); - - g_object_unref (location); - } - - if (xd->gps_direction) { - tracker_resource_set_string(metadata, "nfo:heading", xd->gps_direction); - } - - if (ed->x_resolution) { - gdouble value; - - value = ed->resolution_unit != 3 ? g_strtod (ed->x_resolution, NULL) : g_strtod (ed->x_resolution, NULL) * CM_TO_INCH; - tracker_resource_set_double (metadata, "nfo:horizontalResolution", value); - } - - if (ed->y_resolution) { - gdouble value; - - value = ed->resolution_unit != 3 ? g_strtod (ed->y_resolution, NULL) : g_strtod (ed->y_resolution, NULL) * CM_TO_INCH; - tracker_resource_set_double (metadata, "nfo:verticalResolution", value); - } - - if (xd->regions) { - tracker_xmp_apply_regions_to_resource (metadata, xd); - } - - for (i = 0; i < keywords->len; i++) { - TrackerResource *tag; - const gchar *p; - - p = g_ptr_array_index (keywords, i); - - tag = tracker_extract_new_tag (p); - - tracker_resource_set_relation (metadata, "nao:hasTag", tag); - - g_object_unref (tag); - } - g_ptr_array_free (keywords, TRUE); - - if (g_strcmp0(pd.software, "gnome-screenshot") == 0) { - tracker_resource_add_uri (metadata, "nie:isPartOf", "nfo:image-category-screenshot"); - } - - tracker_exif_free (ed); - tracker_xmp_free (xd); - g_free (pd.creation_time); -} - -static gboolean -guess_dlna_profile (gint depth, - gint width, - gint height, - const gchar **dlna_profile, - const gchar **dlna_mimetype) -{ - const gchar *profile = NULL; - - if (dlna_profile) { - *dlna_profile = NULL; - } - - if (dlna_mimetype) { - *dlna_mimetype = NULL; - } - - if (width == 120 && height == 120) { - profile = "PNG_LRG_ICO"; - } else if (width == 48 && height == 48) { - profile = "PNG_SM_ICO"; - } else if (width <= 160 && height <= 160) { - profile = "PNG_TN"; - } else if (depth <= 32 && width <= 4096 && height <= 4096) { - profile = "PNG_LRG"; - } - - if (profile) { - if (dlna_profile) { - *dlna_profile = profile; - } - - if (dlna_mimetype) { - *dlna_mimetype = "image/png"; - } - - return TRUE; - } - - return FALSE; -} - -G_MODULE_EXPORT gboolean -tracker_extract_get_metadata (TrackerExtractInfo *info) -{ - TrackerResource *metadata; - goffset size; - FILE *f; - png_structp png_ptr; - png_infop info_ptr; - png_infop end_ptr; - png_bytep row_data; - guint row; - png_uint_32 width, height; - gint bit_depth, color_type; - gint interlace_type, compression_type, filter_type; - const gchar *dlna_profile, *dlna_mimetype; - gchar *filename, *uri; - GFile *file; - - file = tracker_extract_info_get_file (info); - filename = g_file_get_path (file); - size = tracker_file_get_size (filename); - - if (size < 64) { - return FALSE; - } - - f = tracker_file_open (filename); - g_free (filename); - - if (!f) { - return FALSE; - } - - png_ptr = png_create_read_struct (PNG_LIBPNG_VER_STRING, - NULL, - NULL, - NULL); - if (!png_ptr) { - tracker_file_close (f, FALSE); - return FALSE; - } - - info_ptr = png_create_info_struct (png_ptr); - if (!info_ptr) { - png_destroy_read_struct (&png_ptr, NULL, NULL); - tracker_file_close (f, FALSE); - return FALSE; - } - - end_ptr = png_create_info_struct (png_ptr); - if (!end_ptr) { - png_destroy_read_struct (&png_ptr, &info_ptr, NULL); - tracker_file_close (f, FALSE); - return FALSE; - } - - if (setjmp (png_jmpbuf (png_ptr))) { - png_destroy_read_struct (&png_ptr, &info_ptr, &end_ptr); - tracker_file_close (f, FALSE); - return FALSE; - } - - png_init_io (png_ptr, f); - png_read_info (png_ptr, info_ptr); - - if (!png_get_IHDR (png_ptr, - info_ptr, - &width, - &height, - &bit_depth, - &color_type, - &interlace_type, - &compression_type, - &filter_type)) { - png_destroy_read_struct (&png_ptr, &info_ptr, &end_ptr); - tracker_file_close (f, FALSE); - return FALSE; - } - - /* Read the image. FIXME We should be able to skip this step and - * just get the info from the end. This causes some errors atm. - */ - row_data = png_malloc (png_ptr, png_get_rowbytes (png_ptr, info_ptr)); - for (row = 0; row < height; row++) - png_read_row (png_ptr, row_data, NULL); - png_free (png_ptr, row_data); - - png_read_end (png_ptr, end_ptr); - - metadata = tracker_resource_new (NULL); - - tracker_resource_add_uri (metadata, "rdf:type", "nfo:Image"); - tracker_resource_add_uri (metadata, "rdf:type", "nmm:Photo"); - - uri = g_file_get_uri (file); - - read_metadata (metadata, png_ptr, info_ptr, end_ptr, uri); - g_free (uri); - - tracker_resource_set_int64 (metadata, "nfo:width", width); - tracker_resource_set_int64 (metadata, "nfo:height", height); - - if (guess_dlna_profile (bit_depth, width, height, &dlna_profile, &dlna_mimetype)) { - tracker_resource_set_string (metadata, "nmm:dlnaProfile", dlna_profile); - tracker_resource_set_string (metadata, "nmm:dlnaMime", dlna_mimetype); - } - - png_destroy_read_struct (&png_ptr, &info_ptr, &end_ptr); - tracker_file_close (f, FALSE); - - tracker_extract_info_set_resource (info, metadata); - g_object_unref (metadata); - - return TRUE; -} |