Support dl-object cache

Added the support of dl-object caches for PCM plugins.
This commit is contained in:
Takashi Iwai 2005-02-14 15:09:09 +00:00
parent 08fc630594
commit d8f7de1b16
5 changed files with 93 additions and 3 deletions

View file

@ -232,4 +232,9 @@ static inline int snd_open_device(const char *filename, int fmode)
#define snd_open_device(filename, fmode) open(filename, fmode); #define snd_open_device(filename, fmode) open(filename, fmode);
#endif #endif
/* dlobj cache */
void *snd_dlobj_cache_lookup(const char *name);
int snd_dlobj_cache_add(const char *name, void *dlobj, void *open_func);
void snd_dlobj_cache_cleanup(void);
#endif #endif

View file

@ -37,10 +37,13 @@ enum {
typedef struct snd_pcm_ioplug snd_pcm_ioplug_t; typedef struct snd_pcm_ioplug snd_pcm_ioplug_t;
typedef struct snd_pcm_ioplug_callback snd_pcm_ioplug_callback_t; typedef struct snd_pcm_ioplug_callback snd_pcm_ioplug_callback_t;
#define SND_PCM_IOPLUG_FLAG_LISTED (1<<0)
/* exported pcm data */ /* exported pcm data */
struct snd_pcm_ioplug { struct snd_pcm_ioplug {
/* must be filled before calling snd_pcm_ioplug_create() */ /* must be filled before calling snd_pcm_ioplug_create() */
const char *name; const char *name;
unsigned int flags; /* SND_PCM_IOPLUG_FLAG_XXX */
int poll_fd; int poll_fd;
unsigned int poll_events; unsigned int poll_events;
unsigned int mmap_rw; /* pseudo mmap */ unsigned int mmap_rw; /* pseudo mmap */

View file

@ -3113,6 +3113,10 @@ int snd_config_update_free_global(void)
snd_config_update_free(snd_config_global_update); snd_config_update_free(snd_config_global_update);
snd_config_global_update = NULL; snd_config_global_update = NULL;
pthread_mutex_unlock(&snd_config_update_mutex); pthread_mutex_unlock(&snd_config_update_mutex);
/* FIXME: better to place this in another place... */
snd_dlobj_cache_cleanup();
return 0; return 0;
} }

View file

@ -29,6 +29,7 @@
#define _GNU_SOURCE #define _GNU_SOURCE
#include <dlfcn.h> #include <dlfcn.h>
#include "list.h"
#include "local.h" #include "local.h"
#ifndef DOC_HIDDEN #ifndef DOC_HIDDEN
@ -143,3 +144,70 @@ void *snd_dlsym(void *handle, const char *name, const char *version)
return NULL; return NULL;
return dlsym(handle, name); return dlsym(handle, name);
} }
/*
* dlobj cache
*
* FIXME: add reference counter and proper locking
*/
struct dlobj_cache {
const char *name;
void *obj;
void *func;
struct list_head list;
};
static LIST_HEAD(pcm_dlobj_list);
void *snd_dlobj_cache_lookup(const char *name)
{
struct list_head *p;
struct dlobj_cache *c;
list_for_each(p, &pcm_dlobj_list) {
c = list_entry(p, struct dlobj_cache, list);
if (strcmp(c->name, name) == 0)
return c->func;
}
return NULL;
}
int snd_dlobj_cache_add(const char *name, void *dlobj, void *open_func)
{
struct list_head *p;
struct dlobj_cache *c;
list_for_each(p, &pcm_dlobj_list) {
c = list_entry(p, struct dlobj_cache, list);
if (strcmp(c->name, name) == 0)
return 0; /* already exists */
}
c = malloc(sizeof(*c));
if (! c)
return -ENOMEM;
c->name = strdup(name);
if (! c->name) {
free(c);
return -ENOMEM;
}
c->obj = dlobj;
c->func = open_func;
list_add_tail(&c->list, &pcm_dlobj_list);
return 0;
}
void snd_dlobj_cache_cleanup(void)
{
struct list_head *p;
struct dlobj_cache *c;
while (! list_empty(&pcm_dlobj_list)) {
p = pcm_dlobj_list.next;
c = list_entry(p, struct dlobj_cache, list);
list_del(p);
snd_dlclose(c->obj);
free(c->name);
free(c);
}
}

View file

@ -2019,6 +2019,11 @@ static int snd_pcm_open_conf(snd_pcm_t **pcmp, const char *name,
#ifndef PIC #ifndef PIC
snd_pcm_open_symbols(); /* this call is for static linking only */ snd_pcm_open_symbols(); /* this call is for static linking only */
#endif #endif
open_func = snd_dlobj_cache_lookup(open_name);
if (open_func) {
err = 0;
goto _err;
}
h = snd_dlopen(lib, RTLD_NOW); h = snd_dlopen(lib, RTLD_NOW);
if (h) if (h)
open_func = snd_dlsym(h, open_name, SND_DLSYM_VERSION(SND_PCM_DLSYM_VERSION)); open_func = snd_dlsym(h, open_name, SND_DLSYM_VERSION(SND_PCM_DLSYM_VERSION));
@ -2032,17 +2037,22 @@ static int snd_pcm_open_conf(snd_pcm_t **pcmp, const char *name,
err = -ENXIO; err = -ENXIO;
} }
_err: _err:
if (type_conf)
snd_config_delete(type_conf);
if (err >= 0) { if (err >= 0) {
err = open_func(pcmp, name, pcm_root, pcm_conf, stream, mode); err = open_func(pcmp, name, pcm_root, pcm_conf, stream, mode);
if (err >= 0) { if (err >= 0) {
if (h /*&& (mode & SND_PCM_KEEP_ALIVE)*/) {
snd_dlobj_cache_add(open_name, h, open_func);
h = NULL;
}
(*pcmp)->dl_handle = h; (*pcmp)->dl_handle = h;
err = 0; err = 0;
} else { } else {
snd_dlclose(h); if (h)
snd_dlclose(h);
} }
} }
if (type_conf)
snd_config_delete(type_conf);
if (buf) if (buf)
free(buf); free(buf);
if (buf1) if (buf1)