queryloaders.c 7.37 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24
/* -*- 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.
 */

25
#include "config.h"
26 27

#include <glib.h>
28
#include <glib/gprintf.h>
29 30 31 32 33 34 35 36 37 38 39 40
#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"

Matthias Clasen's avatar
Matthias Clasen committed
41
#ifdef USE_LA_MODULES
42 43 44 45 46 47
#define SOEXT ".la"
#else
#define SOEXT ("." G_MODULE_SUFFIX)
#endif
#define SOEXT_LEN (strlen (SOEXT))

48 49 50 51
#ifdef G_OS_WIN32
#include <windows.h>
#endif

52 53 54 55
static void
print_escaped (const char *str)
{
	gchar *tmp = g_strescape (str, "");
56
	g_printf ("\"%s\" ", tmp);
57 58 59
	g_free (tmp);
}

60 61 62 63 64 65
static int
loader_sanity_check (const char *path, GdkPixbufFormat *info, GdkPixbufModule *vtable)
{
	const GdkPixbufModulePattern *pattern;
	const char *error = "";

66 67 68 69
	for (pattern = info->signature; pattern->prefix; pattern++) 
	{
		int prefix_len = strlen (pattern->prefix);
		if (prefix_len == 0) 
70 71 72 73 74
		{
			error = "empty pattern";

			goto error;
		}
75 76 77 78 79 80 81 82 83
		if (pattern->mask)
		{
			int mask_len = strlen (pattern->mask);
			if (mask_len != prefix_len)
			{
				error = "mask length mismatch";
				
				goto error;
			}
Matthias Clasen's avatar
Matthias Clasen committed
84
			if (strspn (pattern->mask, " !xzn*") < mask_len) 
85 86 87 88 89 90 91
			{
				error = "bad char in mask";
				
				goto error;
			}
		}
	}
92 93 94 95 96 97 98 99 100 101 102 103 104 105 106

	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;
	}

107
	if ((info->flags & GDK_PIXBUF_FORMAT_WRITABLE) && !(vtable->save || vtable->save_to_callback)) 
108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129
	{
		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);
130
	g_printf ("\"%s\" %u \"%s\" \"%s\" \"%s\"\n", 
131
		  info->name, info->flags, 
132
		  info->domain ? info->domain : GETTEXT_PACKAGE, info->description, info->license);
133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148
	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");
}

149 150 151 152 153 154 155 156
static void
query_module (const char *dir, const char *file)
{
	char *path;
	GModule *module;
	void                    (*fill_info)     (GdkPixbufFormat *info);
	void                    (*fill_vtable)   (GdkPixbufModule *module);

157 158 159
	if (g_path_is_absolute (file)) 
		path = g_strdup (file);
	else
160 161 162 163 164 165
		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)) {
jacob berkman's avatar
jacob berkman committed
166
		GdkPixbufFormat *info;
167 168
		GdkPixbufModule *vtable;
		
169 170 171 172 173 174 175 176 177 178 179 180 181
#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	
jacob berkman's avatar
jacob berkman committed
182
		info = g_new0 (GdkPixbufFormat, 1);
183
		vtable = g_new0 (GdkPixbufModule, 1);
184 185 186

		vtable->module = module;

187
		(*fill_info) (info);
188 189 190 191 192
		(*fill_vtable) (vtable);
		
		if (loader_sanity_check (path, info, vtable)) 
			write_loader_info (path, info);
		
193
		g_free (info);
194
		g_free (vtable);
195 196
	}
	else {
197 198 199 200 201
		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);
202 203 204 205 206 207 208 209 210
	}
	if (module)
		g_module_close (module);
	g_free (path);
}

int main (int argc, char **argv)
{
	gint i;
211
	gchar *prgname;
212

213
#ifdef G_OS_WIN32
214 215
	gchar *libdir;
	gchar *runtime_prefix;
216 217 218
	gchar *slash;

	if (g_ascii_strncasecmp (PIXBUF_LIBDIR, GTK_PREFIX, strlen (GTK_PREFIX)) == 0 &&
219
	    G_IS_DIR_SEPARATOR (PIXBUF_LIBDIR[strlen (GTK_PREFIX)])) {
220 221 222 223 224 225
		/* 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
226
		 * the installation directory itself.
227
		 */
228 229 230 231 232 233 234 235 236 237
		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);
		}
238 239 240 241 242 243 244
		slash = strrchr (runtime_prefix, '\\');
		*slash = '\0';
		slash = strrchr (runtime_prefix, '\\');
		if (slash != NULL && g_ascii_strcasecmp (slash + 1, "bin") == 0) {
			*slash = '\0';
		}
		
245 246 247 248 249 250 251
		libdir = g_strconcat (runtime_prefix,
				      "/",
				      PIXBUF_LIBDIR + strlen (GTK_PREFIX) + 1,
				      NULL);
	}
	else {
		libdir = PIXBUF_LIBDIR;
252 253 254 255 256 257
	}

#undef PIXBUF_LIBDIR
#define PIXBUF_LIBDIR libdir

#endif
258
	prgname = g_get_prgname ();
259
	g_printf ("# GdkPixbuf Image Loader Modules file\n"
260 261 262 263 264
		  "# Automatically generated file, do not edit\n"
		  "# Created by %s from gtk+-%s\n"
		  "#\n",
		  (prgname ? prgname : "gdk-pixbuf-query-loaders"),
		  GDK_PIXBUF_VERSION);
265 266 267 268
  
	if (argc == 1) {
#ifdef USE_GMODULE
		const char *path;
Manish Singh's avatar
Manish Singh committed
269
		GDir *dir;
270 271
    
		path = g_getenv ("GDK_PIXBUF_MODULEDIR");
272 273 274 275
#ifdef G_OS_WIN32
		if (path != NULL && *path != '\0')
			path = g_locale_to_utf8 (path, -1, NULL, NULL, NULL);
#endif
276 277 278
		if (path == NULL || *path == '\0')
			path = PIXBUF_LIBDIR;

279
		g_printf ("# LoaderDir = %s\n#\n", path);
280

Manish Singh's avatar
Manish Singh committed
281
		dir = g_dir_open (path, 0, NULL);
282 283 284 285 286 287 288 289 290 291 292 293 294
		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
295
		g_printf ("# dynamic loading of modules not supported\n");
296 297 298 299 300
#endif
	}
	else {
		char *cwd = g_get_current_dir ();

301 302 303 304 305 306 307 308
		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);
		}
309 310 311 312 313
		g_free (cwd);
	}

	return 0;
}