]> Pileus Git - ~andy/gtk/blobdiff - gdk-pixbuf/queryloaders.c
applied JPEG loader fix from maemo. this fix makes sure stop_load()
[~andy/gtk] / gdk-pixbuf / queryloaders.c
index 15c3bdb98f2dfa6c7abc082ffcebf1a30fb5ede5..11be6036ca7dc492a7043775ba3288eda7eebb9d 100644 (file)
 #include "gdk-pixbuf/gdk-pixbuf-private.h"
 #include "gdk-pixbuf/gdk-pixbuf-io.h"
 
-#if USE_LA_MODULES
+#ifdef USE_LA_MODULES
 #define SOEXT ".la"
 #else
 #define SOEXT ("." G_MODULE_SUFFIX)
 #endif
 #define SOEXT_LEN (strlen (SOEXT))
 
+#ifdef G_OS_WIN32
+#include <windows.h>
+#endif
+
 static void
 print_escaped (const char *str)
 {
@@ -53,6 +57,95 @@ print_escaped (const char *str)
        g_free (tmp);
 }
 
+static int
+loader_sanity_check (const char *path, GdkPixbufFormat *info, GdkPixbufModule *vtable)
+{
+       const GdkPixbufModulePattern *pattern;
+       const char *error = "";
+
+       for (pattern = info->signature; pattern->prefix; pattern++) 
+       {
+               int prefix_len = strlen (pattern->prefix);
+               if (prefix_len == 0) 
+               {
+                       error = "empty pattern";
+
+                       goto error;
+               }
+               if (pattern->mask)
+               {
+                       int mask_len = strlen (pattern->mask);
+                       if (mask_len != prefix_len)
+                       {
+                               error = "mask length mismatch";
+                               
+                               goto error;
+                       }
+                       if (strspn (pattern->mask, " !xzn*") < mask_len) 
+                       {
+                               error = "bad char in mask";
+                               
+                               goto error;
+                       }
+               }
+       }
+
+       if (!vtable->load && !vtable->begin_load && !vtable->load_animation)
+       {
+               error = "no load method implemented";
+
+               goto error;
+       }
+
+       if (vtable->begin_load && (!vtable->stop_load || !vtable->load_increment))
+       {
+               error = "incremental loading support incomplete";
+
+               goto error;
+       }
+
+       if ((info->flags & GDK_PIXBUF_FORMAT_WRITABLE) && !vtable->save) 
+       {
+               error = "loader claims to support saving but doesn't implement save";
+               goto error;
+       }
+           
+       return 1;
+
+ error:
+       g_fprintf (stderr, "Loader sanity check failed for %s: %s\n", 
+                  path, error);
+       
+       return 0;
+}
+
+static void 
+write_loader_info (const char *path, GdkPixbufFormat *info)
+{
+       const GdkPixbufModulePattern *pattern;
+       char **mime; 
+       char **ext; 
+
+       g_printf("\"%s\"\n", path);
+       g_printf ("\"%s\" %u \"%s\" \"%s\"\n", 
+                 info->name, info->flags, 
+                 info->domain ? info->domain : GETTEXT_PACKAGE, info->description);
+       for (mime = info->mime_types; *mime; mime++) {
+               g_printf ("\"%s\" ", *mime);
+       }
+       g_printf ("\"\"\n");
+       for (ext = info->extensions; *ext; ext++) {
+               g_printf ("\"%s\" ", *ext);
+       }
+       g_printf ("\"\"\n");
+       for (pattern = info->signature; pattern->prefix; pattern++) {
+               print_escaped (pattern->prefix);
+               print_escaped (pattern->mask ? (const char *)pattern->mask : "");
+               g_printf ("%d\n", pattern->relevance);
+       }
+       g_printf ("\n");
+}
+
 static void
 query_module (const char *dir, const char *file)
 {
@@ -60,9 +153,6 @@ query_module (const char *dir, const char *file)
        GModule *module;
        void                    (*fill_info)     (GdkPixbufFormat *info);
        void                    (*fill_vtable)   (GdkPixbufModule *module);
-       char **mime; 
-       char **ext; 
-       const GdkPixbufModulePattern *pattern;
 
        if (g_path_is_absolute (file)) 
                path = g_strdup (file);
@@ -74,27 +164,34 @@ query_module (const char *dir, const char *file)
            g_module_symbol (module, "fill_info", (gpointer *) &fill_info) &&
            g_module_symbol (module, "fill_vtable", (gpointer *) &fill_vtable)) {
                GdkPixbufFormat *info;
-               g_printf("\"%s\"\n", path);
+               GdkPixbufModule *vtable;
+               
+#ifdef G_OS_WIN32
+               /* Replace backslashes in path with forward slashes, so that
+                * it reads in without problems.
+                */
+               {
+                       char *p = path;
+                       while (*p) {
+                               if (*p == '\\')
+                                       *p = '/';
+                               p++;
+                       }
+               }
+#endif 
                info = g_new0 (GdkPixbufFormat, 1);
+               vtable = g_new0 (GdkPixbufModule, 1);
+
+               vtable->module = module;
+
                (*fill_info) (info);
-               g_printf ("\"%s\" %d \"%s\" \"%s\"\n", 
-                      info->name, info->flags, 
-                      info->domain ? info->domain : GETTEXT_PACKAGE, info->description);
-               for (mime = info->mime_types; *mime; mime++) {
-                       g_printf ("\"%s\" ", *mime);
-               }
-               g_printf ("\"\"\n");
-               for (ext = info->extensions; *ext; ext++) {
-                       g_printf ("\"%s\" ", *ext);
-               }
-               g_printf ("\"\"\n");
-               for (pattern = info->signature; pattern->prefix; pattern++) {
-                       print_escaped (pattern->prefix);
-                       print_escaped (pattern->mask ? (const char *)pattern->mask : "");
-                       g_printf ("%d\n", pattern->relevance);
-               }
-               g_printf ("\n");
+               (*fill_vtable) (vtable);
+               
+               if (loader_sanity_check (path, info, vtable)) 
+                       write_loader_info (path, info);
+               
                g_free (info);
+               g_free (vtable);
        }
        else {
                if (module == NULL)
@@ -111,10 +208,60 @@ query_module (const char *dir, const char *file)
 int main (int argc, char **argv)
 {
        gint i;
+       gchar *prgname;
 
+#ifdef G_OS_WIN32
+       gchar *libdir;
+       gchar *runtime_prefix;
+       gchar *slash;
+
+       if (g_ascii_strncasecmp (PIXBUF_LIBDIR, GTK_PREFIX, strlen (GTK_PREFIX)) == 0 &&
+           G_IS_DIR_SEPARATOR (PIXBUF_LIBDIR[strlen (GTK_PREFIX)])) {
+               /* GTK_PREFIX is a prefix of PIXBUF_LIBDIR, as it
+                * normally is. Replace that prefix in PIXBUF_LIBDIR
+                * with the installation directory on this machine.
+                * We assume this invokation of
+                * gdk-pixbuf-query-loaders is run from either a "bin"
+                * subdirectory of the installation directory, or in
+                * the installation directory itself.
+                */
+               if (G_WIN32_HAVE_WIDECHAR_API ()) {
+                       wchar_t fn[1000];
+                       GetModuleFileNameW (NULL, fn, G_N_ELEMENTS (fn));
+                       runtime_prefix = g_utf16_to_utf8 (fn, -1, NULL, NULL, NULL);
+               }
+               else {
+                       char fn[1000];
+                       GetModuleFileNameA (NULL, fn, G_N_ELEMENTS (fn));
+                       runtime_prefix = g_locale_to_utf8 (fn, -1, NULL, NULL, NULL);
+               }
+               slash = strrchr (runtime_prefix, '\\');
+               *slash = '\0';
+               slash = strrchr (runtime_prefix, '\\');
+               if (slash != NULL && g_ascii_strcasecmp (slash + 1, "bin") == 0) {
+                       *slash = '\0';
+               }
+               
+               libdir = g_strconcat (runtime_prefix,
+                                     "/",
+                                     PIXBUF_LIBDIR + strlen (GTK_PREFIX) + 1,
+                                     NULL);
+       }
+       else {
+               libdir = PIXBUF_LIBDIR;
+       }
+
+#undef PIXBUF_LIBDIR
+#define PIXBUF_LIBDIR libdir
+
+#endif
+       prgname = g_get_prgname ();
        g_printf ("# GdkPixbuf Image Loader Modules file\n"
-               "# Automatically generated file, do not edit\n"
-               "#\n");
+                 "# Automatically generated file, do not edit\n"
+                 "# Created by %s from gtk+-%s\n"
+                 "#\n",
+                 (prgname ? prgname : "gdk-pixbuf-query-loaders"),
+                 GDK_PIXBUF_VERSION);
   
        if (argc == 1) {
 #ifdef USE_GMODULE
@@ -122,6 +269,10 @@ int main (int argc, char **argv)
                GDir *dir;
     
                path = g_getenv ("GDK_PIXBUF_MODULEDIR");
+#ifdef G_OS_WIN32
+               if (path != NULL && *path != '\0')
+                       path = g_locale_to_utf8 (path, -1, NULL, NULL, NULL);
+#endif
                if (path == NULL || *path == '\0')
                        path = PIXBUF_LIBDIR;
 
@@ -147,9 +298,14 @@ int main (int argc, char **argv)
        else {
                char *cwd = g_get_current_dir ();
 
-               for (i = 1; i < argc; i++)
-                       query_module (cwd, argv[i]);
-
+               for (i = 1; i < argc; i++) {
+                       char *infilename = argv[i];
+#ifdef G_OS_WIN32
+                       infilename = g_locale_to_utf8 (infilename,
+                                                      -1, NULL, NULL, NULL);
+#endif
+                       query_module (cwd, infilename);
+               }
                g_free (cwd);
        }