/** * \file dlmisc.c * \brief dynamic loader helpers * \author Jaroslav Kysela * \date 2001 * * Dynamic loader helpers */ /* * Dynamic loader helpers * Copyright (c) 2000 by Jaroslav Kysela * * * This library is free software; you can redistribute it and/or modify * it under the terms of the GNU Lesser General Public License as * published by the Free Software Foundation; either version 2.1 of * the License, or (at your option) any later version. * * This program 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 Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser 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 "list.h" #include "local.h" #ifdef HAVE_LIBPTHREAD #include #endif #ifndef DOC_HIDDEN #ifndef PIC struct snd_dlsym_link *snd_dlsym_start = NULL; #endif #endif /** * \brief Opens a dynamic library - ALSA wrapper for \c dlopen. * \param name name of the library, similar to \c dlopen. * \param mode mode flags, similar to \c dlopen. * \return Library handle if successful, otherwise \c NULL. * * This function can emulate dynamic linking for the static build of * the alsa-lib library. In that case, \p name is set to \c NULL. */ void *snd_dlopen(const char *name, int mode) { #ifndef PIC if (name == NULL) return &snd_dlsym_start; #else #ifdef HAVE_LIBDL if (name == NULL) { static const char * self = NULL; if (self == NULL) { Dl_info dlinfo; if (dladdr(snd_dlopen, &dlinfo) > 0) self = dlinfo.dli_fname; } name = self; } #endif #endif #ifdef HAVE_LIBDL return dlopen(name, mode); #else return NULL; #endif } /** * \brief Closes a dynamic library - ALSA wrapper for \c dlclose. * \param handle Library handle, similar to \c dlclose. * \return Zero if successful, otherwise an error code. * * This function can emulate dynamic linking for the static build of * the alsa-lib library. */ int snd_dlclose(void *handle) { #ifndef PIC if (handle == &snd_dlsym_start) return 0; #endif #ifdef HAVE_LIBDL return dlclose(handle); #else return 0; #endif } /** * \brief Verifies a dynamically loaded symbol. * \param handle Library handle, similar to \c dlsym. * \param name Symbol name. * \param version Version of the symbol. * \return Zero is successful, otherwise a negative error code. * * This function checks that the symbol with the version appended to its name * does exist in the library. */ static int snd_dlsym_verify(void *handle, const char *name, const char *version) { #ifdef HAVE_LIBDL int res; char *vname; if (handle == NULL) return -EINVAL; vname = alloca(1 + strlen(name) + strlen(version) + 1); if (vname == NULL) return -ENOMEM; vname[0] = '_'; strcpy(vname + 1, name); strcat(vname, version); res = dlsym(handle, vname) == NULL ? -ENOENT : 0; // printf("dlsym verify: %i, vname = '%s'\n", res, vname); if (res < 0) SNDERR("unable to verify version for symbol %s", name); return res; #else return 0; #endif } /** * \brief Resolves a symbol from a dynamic library - ALSA wrapper for \c dlsym. * \param handle Library handle, similar to \c dlsym. * \param name Symbol name. * \param version Version of the symbol. * * This function can emulate dynamic linking for the static build of * the alsa-lib library. * * This special version of the \c dlsym function checks also the version * of the symbol. A versioned symbol should be defined using the * #SND_DLSYM_BUILD_VERSION macro. */ void *snd_dlsym(void *handle, const char *name, const char *version) { int err; #ifndef PIC if (handle == &snd_dlsym_start) { /* it's the funny part: */ /* we are looking for a symbol in a static library */ struct snd_dlsym_link *link = snd_dlsym_start; while (link) { if (!strcmp(name, link->dlsym_name)) return (void *)link->dlsym_ptr; link = link->next; } return NULL; } #endif #ifdef HAVE_LIBDL if (version) { err = snd_dlsym_verify(handle, name, version); if (err < 0) return NULL; } return dlsym(handle, name); #else return NULL; #endif } /* * dlobj cache */ #ifndef DOC_HIDDEN struct dlobj_cache { const char *lib; const char *name; void *dlobj; void *func; unsigned int refcnt; struct list_head list; }; #ifdef HAVE_LIBPTHREAD static pthread_mutex_t snd_dlobj_mutex = PTHREAD_MUTEX_INITIALIZER; static inline void snd_dlobj_lock(void) { pthread_mutex_lock(&snd_dlobj_mutex); } static inline void snd_dlobj_unlock(void) { pthread_mutex_unlock(&snd_dlobj_mutex); } #else static inline void snd_dlobj_lock(void) {} static inline void snd_dlobj_unlock(void) {} #endif static LIST_HEAD(pcm_dlobj_list); void *snd_dlobj_cache_get(const char *lib, const char *name, const char *version, int verbose) { struct list_head *p; struct dlobj_cache *c; void *func, *dlobj = NULL; int dlobj_close = 0; snd_dlobj_lock(); list_for_each(p, &pcm_dlobj_list) { c = list_entry(p, struct dlobj_cache, list); if (c->lib && lib && strcmp(c->lib, lib) != 0) continue; if (!c->lib && lib) continue; if (!lib && c->lib) continue; dlobj = c->dlobj; if (strcmp(c->name, name) == 0) { c->refcnt++; func = c->func; snd_dlobj_unlock(); return func; } } if (dlobj == NULL) { dlobj = snd_dlopen(lib, RTLD_NOW); if (dlobj == NULL) { if (verbose) SNDERR("Cannot open shared library %s", lib ? lib : "[builtin]"); snd_dlobj_unlock(); return NULL; } dlobj_close = 1; } func = snd_dlsym(dlobj, name, version); if (func == NULL) { if (verbose) SNDERR("symbol %s is not defined inside %s", name, lib ? lib : "[builtin]"); goto __err; } c = malloc(sizeof(*c)); if (! c) goto __err; c->refcnt = 1; c->lib = lib ? strdup(lib) : NULL; c->name = strdup(name); if ((lib && ! c->lib) || ! c->name) { free((void *)c->name); free((void *)c->lib); free(c); __err: if (dlobj_close) snd_dlclose(dlobj); snd_dlobj_unlock(); return NULL; } c->dlobj = dlobj; c->func = func; list_add_tail(&c->list, &pcm_dlobj_list); snd_dlobj_unlock(); return func; } int snd_dlobj_cache_put(void *func) { struct list_head *p; struct dlobj_cache *c; unsigned int refcnt; snd_dlobj_lock(); list_for_each(p, &pcm_dlobj_list) { c = list_entry(p, struct dlobj_cache, list); if (c->func == func) { refcnt = c->refcnt; if (c->refcnt > 0) c->refcnt--; snd_dlobj_unlock(); return refcnt == 1 ? 0 : -EINVAL; } } snd_dlobj_unlock(); return -ENOENT; } void snd_dlobj_cache_cleanup(void) { struct list_head *p, *npos; struct dlobj_cache *c; snd_dlobj_lock(); list_for_each_safe(p, npos, &pcm_dlobj_list) { c = list_entry(p, struct dlobj_cache, list); if (c->refcnt == 0) { list_del(p); snd_dlclose(c->dlobj); free((void *)c->name); /* shut up gcc warning */ free((void *)c->lib); /* shut up gcc warning */ free(c); } } snd_dlobj_unlock(); } #endif