os/ossrv/glib/gmodule/gmodule.c
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
     1 /* GMODULE - GLIB wrapper code for dynamic module loading
     2  * Copyright (C) 1998 Tim Janik
     3  * Portions copyright (c) 2006-2009 Nokia Corporation.  All rights reserved.
     4  *
     5  * This library is free software; you can redistribute it and/or
     6  * modify it under the terms of the GNU Lesser General Public
     7  * License as published by the Free Software Foundation; either
     8  * version 2 of the License, or (at your option) any later version.
     9  *
    10  * This library is distributed in the hope that it will be useful,
    11  * but WITHOUT ANY WARRANTY; without even the implied warranty of
    12  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.	 See the GNU
    13  * Lesser General Public License for more details.
    14  *
    15  * You should have received a copy of the GNU Lesser General Public
    16  * License along with this library; if not, write to the
    17  * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
    18  * Boston, MA 02111-1307, USA.
    19  */
    20 
    21 /*
    22  * Modified by the GLib Team and others 1997-2000.  See the AUTHORS
    23  * file for a list of people on the GLib Team.  See the ChangeLog
    24  * files for a list of changes.  These files are distributed with
    25  * GLib at ftp://ftp.gtk.org/pub/gtk/. 
    26  */
    27 
    28 /* 
    29  * MT safe
    30  */
    31 
    32 #include "config.h"
    33 
    34 #include "glib.h"
    35 #include "gmodule.h"
    36 
    37 #include <errno.h>
    38 #include <string.h>
    39 #include <sys/types.h>
    40 #include <sys/stat.h>
    41 #include <fcntl.h>
    42 #ifdef HAVE_UNISTD_H
    43 #include <unistd.h>
    44 #endif
    45 #ifdef G_OS_WIN32
    46 #include <io.h>		/* For open() and close() prototypes. */
    47 #endif
    48 
    49 #include "gmoduleconf.h"
    50 #include "gstdio.h" 
    51 
    52 #ifdef __SYMBIAN32__
    53 #include "gmodule_wsd.h"
    54 #endif /* __SYMBIAN32__ */
    55 
    56 /* We maintain a list of modules, so we can reference count them.
    57  * That's needed because some platforms don't support refernce counts on
    58  * modules e.g. the shl_* implementation of HP-UX
    59  * (http://www.stat.umn.edu/~luke/xls/projects/dlbasics/dlbasics.html).
    60  * Also, the module for the program itself is kept seperatedly for
    61  * faster access and because it has special semantics.
    62  */
    63 
    64 
    65 /* --- structures --- */
    66 struct _GModule
    67 {
    68   gchar	*file_name;
    69 #if defined (G_OS_WIN32) && !defined(_WIN64)
    70   gchar *cp_file_name;
    71 #endif
    72   gpointer handle;
    73   guint ref_count : 31;
    74   guint is_resident : 1;
    75   GModuleUnload unload;
    76   GModule *next;
    77 };
    78 
    79 
    80 /* --- prototypes --- */
    81 static gpointer		_g_module_open		(const gchar	*file_name,
    82 						 gboolean	 bind_lazy,
    83 						 gboolean	 bind_local);
    84 static void		_g_module_close		(gpointer	 handle,
    85 						 gboolean	 is_unref);
    86 static gpointer		_g_module_self		(void);
    87 static gpointer		_g_module_symbol	(gpointer	 handle,
    88 						 const gchar	*symbol_name);
    89 static gchar*		_g_module_build_path	(const gchar	*directory,
    90 						 const gchar	*module_name);
    91 static inline void	g_module_set_error	(const gchar	*error);
    92 static inline GModule*	g_module_find_by_handle (gpointer	 handle);
    93 static inline GModule*	g_module_find_by_name	(const gchar	*name);
    94 
    95 
    96 /* --- variables --- */
    97 #if EMULATOR
    98 
    99 PLS(modules,gmodule,GModule *)
   100 PLS(main_module,gmodule,GModule *)
   101 PLS(module_error_private,gmodule,GStaticPrivate)
   102 PLS(module_debug_initialized,gmodule,gboolean)
   103 PLS(module_debug_flags,gmodule,guint)
   104 
   105 #define modules (*FUNCTION_NAME(modules,gmodule)())
   106 #define main_module (*FUNCTION_NAME(main_module,gmodule)())
   107 #define module_error_private (*FUNCTION_NAME(module_error_private,gmodule)())
   108 #define module_debug_initialized (*FUNCTION_NAME(module_debug_initialized,gmodule)())
   109 #define module_debug_flags (*FUNCTION_NAME(module_debug_flags,gmodule)())
   110 
   111 #else
   112 static GModule	     *modules = NULL;
   113 static GModule	     *main_module = NULL;
   114 static GStaticPrivate module_error_private = G_STATIC_PRIVATE_INIT;
   115 static gboolean	      module_debug_initialized = FALSE;
   116 static guint	      module_debug_flags = 0;
   117 #endif /* EMULATOR */
   118 
   119 /* --- inline functions --- */
   120 static inline GModule*
   121 g_module_find_by_handle (gpointer handle)
   122 {
   123   GModule *module;
   124   GModule *retval = NULL;
   125   
   126   if (main_module && main_module->handle == handle)
   127     retval = main_module;
   128   else
   129     for (module = modules; module; module = module->next)
   130       if (handle == module->handle)
   131 	{
   132 	  retval = module;
   133 	  break;
   134 	}
   135 
   136   return retval;
   137 }
   138 
   139 static inline GModule*
   140 g_module_find_by_name (const gchar *name)
   141 {
   142   GModule *module;
   143   GModule *retval = NULL;
   144   
   145   for (module = modules; module; module = module->next)
   146     if (strcmp (name, module->file_name) == 0)
   147 	{
   148 	  retval = module;
   149 	  break;
   150 	}
   151 
   152   return retval;
   153 }
   154 
   155 static inline void
   156 g_module_set_error_unduped (gchar *error)
   157 {
   158   g_static_private_set (&module_error_private, error, g_free);
   159   errno = 0;
   160 }
   161 
   162 static inline void
   163 g_module_set_error (const gchar *error)
   164 {
   165   g_module_set_error_unduped (g_strdup (error));
   166 }
   167 
   168 
   169 /* --- include platform specifc code --- */
   170 #define	SUPPORT_OR_RETURN(rv)	{ g_module_set_error (NULL); }
   171 #if	(G_MODULE_IMPL == G_MODULE_IMPL_DL)
   172 #include "gmodule-dl.c"
   173 #elif	(G_MODULE_IMPL == G_MODULE_IMPL_DLD)
   174 #include "gmodule-dld.c"
   175 #elif	(G_MODULE_IMPL == G_MODULE_IMPL_WIN32)
   176 #include "gmodule-win32.c"
   177 #elif	(G_MODULE_IMPL == G_MODULE_IMPL_DYLD)
   178 #include "gmodule-dyld.c"
   179 #elif	(G_MODULE_IMPL == G_MODULE_IMPL_AR)
   180 #include "gmodule-ar.c"
   181 #else
   182 #undef	SUPPORT_OR_RETURN
   183 #define	SUPPORT_OR_RETURN(rv)	{ g_module_set_error ("dynamic modules are " \
   184                                               "not supported by this system"); return rv; }
   185 static gpointer
   186 _g_module_open (const gchar	*file_name,
   187 		gboolean	 bind_lazy,
   188 		gboolean	 bind_local)
   189 {
   190   return NULL;
   191 }
   192 static void
   193 _g_module_close	(gpointer	 handle,
   194 		 gboolean	 is_unref)
   195 {
   196 }
   197 static gpointer
   198 _g_module_self (void)
   199 {
   200   return NULL;
   201 }
   202 static gpointer
   203 _g_module_symbol (gpointer	 handle,
   204 		  const gchar	*symbol_name)
   205 {
   206   return NULL;
   207 }
   208 static gchar*
   209 _g_module_build_path (const gchar *directory,
   210 		      const gchar *module_name)
   211 {
   212   return NULL;
   213 }
   214 #endif	/* no implementation */
   215 
   216 /* --- functions --- */
   217 EXPORT_C gboolean
   218 g_module_supported (void)
   219 {
   220   SUPPORT_OR_RETURN (FALSE);
   221   
   222   return TRUE;
   223 }
   224 
   225 static gchar*
   226 parse_libtool_archive (const gchar* libtool_name)
   227 {
   228   const guint TOKEN_DLNAME = G_TOKEN_LAST + 1;
   229   const guint TOKEN_INSTALLED = G_TOKEN_LAST + 2;
   230   const guint TOKEN_LIBDIR = G_TOKEN_LAST + 3;
   231   gchar *lt_dlname = NULL;
   232   gboolean lt_installed = TRUE;
   233   gchar *lt_libdir = NULL;
   234   gchar *name;
   235   GTokenType token;
   236   GScanner *scanner;
   237   
   238   int fd = g_open (libtool_name, O_RDONLY, 0);
   239   if (fd < 0)
   240     {
   241       gchar *display_libtool_name = g_filename_display_name (libtool_name);
   242       g_module_set_error_unduped (g_strdup_printf ("failed to open libtool archive \"%s\"", display_libtool_name));
   243       g_free (display_libtool_name);
   244       return NULL;
   245     }
   246   /* search libtool's dlname specification  */
   247   scanner = g_scanner_new (NULL);
   248   g_scanner_input_file (scanner, fd);
   249   scanner->config->symbol_2_token = TRUE;
   250   g_scanner_scope_add_symbol (scanner, 0, "dlname", 
   251 			      GUINT_TO_POINTER (TOKEN_DLNAME));
   252   g_scanner_scope_add_symbol (scanner, 0, "installed", 
   253 			      GUINT_TO_POINTER (TOKEN_INSTALLED));
   254   g_scanner_scope_add_symbol (scanner, 0, "libdir", 
   255 			      GUINT_TO_POINTER (TOKEN_LIBDIR));
   256   while (!g_scanner_eof (scanner))
   257     {
   258       token = g_scanner_get_next_token (scanner);
   259       if (token == TOKEN_DLNAME || token == TOKEN_INSTALLED || 
   260 	  token == TOKEN_LIBDIR)
   261 	{
   262 	  if (g_scanner_get_next_token (scanner) != '=' ||
   263 	      g_scanner_get_next_token (scanner) != 
   264 	      (token == TOKEN_INSTALLED ? 
   265 	       G_TOKEN_IDENTIFIER : G_TOKEN_STRING))
   266 	    {
   267 	      gchar *display_libtool_name = g_filename_display_name (libtool_name);
   268 	      g_module_set_error_unduped (g_strdup_printf ("unable to parse libtool archive \"%s\"", display_libtool_name));
   269 	      g_free (display_libtool_name);
   270 
   271 	      g_free (lt_dlname);
   272 	      g_free (lt_libdir);
   273 	      g_scanner_destroy (scanner);
   274 	      close (fd);
   275 
   276 	      return NULL;
   277 	    }
   278 	  else
   279 	    {
   280 	      if (token == TOKEN_DLNAME)
   281 		{
   282 		  g_free (lt_dlname);
   283 		  lt_dlname = g_strdup (scanner->value.v_string);
   284 		}
   285 	      else if (token == TOKEN_INSTALLED)
   286 		lt_installed = 
   287 		  strcmp (scanner->value.v_identifier, "yes") == 0;
   288 	      else /* token == TOKEN_LIBDIR */
   289 		{
   290 		  g_free (lt_libdir);
   291 		  lt_libdir = g_strdup (scanner->value.v_string);
   292 		}
   293 	    }
   294 	}      
   295     }
   296 
   297   if (!lt_installed)
   298     {
   299       gchar *dir = g_path_get_dirname (libtool_name);
   300       g_free (lt_libdir);
   301       lt_libdir = g_strconcat (dir, G_DIR_SEPARATOR_S ".libs", NULL);
   302       g_free (dir);
   303     }
   304 
   305   name = g_strconcat (lt_libdir, G_DIR_SEPARATOR_S, lt_dlname, NULL);
   306   
   307   g_free (lt_dlname);
   308   g_free (lt_libdir);
   309   g_scanner_destroy (scanner);
   310   close (fd);
   311 
   312   return name;
   313 }
   314 
   315 static inline gboolean
   316 str_check_suffix (const gchar* string,
   317 		  const gchar* suffix)
   318 {
   319   gsize string_len = strlen (string);    
   320   gsize suffix_len = strlen (suffix);    
   321 
   322   return string_len >= suffix_len && 
   323     strcmp (string + string_len - suffix_len, suffix) == 0;
   324 }
   325 
   326 enum
   327 {
   328   G_MODULE_DEBUG_RESIDENT_MODULES = 1 << 0,
   329   G_MODULE_DEBUG_BIND_NOW_MODULES = 1 << 1
   330 };
   331 
   332 static void
   333 _g_module_debug_init (void)
   334 {
   335   const GDebugKey keys[] = {
   336     { "resident-modules", G_MODULE_DEBUG_RESIDENT_MODULES },
   337     { "bind-now-modules", G_MODULE_DEBUG_BIND_NOW_MODULES }
   338   };
   339   const gchar *env;
   340 
   341   env = g_getenv ("G_DEBUG");
   342 
   343   module_debug_flags =
   344     !env ? 0 : g_parse_debug_string (env, keys, G_N_ELEMENTS (keys));
   345 
   346   module_debug_initialized = TRUE;
   347 }
   348 
   349 #if EMULATOR
   350 PLS(g_module_global_lock,gmodule,GStaticRecMutex)
   351 #define g_module_global_lock (*FUNCTION_NAME(g_module_global_lock,gmodule)())
   352 #else
   353 static GStaticRecMutex g_module_global_lock = G_STATIC_REC_MUTEX_INIT;
   354 #endif /* EMULATOR */
   355 
   356 EXPORT_C GModule*
   357 g_module_open (const gchar    *file_name,
   358 	       GModuleFlags    flags)
   359 {
   360   GModule *module;
   361   gpointer handle = NULL;
   362   gchar *name = NULL;
   363   
   364   SUPPORT_OR_RETURN (NULL);
   365   
   366   g_static_rec_mutex_lock (&g_module_global_lock);
   367 
   368   if (G_UNLIKELY (!module_debug_initialized))
   369     _g_module_debug_init ();
   370 
   371   if (module_debug_flags & G_MODULE_DEBUG_BIND_NOW_MODULES)
   372     flags &= ~G_MODULE_BIND_LAZY;
   373 
   374   if (!file_name)
   375     {      
   376       if (!main_module)
   377 	{
   378 	  handle = _g_module_self ();
   379 	  if (handle)
   380 	    {
   381 	      main_module = g_new (GModule, 1);
   382 	      main_module->file_name = NULL;
   383 #if defined (G_OS_WIN32) && !defined(_WIN64)
   384 	      main_module->cp_file_name = NULL;
   385 #endif
   386 	      main_module->handle = handle;
   387 	      main_module->ref_count = 1;
   388 	      main_module->is_resident = TRUE;
   389 	      main_module->unload = NULL;
   390 	      main_module->next = NULL;
   391 	    }
   392 	}
   393       else
   394 	main_module->ref_count++;
   395 
   396       g_static_rec_mutex_unlock (&g_module_global_lock);
   397       return main_module;
   398     }
   399   
   400   /* we first search the module list by name */
   401   module = g_module_find_by_name (file_name);
   402   if (module)
   403     {
   404       module->ref_count++;
   405       
   406       g_static_rec_mutex_unlock (&g_module_global_lock);
   407       return module;
   408     }
   409 
   410   /* check whether we have a readable file right away */
   411   if (g_file_test (file_name, G_FILE_TEST_IS_REGULAR))
   412     name = g_strdup (file_name);
   413   /* try completing file name with standard library suffix */
   414   if (!name)
   415     {
   416       name = g_strconcat (file_name, "." G_MODULE_SUFFIX, NULL);
   417       if (!g_file_test (name, G_FILE_TEST_IS_REGULAR))
   418 	{
   419 	  g_free (name);
   420 	  name = NULL;
   421 	}
   422     }
   423   /* try completing by appending libtool suffix */
   424   if (!name)
   425     {
   426       name = g_strconcat (file_name, ".la", NULL);
   427       if (!g_file_test (name, G_FILE_TEST_IS_REGULAR))
   428 	{
   429 	  g_free (name);
   430 	  name = NULL;
   431 	}
   432     }
   433   /* we can't access() the file, lets hope the platform backends finds
   434    * it via library paths
   435    */
   436   if (!name)
   437     {
   438       gchar *dot = strrchr (file_name, '.');
   439       gchar *slash = strrchr (file_name, G_DIR_SEPARATOR);
   440       
   441       /* make sure the name has a suffix */
   442       if (!dot || dot < slash)
   443 	name = g_strconcat (file_name, "." G_MODULE_SUFFIX, NULL);
   444       else
   445 	name = g_strdup (file_name);
   446     }
   447 
   448   /* ok, try loading the module */
   449   if (name)
   450     {
   451       /* if it's a libtool archive, figure library file to load */
   452       if (str_check_suffix (name, ".la")) /* libtool archive? */
   453 	{
   454 	  gchar *real_name = parse_libtool_archive (name);
   455 
   456 	  /* real_name might be NULL, but then module error is already set */
   457 	  if (real_name)
   458 	    {
   459 	      g_free (name);
   460 	      name = real_name;
   461             }
   462 	}
   463       if (name)
   464 	handle = _g_module_open (name, (flags & G_MODULE_BIND_LAZY) != 0,
   465 			(flags & G_MODULE_BIND_LOCAL) != 0);
   466     }
   467   else
   468     {
   469       gchar *display_file_name = g_filename_display_name (file_name);
   470       g_module_set_error_unduped (g_strdup_printf ("unable to access file \"%s\"", display_file_name));
   471       g_free (display_file_name);
   472     }
   473   g_free (name);
   474 
   475   if (handle)
   476     {
   477       gchar *saved_error;
   478       GModuleCheckInit check_init;
   479       const gchar *check_failed = NULL;
   480       
   481       /* search the module list by handle, since file names are not unique */
   482       module = g_module_find_by_handle (handle);
   483       if (module)
   484 	{
   485 	  _g_module_close (module->handle, TRUE);
   486 	  module->ref_count++;
   487 	  g_module_set_error (NULL);
   488 	  
   489 	  g_static_rec_mutex_unlock (&g_module_global_lock);
   490 	  return module;
   491 	}
   492       
   493       saved_error = g_strdup (g_module_error ());
   494       g_module_set_error (NULL);
   495       
   496       module = g_new (GModule, 1);
   497       module->file_name = g_strdup (file_name);
   498 #if defined (G_OS_WIN32) && !defined(_WIN64)
   499       module->cp_file_name = g_locale_from_utf8 (file_name, -1,
   500 						 NULL, NULL, NULL);
   501 #endif
   502       module->handle = handle;
   503       module->ref_count = 1;
   504       module->is_resident = FALSE;
   505       module->unload = NULL;
   506       module->next = modules;
   507       modules = module;
   508       
   509       /* check initialization */
   510       if (g_module_symbol (module, "g_module_check_init", (gpointer) &check_init) && check_init != NULL)
   511 	check_failed = check_init (module);
   512       
   513       /* we don't call unload() if the initialization check failed. */
   514       if (!check_failed)
   515 	g_module_symbol (module, "g_module_unload", (gpointer) &module->unload);
   516       
   517       if (check_failed)
   518 	{
   519 	  gchar *error;
   520 
   521 	  error = g_strconcat ("GModule (", 
   522                                file_name ? file_name : "NULL", 
   523                                ") initialization check failed: ", 
   524                                check_failed, NULL);
   525 	  g_module_close (module);
   526 	  module = NULL;
   527 	  g_module_set_error (error);
   528 	  g_free (error);
   529 	}
   530       else
   531 	g_module_set_error (saved_error);
   532 
   533       g_free (saved_error);
   534     }
   535 
   536   if (module != NULL &&
   537       (module_debug_flags & G_MODULE_DEBUG_RESIDENT_MODULES))
   538     g_module_make_resident (module);
   539 
   540   g_static_rec_mutex_unlock (&g_module_global_lock);
   541   return module;
   542 }
   543 
   544 #if defined (G_OS_WIN32) && !defined(_WIN64)
   545 
   546 #undef g_module_open
   547 
   548 GModule*
   549 g_module_open (const gchar    *file_name,
   550 	       GModuleFlags    flags)
   551 {
   552   gchar *utf8_file_name = g_locale_to_utf8 (file_name, -1, NULL, NULL, NULL);
   553   GModule *retval = g_module_open_utf8 (utf8_file_name, flags);
   554 
   555   g_free (utf8_file_name);
   556 
   557   return retval;
   558 }
   559 
   560 #endif
   561 
   562 EXPORT_C gboolean
   563 g_module_close (GModule	       *module)
   564 {
   565   SUPPORT_OR_RETURN (FALSE);
   566   
   567   g_return_val_if_fail (module != NULL, FALSE);
   568   g_return_val_if_fail (module->ref_count > 0, FALSE);
   569   
   570   g_static_rec_mutex_lock (&g_module_global_lock);
   571 
   572   module->ref_count--;
   573   
   574   if (!module->ref_count && !module->is_resident && module->unload)
   575     {
   576       GModuleUnload unload;
   577 
   578       unload = module->unload;
   579       module->unload = NULL;
   580       unload (module);
   581     }
   582 
   583   if (!module->ref_count && !module->is_resident)
   584     {
   585       GModule *last;
   586       GModule *node;
   587       
   588       last = NULL;
   589       
   590       node = modules;
   591       while (node)
   592 	{
   593 	  if (node == module)
   594 	    {
   595 	      if (last)
   596 		last->next = node->next;
   597 	      else
   598 		modules = node->next;
   599 	      break;
   600 	    }
   601 	  last = node;
   602 	  node = last->next;
   603 	}
   604       module->next = NULL;
   605       
   606       _g_module_close (module->handle, FALSE);
   607       g_free (module->file_name);
   608 #if defined (G_OS_WIN32) && !defined(_WIN64)
   609       g_free (module->cp_file_name);
   610 #endif
   611       g_free (module);
   612     }
   613   
   614   g_static_rec_mutex_unlock (&g_module_global_lock);
   615   return g_module_error() == NULL;
   616 }
   617 
   618 EXPORT_C void
   619 g_module_make_resident (GModule *module)
   620 {
   621   g_return_if_fail (module != NULL);
   622 
   623   module->is_resident = TRUE;
   624 }
   625 
   626 EXPORT_C G_CONST_RETURN gchar*
   627 g_module_error (void)
   628 {
   629   return g_static_private_get (&module_error_private);
   630 }
   631 
   632 EXPORT_C gboolean
   633 g_module_symbol (GModule	*module,
   634 		 const gchar	*symbol_name,
   635 		 gpointer	*symbol)
   636 {
   637   const gchar *module_error;
   638 
   639   if (symbol)
   640     *symbol = NULL;
   641   SUPPORT_OR_RETURN (FALSE);
   642   
   643   g_return_val_if_fail (module != NULL, FALSE);
   644   g_return_val_if_fail (symbol_name != NULL, FALSE);
   645   g_return_val_if_fail (symbol != NULL, FALSE);
   646   
   647   g_static_rec_mutex_lock (&g_module_global_lock);
   648 
   649 #ifdef	G_MODULE_NEED_USCORE
   650   {
   651     gchar *name;
   652 
   653     name = g_strconcat ("_", symbol_name, NULL);
   654     *symbol = _g_module_symbol (module->handle, name);
   655     g_free (name);
   656   }
   657 #else	/* !G_MODULE_NEED_USCORE */
   658   *symbol = _g_module_symbol (module->handle, symbol_name);
   659 #endif	/* !G_MODULE_NEED_USCORE */
   660   
   661   module_error = g_module_error ();
   662   if (module_error)
   663     {
   664       gchar *error;
   665 
   666       error = g_strconcat ("`", symbol_name, "': ", module_error, NULL);
   667       g_module_set_error (error);
   668       g_free (error);
   669       *symbol = NULL;
   670     }
   671   
   672   g_static_rec_mutex_unlock (&g_module_global_lock);
   673   return !module_error;
   674 }
   675 
   676 EXPORT_C G_CONST_RETURN gchar*
   677 g_module_name (GModule *module)
   678 {
   679   g_return_val_if_fail (module != NULL, NULL);
   680   
   681   if (module == main_module)
   682     return "main";
   683   
   684   return module->file_name;
   685 }
   686 
   687 #if defined (G_OS_WIN32) && !defined(_WIN64)
   688 
   689 #undef g_module_name
   690 
   691 G_CONST_RETURN gchar*
   692 g_module_name (GModule *module)
   693 {
   694   g_return_val_if_fail (module != NULL, NULL);
   695   
   696   if (module == main_module)
   697     return "main";
   698   
   699   return module->cp_file_name;
   700 }
   701 
   702 #endif
   703 
   704 EXPORT_C gchar*
   705 g_module_build_path (const gchar *directory,
   706 		     const gchar *module_name)
   707 {
   708   g_return_val_if_fail (module_name != NULL, NULL);
   709   
   710   return _g_module_build_path (directory, module_name);
   711 }
   712 
   713 #define __GMODULE_C__