forked from AuroraMiddleware/gtk
6ba75ff95b
2004-12-05 Tor Lillqvist <tml@iki.fi> * gdk-pixbuf-animation.c * gdk-pixbuf-io.c * io-xbm.c * io-xpm.c * make-inline-pixbuf.c * queryloaders.c: Use gstdio wrappers. Document that file names are in the GLib file name encoding. * gdk-pixbuf-csource.c * queryloaders.c: On Windows, convert command line arguments and environment variable values from locale encoding to UTF-8. * queryloaders.c: On Windows, use wide character API when available. * Makefile.am * gdk-pixbuf-core.h * gdk-pixbuf-io.c * gdk-pixbuf-animation.h * gdk-pixbuf-animation.c: Like in GLib, for DLL ABI stability on Windows, add binary compatibility versions of functions that take file names as arguments. They use the system codepage, not GLib file name encoding (which is UTF-8 on Windows). Use #defines to make newly compiled code use the "real" functions that use the GLib file name encoding scheme.
310 lines
7.2 KiB
C
310 lines
7.2 KiB
C
/* -*- mode: C; c-file-style: "linux" -*- */
|
|
/* GdkPixbuf library
|
|
* queryloaders.c:
|
|
*
|
|
* Copyright (C) 2002 The Free Software Foundation
|
|
*
|
|
* Author: Matthias Clasen <maclas@gmx.de>
|
|
*
|
|
* 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; if not, write to the
|
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
|
* Boston, MA 02111-1307, USA.
|
|
*/
|
|
|
|
#include <config.h>
|
|
|
|
#include <glib.h>
|
|
#include <glib/gprintf.h>
|
|
#include <gmodule.h>
|
|
|
|
#include <errno.h>
|
|
#include <string.h>
|
|
#ifdef HAVE_UNISTD_H
|
|
#include <unistd.h>
|
|
#endif
|
|
|
|
#include "gdk-pixbuf/gdk-pixbuf.h"
|
|
#include "gdk-pixbuf/gdk-pixbuf-private.h"
|
|
#include "gdk-pixbuf/gdk-pixbuf-io.h"
|
|
|
|
#if 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)
|
|
{
|
|
gchar *tmp = g_strescape (str, "");
|
|
g_printf ("\"%s\" ", tmp);
|
|
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\" %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");
|
|
}
|
|
|
|
static void
|
|
query_module (const char *dir, const char *file)
|
|
{
|
|
char *path;
|
|
GModule *module;
|
|
void (*fill_info) (GdkPixbufFormat *info);
|
|
void (*fill_vtable) (GdkPixbufModule *module);
|
|
|
|
if (g_path_is_absolute (file))
|
|
path = g_strdup (file);
|
|
else
|
|
path = g_build_filename (dir, file, NULL);
|
|
|
|
module = g_module_open (path, 0);
|
|
if (module &&
|
|
g_module_symbol (module, "fill_info", (gpointer *) &fill_info) &&
|
|
g_module_symbol (module, "fill_vtable", (gpointer *) &fill_vtable)) {
|
|
GdkPixbufFormat *info;
|
|
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);
|
|
(*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)
|
|
g_fprintf (stderr, "g_module_open() failed for %s: %s\n", path,
|
|
g_module_error());
|
|
else
|
|
g_fprintf (stderr, "Cannot load loader %s\n", path);
|
|
}
|
|
if (module)
|
|
g_module_close (module);
|
|
g_free (path);
|
|
}
|
|
|
|
int main (int argc, char **argv)
|
|
{
|
|
gint i;
|
|
|
|
#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
|
|
|
|
g_printf ("# GdkPixbuf Image Loader Modules file\n"
|
|
"# Automatically generated file, do not edit\n"
|
|
"#\n");
|
|
|
|
if (argc == 1) {
|
|
#ifdef USE_GMODULE
|
|
const char *path;
|
|
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;
|
|
|
|
g_printf ("# LoaderDir = %s\n#\n", path);
|
|
|
|
dir = g_dir_open (path, 0, NULL);
|
|
if (dir) {
|
|
const char *dent;
|
|
|
|
while ((dent = g_dir_read_name (dir))) {
|
|
gint len = strlen (dent);
|
|
if (len > SOEXT_LEN &&
|
|
strcmp (dent + len - SOEXT_LEN, SOEXT) == 0) {
|
|
query_module (path, dent);
|
|
}
|
|
}
|
|
g_dir_close (dir);
|
|
}
|
|
#else
|
|
g_printf ("# dynamic loading of modules not supported\n");
|
|
#endif
|
|
}
|
|
else {
|
|
char *cwd = g_get_current_dir ();
|
|
|
|
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);
|
|
}
|
|
|
|
return 0;
|
|
}
|