Mercurial > hg > xemacs-beta
diff src/sysdll.c @ 428:3ecd8885ac67 r21-2-22
Import from CVS: tag r21-2-22
author | cvs |
---|---|
date | Mon, 13 Aug 2007 11:28:15 +0200 |
parents | |
children | a5df635868b2 |
line wrap: on
line diff
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/src/sysdll.c Mon Aug 13 11:28:15 2007 +0200 @@ -0,0 +1,293 @@ +/* sysdll.c --- system dependent support for dynamic linked libraries + Copyright (C) 1998 Free Software Foundation, Inc. + Author: William Perry <wmperry@aventail.com> + +This file is part of XEmacs. + +XEmacs is free software; you can redistribute it and/or modify it +under the terms of the GNU General Public License as published by the +Free Software Foundation; either version 2, or (at your option) any +later version. + +XEmacs 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 General Public License +for more details. + +You should have received a copy of the GNU General Public License +along with XEmacs; see the file COPYING. If not, write to the Free +Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA +02111-1307, USA. */ + +#ifdef HAVE_CONFIG_H +#include <config.h> +#endif + +#include "sysdll.h" + +/* This whole file is conditional upon HAVE_SHLIB */ +#ifdef HAVE_SHLIB + +/* Thankfully, most systems follow the ELFish dlopen() method. +** HAVE__DLOPEN is lame, but SCO has their dl* functions as _dl*, and +** unless you include dlfcn.h you don't get the macros to mask them, and +** autoconf fails to find them. No longer true as of 5.0.5. +** +** Anybody who wants to use this on SCO needs to have their configure.in +** look for _dlopen() as well as dlopen() +*/ +#if defined(HAVE_DLOPEN) || defined(HAVE__DLOPEN) || defined(HAVE_DLFCN_H) +#include <dlfcn.h> + +#ifndef RTLD_LAZY +# define RTLD_LAZY 1 +#endif /* RTLD_LAZY isn't defined under FreeBSD - ick */ + +#ifndef RTLD_GLOBAL +# define RTLD_GLOBAL 0 +#endif + +int +dll_init (CONST char *arg) +{ + return 0; +} + +dll_handle +dll_open (CONST char *fname) +{ + return (dll_handle)dlopen (fname, RTLD_LAZY | RTLD_GLOBAL); +} + +int +dll_close (dll_handle h) +{ + return dlclose((void *)h); +} + +dll_func +dll_function (dll_handle h, CONST char *n) +{ +#ifdef DLSYM_NEEDS_UNDERSCORE + char *buf = alloca_array (char, strlen (n) + 2); + *buf = '_'; + (void)strcpy(buf + 1, n); + n = buf; +#endif + return (dll_func)dlsym ((void *)h, n); +} + +dll_var +dll_variable (dll_handle h, CONST char *n) +{ +#ifdef DLSYM_NEEDS_UNDERSCORE + char *buf = alloca_array (char, strlen (n) + 2); + *buf = '_'; + (void)strcpy(buf + 1, n); + n = buf; +#endif + return (dll_var)dlsym ((void *)h, n); +} + +CONST char * +dll_error (dll_handle h) +{ +#if defined(HAVE_DLERROR) || defined(dlerror) + return (CONST char *)dlerror (); +#elif defined(HAVE__DLERROR) + return (const char *)_dlerror(); +#else + return "Shared library error"; +#endif +} + +#elif defined(HAVE_SHL_LOAD) +/* This is the HP/UX version */ +#include <dl.h> +int +dll_init (CONST char *arg) +{ + return 0; +} + +dll_handle +dll_open (CONST char *fname) +{ + shl_t h = shl_load (fname, BIND_DEFERRED,0L); + shl_t *hp = NULL; + + if (h) + { + hp = (shl_t *)malloc (sizeof (shl_t)); + if (!hp) + shl_unload(h); + else + *hp = h; + } + return (dll_handle)hp; +} + +int +dll_close (dll_handle h) +{ + shl_t hp = *((shl_t *)h); + free (hp); + return shl_unload(h); +} + +dll_func +dll_function (dll_handle h, CONST char *n) +{ + long handle = 0L; + + if (shl_findsym ((shl_t *)h, n, TYPE_PROCEDURE, &handle)) + return NULL; + + return (dll_func)handle; +} + +dll_var +dll_variable (dll_handle h, CONST char *n) +{ + long handle = 0L; + + if (shl_findsym ((shl_t *)h, n, TYPE_DATA, &handle)) + return NULL; + + return (dll_var)handle; +} + +CONST char * +dll_error (dll_handle h) +{ + /* #### WTF?! Shouldn't this at least attempt to get strerror or + something? --hniksic */ + return "Generic shared library error"; +} + +#elif defined(HAVE_INIT_DLD) +#include <dld.h> +int +dll_init (CONST char *arg) +{ + char *real_exe = dld_find_executable (arg); + int rc; + + rc = dld_init (real_exe); + if (rc) + { + dld_perror (exe); + return -1; + } + return 0; +} + +dll_handle +dll_open (CONST char *fname) +{ + rc = dld_link (fname); + if (rc) + return NULL; + + return (dll_handle)1; +} + +int +dll_close (dll_handle h) +{ + /* *sigh* DLD is pretty lame and doesn't return a handle that you can use + ** later on to free the file - you have to remember the filename and + ** use that as the unlinker. We should eventually keep a linked list + ** of loaded modules and then use the node pointer as the unique id + ** for the shared library. Wheeee. But not now. + */ + return 1; +} + +DLL_FUNC +dll_function (dll_handle h, CONST char *n) +{ + return dld_get_func(n); +} + +DLL_FUNC +dll_variable (dll_handle h, CONST char *n) +{ + return dld_get_symbol(n); +} +#elif defined(_WINDOWS) || defined(WIN32) +int +dll_init (CONST char *arg) +{ + return 0; +} + +dll_handle +dll_open (CONST char *fname) +{ + return (dll_handle)LoadLibrary (fname); +} + +int +dll_close (dll_handle h) +{ + return FreeLibrary (h); +} + +dll_func +dll_function (dll_handle h, CONST char *n) +{ + return (dll_func)GetProcAddress (h,n); +} + +dll_func +dll_variable (dll_handle h, CONST char *n) +{ + return (dll_func)GetProcAddress (h,n); +} + +CONST char * +dll_error (dll_handle h) +{ + return "Windows DLL Error"; +} +#else +/* Catchall if we don't know about this systems method of dynamic loading */ +int +dll_init (CONST char *arg) +{ + return -1; +} + +dll_handle +dll_open (CONST char *fname) +{ + return NULL; +} + +int +dll_close (dll_handle h) +{ + return 0; +} + +dll_func +dll_function (dll_handle h, CONST char *n) +{ + return NULL; +} + +dll_func +dll_variable (dll_handle h, CONST char *n) +{ + return NULL; +} + +CONST char * +dll_error (dll_handle h) +{ + return "Shared libraries not implemented on this system"; +} +#endif /* System conditionals */ + +#endif /* HAVE_SHLIB */