os/ossrv/glib/gmodule/gmodule-dl.c
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
sl@0
     1
/* GMODULE - GLIB wrapper code for dynamic module loading
sl@0
     2
 * Copyright (C) 1998, 2000 Tim Janik
sl@0
     3
 * Portions copyright (c) 2006-2009 Nokia Corporation.  All rights reserved.
sl@0
     4
 * This library is free software; you can redistribute it and/or
sl@0
     5
 * modify it under the terms of the GNU Lesser General Public
sl@0
     6
 * License as published by the Free Software Foundation; either
sl@0
     7
 * version 2 of the License, or (at your option) any later version.
sl@0
     8
 *
sl@0
     9
 * This library is distributed in the hope that it will be useful,
sl@0
    10
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
sl@0
    11
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.	 See the GNU
sl@0
    12
 * Lesser General Public License for more details.
sl@0
    13
 *
sl@0
    14
 * You should have received a copy of the GNU Lesser General Public
sl@0
    15
 * License along with this library; if not, write to the
sl@0
    16
 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
sl@0
    17
 * Boston, MA 02111-1307, USA.
sl@0
    18
 */
sl@0
    19
sl@0
    20
/*
sl@0
    21
 * Modified by the GLib Team and others 1997-2000.  See the AUTHORS
sl@0
    22
 * file for a list of people on the GLib Team.  See the ChangeLog
sl@0
    23
 * files for a list of changes.  These files are distributed with
sl@0
    24
 * GLib at ftp://ftp.gtk.org/pub/gtk/. 
sl@0
    25
 */
sl@0
    26
sl@0
    27
/* 
sl@0
    28
 * MT safe
sl@0
    29
 */
sl@0
    30
#include "config.h"
sl@0
    31
sl@0
    32
#include <dlfcn.h>
sl@0
    33
sl@0
    34
/* Perl includes <nlist.h> and <link.h> instead of <dlfcn.h> on some systmes? */
sl@0
    35
sl@0
    36
sl@0
    37
/* dlerror() is not implemented on all systems
sl@0
    38
 */
sl@0
    39
#ifndef	G_MODULE_HAVE_DLERROR
sl@0
    40
#  ifdef __NetBSD__
sl@0
    41
#    define dlerror()	g_strerror (errno)
sl@0
    42
#  else /* !__NetBSD__ */
sl@0
    43
/* could we rely on errno's state here? */
sl@0
    44
#    define dlerror()	"unknown dl-error"
sl@0
    45
#  endif /* !__NetBSD__ */
sl@0
    46
#endif	/* G_MODULE_HAVE_DLERROR */
sl@0
    47
sl@0
    48
/* some flags are missing on some systems, so we provide
sl@0
    49
 * harmless defaults.
sl@0
    50
 * The Perl sources say, RTLD_LAZY needs to be defined as (1),
sl@0
    51
 * at least for Solaris 1.
sl@0
    52
 *
sl@0
    53
 * Mandatory:
sl@0
    54
 * RTLD_LAZY   - resolve undefined symbols as code from the dynamic library
sl@0
    55
 *		 is executed.
sl@0
    56
 * RTLD_NOW    - resolve all undefined symbols before dlopen returns, and fail
sl@0
    57
 *		 if this cannot be done.
sl@0
    58
 * Optionally:
sl@0
    59
 * RTLD_GLOBAL - the external symbols defined in the library will be made
sl@0
    60
 *		 available to subsequently loaded libraries.
sl@0
    61
 */
sl@0
    62
#ifndef	RTLD_LAZY
sl@0
    63
#define	RTLD_LAZY	1
sl@0
    64
#endif	/* RTLD_LAZY */
sl@0
    65
#ifndef	RTLD_NOW
sl@0
    66
#define	RTLD_NOW	0
sl@0
    67
#endif	/* RTLD_NOW */
sl@0
    68
/* some systems (OSF1 V5.0) have broken RTLD_GLOBAL linkage */
sl@0
    69
#ifdef G_MODULE_BROKEN_RTLD_GLOBAL
sl@0
    70
#undef	RTLD_GLOBAL
sl@0
    71
#endif /* G_MODULE_BROKEN_RTLD_GLOBAL */
sl@0
    72
#ifndef	RTLD_GLOBAL
sl@0
    73
#define	RTLD_GLOBAL	0
sl@0
    74
#endif	/* RTLD_GLOBAL */
sl@0
    75
sl@0
    76
sl@0
    77
/* --- functions --- */
sl@0
    78
static gchar*
sl@0
    79
fetch_dlerror (gboolean replace_null)
sl@0
    80
{
sl@0
    81
  gchar *msg = dlerror ();
sl@0
    82
sl@0
    83
  /* make sure we always return an error message != NULL, if
sl@0
    84
   * expected to do so. */
sl@0
    85
sl@0
    86
  if (!msg && replace_null)
sl@0
    87
    return "unknown dl-error";
sl@0
    88
sl@0
    89
  return msg;
sl@0
    90
}
sl@0
    91
sl@0
    92
static gpointer
sl@0
    93
_g_module_open (const gchar *file_name,
sl@0
    94
		gboolean     bind_lazy,
sl@0
    95
		gboolean     bind_local)
sl@0
    96
{
sl@0
    97
  gpointer handle;
sl@0
    98
  
sl@0
    99
  handle = dlopen (file_name,
sl@0
   100
		   (bind_local ? 0 : RTLD_GLOBAL) | (bind_lazy ? RTLD_LAZY : RTLD_NOW));
sl@0
   101
  if (!handle)
sl@0
   102
    g_module_set_error (fetch_dlerror (TRUE));
sl@0
   103
  
sl@0
   104
  return handle;
sl@0
   105
}
sl@0
   106
sl@0
   107
static gpointer
sl@0
   108
_g_module_self (void)
sl@0
   109
{
sl@0
   110
  gpointer handle;
sl@0
   111
  
sl@0
   112
  /* to query symbols from the program itself, special link options
sl@0
   113
   * are required on some systems.
sl@0
   114
   */
sl@0
   115
  
sl@0
   116
  handle = dlopen (NULL, RTLD_GLOBAL | RTLD_LAZY);
sl@0
   117
  if (!handle)
sl@0
   118
    g_module_set_error (fetch_dlerror (TRUE));
sl@0
   119
  
sl@0
   120
  return handle;
sl@0
   121
}
sl@0
   122
sl@0
   123
static void
sl@0
   124
_g_module_close (gpointer handle,
sl@0
   125
		 gboolean is_unref)
sl@0
   126
{
sl@0
   127
  /* are there any systems out there that have dlopen()/dlclose()
sl@0
   128
   * without a reference count implementation?
sl@0
   129
   */
sl@0
   130
  is_unref |= 1;
sl@0
   131
  
sl@0
   132
  if (is_unref)
sl@0
   133
    {
sl@0
   134
      if (dlclose (handle) != 0)
sl@0
   135
	g_module_set_error (fetch_dlerror (TRUE));
sl@0
   136
    }
sl@0
   137
}
sl@0
   138
sl@0
   139
static gpointer
sl@0
   140
_g_module_symbol (gpointer     handle,
sl@0
   141
		  const gchar *symbol_name)
sl@0
   142
{
sl@0
   143
  gpointer p;
sl@0
   144
  gchar *msg;
sl@0
   145
sl@0
   146
  fetch_dlerror (FALSE);
sl@0
   147
  p = dlsym (handle, symbol_name);
sl@0
   148
  msg = fetch_dlerror (FALSE);
sl@0
   149
  if (msg)
sl@0
   150
    g_module_set_error (msg);
sl@0
   151
  
sl@0
   152
  return p;
sl@0
   153
}
sl@0
   154
sl@0
   155
static gchar*
sl@0
   156
_g_module_build_path (const gchar *directory,
sl@0
   157
		      const gchar *module_name)
sl@0
   158
{
sl@0
   159
  if (directory && *directory) {
sl@0
   160
    #ifdef __SYMBIAN32__
sl@0
   161
	return g_strconcat (directory, "\\", module_name, "." G_MODULE_SUFFIX, NULL);
sl@0
   162
    #else
sl@0
   163
    if (strncmp (module_name, "lib", 3) == 0)
sl@0
   164
      return g_strconcat (directory, "/", module_name, NULL);
sl@0
   165
    else
sl@0
   166
      return g_strconcat (directory, "/lib", module_name, "." G_MODULE_SUFFIX, NULL);
sl@0
   167
    #endif /* __SYMBIAN32__ */
sl@0
   168
  } else if (strncmp (module_name, "lib", 3) == 0)
sl@0
   169
    return g_strdup (module_name);
sl@0
   170
  else
sl@0
   171
    #ifdef __SYMBIAN32__
sl@0
   172
    return g_strconcat (module_name, "." G_MODULE_SUFFIX, NULL);
sl@0
   173
    #else
sl@0
   174
    return g_strconcat ("lib", module_name, "." G_MODULE_SUFFIX, NULL);
sl@0
   175
    #endif /* __SYMBIAN32__ */
sl@0
   176
}