This source file includes following definitions.
- load_all_modules
- gf_module_load_static
- gf_modules_new
- gf_modules_del
- gf_module_is_loaded
- gf_modules_get_count
- gf_modules_get_module_directories
- gf_modules_load_interface
- gf_modules_load_interface_by_name
- gf_modules_close_interface
- gf_modules_get_option
- gf_modules_set_option
- gf_modules_get_config
- gf_modules_get_file_name
- gf_module_get_file_name
#include "module_wrap.h"
#include <gpac/config_file.h>
#include <gpac/tools.h>
#include <gpac/network.h>
#ifndef GPAC_MODULE_CUSTOM_LOAD
static void load_all_modules(GF_ModuleManager *mgr)
{
#define LOAD_PLUGIN( __name ) { \
GF_InterfaceRegister *gf_register_module_##__name(); \
pr = gf_register_module_##__name();\
if (!pr) {\
GF_LOG(GF_LOG_ERROR, GF_LOG_CORE, ("Failed to statically load module ##__name\n"));\
} else {\
gf_list_add(mgr->plugin_registry, pr); \
} \
}
#ifdef GPAC_STATIC_MODULES
GF_InterfaceRegister *pr;
#ifdef GPAC_HAS_FAAD
LOAD_PLUGIN(aac_in);
#endif
#ifdef GPAC_HAS_AC3
LOAD_PLUGIN(ac3);
#endif
LOAD_PLUGIN(amr_in);
#ifdef GPAC_HAS_ALSA
LOAD_PLUGIN(alsa);
#endif
LOAD_PLUGIN(audio_filter);
LOAD_PLUGIN(bifs);
#ifndef GPAC_DISABLE_SMGR
LOAD_PLUGIN(ctx_load);
#endif
#ifdef GPAC_HAS_DIRECTFB
LOAD_PLUGIN(directfb_out);
#endif
LOAD_PLUGIN(dummy_in);
#ifdef GPAC_HAS_DIRECTX
LOAD_PLUGIN(dx_out);
#endif
#ifdef GPAC_HAS_FFMPEG
LOAD_PLUGIN(ffmpeg);
#endif
#ifdef GPAC_HAS_FREENECT
LOAD_PLUGIN(freenect);
#endif
#ifdef GPAC_HAS_FREETYPE
LOAD_PLUGIN(ftfont);
#endif
#ifdef GPAC_HAS_SPIDERMONKEY
LOAD_PLUGIN(gpac_js);
#endif
LOAD_PLUGIN(img_in);
LOAD_PLUGIN(isma_ea);
LOAD_PLUGIN(isom);
#ifdef GPAC_HAS_JACK
LOAD_PLUGIN(jack);
#endif
#ifndef GPAC_DISABLE_SVG
LOAD_PLUGIN(laser);
#endif
#ifdef GPAC_HAS_MAD
LOAD_PLUGIN(mp3_in);
#endif
LOAD_PLUGIN(mpd_in);
#ifndef GPAC_DISABLE_MEDIA_IMPORT
LOAD_PLUGIN(mpegts_in);
#endif
#ifdef GPAC_HAS_SPIDERMONKEY
LOAD_PLUGIN(mse_in);
#endif
LOAD_PLUGIN(odf_dec);
#ifdef GPAC_HAS_OGG
LOAD_PLUGIN(ogg_in);
#endif
#ifdef GPAC_HAS_OPENHEVC
LOAD_PLUGIN(openhevc);
#endif
#ifdef GPAC_HAS_OPENSVC
LOAD_PLUGIN(opensvc);
#endif
#ifndef GPAC_DISABLE_LOADER_BT
LOAD_PLUGIN(osd);
#endif
#ifdef GPAC_HAS_OSS
LOAD_PLUGIN(oss);
#endif
#ifdef GPAC_HAS_PULSEAUDIO
LOAD_PLUGIN(pulseaudio);
#endif
LOAD_PLUGIN(raw_out);
#ifdef GPAC_HAS_FFMPEG
#endif
LOAD_PLUGIN(rtp_in);
LOAD_PLUGIN(saf_in);
#ifdef GPAC_HAS_SDL
LOAD_PLUGIN(sdl_out);
#endif
LOAD_PLUGIN(soft_raster);
#if !defined(GPAC_DISABLE_SMGR) && !defined(GPAC_DISABLE_SVG)
LOAD_PLUGIN(svg_in);
#endif
LOAD_PLUGIN(timedtext);
LOAD_PLUGIN(validator);
#ifdef GPAC_HAS_WAVEOUT
LOAD_PLUGIN(wave_out);
#endif
#ifndef GPAC_DISABLE_VTT
LOAD_PLUGIN(vtt_dec);
#endif
#ifndef GPAC_DISABLE_SVG
LOAD_PLUGIN(widgetman);
#endif
#ifdef GPAC_HAS_X11
LOAD_PLUGIN(x11_out);
#endif
#ifdef GPAC_HAS_XVID
LOAD_PLUGIN(xvid);
#endif
#if defined(GPAC_IPHONE) || defined(__DARWIN__) || defined(__APPLE__)
LOAD_PLUGIN(vtb);
#endif
#ifdef GPAC_IPHONE
#endif
#endif
#undef LOAD_PLUGIN
}
#endif
GF_EXPORT
GF_Err gf_module_load_static(GF_ModuleManager *pm, GF_InterfaceRegister *(*register_module)())
{
GF_InterfaceRegister *pr = register_module();
GF_Err rc;
if (!pr) {
GF_LOG(GF_LOG_ERROR, GF_LOG_CORE, ("Failed to statically loaded module\n"));
return GF_NOT_SUPPORTED;
}
rc = gf_list_add(pm->plugin_registry, pr);
if (rc != GF_OK) {
GF_LOG(GF_LOG_ERROR, GF_LOG_CORE, ("Failed to statically loaded module\n"));
return rc;
}
return GF_OK;
}
GF_EXPORT
GF_ModuleManager *gf_modules_new(const char *directory, GF_Config *config)
{
GF_ModuleManager *tmp;
u32 loadedModules;
const char *opt;
u32 num_dirs = 0;
if (!config) return NULL;
GF_SAFEALLOC(tmp, GF_ModuleManager);
if (!tmp) return NULL;
tmp->cfg = config;
tmp->mutex = gf_mx_new("Module Manager");
gf_modules_get_module_directories(tmp, &num_dirs);
tmp->plug_list = gf_list_new();
if (!tmp->plug_list) {
GF_LOG(GF_LOG_ERROR, GF_LOG_CORE, ("OUT OF MEMORY, cannot create list of modules !!!\n"));
gf_free(tmp);
return NULL;
}
tmp->plugin_registry = gf_list_new();
if (!tmp->plugin_registry) {
GF_LOG(GF_LOG_ERROR, GF_LOG_CORE, ("OUT OF MEMORY, cannot create list of static module registers !!!\n"));
gf_list_del(tmp->plug_list);
gf_free(tmp);
return NULL;
}
opt = gf_cfg_get_key(config, "Systems", "ModuleUnload");
if (opt && !strcmp(opt, "no")) {
tmp->no_unload = GF_TRUE;
}
#ifndef GPAC_MODULE_CUSTOM_LOAD
load_all_modules(tmp);
#endif
loadedModules = gf_modules_refresh(tmp);
GF_LOG(GF_LOG_INFO, GF_LOG_CORE, ("Loaded %d modules from directory %s.\n", loadedModules, directory));
return tmp;
}
GF_EXPORT
void gf_modules_del(GF_ModuleManager *pm)
{
u32 i;
ModuleInstance *inst;
if (!pm) return;
while (gf_list_count(pm->plug_list)) {
inst = (ModuleInstance *) gf_list_get(pm->plug_list, 0);
gf_modules_free_module(inst);
gf_list_rem(pm->plug_list, 0);
}
gf_list_del(pm->plug_list);
for (i = 0; i < pm->num_dirs; i++) {
gf_free((void*)pm->dirs[i]);
}
while (gf_list_count(pm->plugin_registry)) {
GF_InterfaceRegister *reg = (GF_InterfaceRegister *) gf_list_get(pm->plugin_registry, 0);
gf_free(reg);
gf_list_rem(pm->plugin_registry, 0);
}
if (pm->plugin_registry) gf_list_del(pm->plugin_registry);
gf_mx_del(pm->mutex);
gf_free(pm);
}
Bool gf_module_is_loaded(GF_ModuleManager *pm, char *filename)
{
u32 i = 0;
ModuleInstance *inst;
while ( (inst = (ModuleInstance *) gf_list_enum(pm->plug_list, &i) ) ) {
if (!strcmp(inst->name, filename)) return GF_TRUE;
}
return GF_FALSE;
}
GF_EXPORT
u32 gf_modules_get_count(GF_ModuleManager *pm)
{
if (!pm) return 0;
return gf_list_count(pm->plug_list);
}
GF_EXPORT
const char **gf_modules_get_module_directories(GF_ModuleManager *pm, u32* num_dirs)
{
char* directories;
char* tmp_dirs;
char * pch;
if (!pm) return NULL;
if (pm->num_dirs > 0 ) {
*num_dirs = pm->num_dirs;
return pm->dirs;
}
if (!pm->cfg) return NULL;
directories = (char*)gf_cfg_get_key(pm->cfg, "General", "ModulesDirectory");
if (! directories) {
#ifndef GPAC_IPHONE
GF_LOG(GF_LOG_ERROR, GF_LOG_CORE, ("Module directory not found - check the configuration file exit and the \"ModulesDirectory\" key is set\n"));
#endif
return NULL;
}
tmp_dirs = directories;
pch = strtok (tmp_dirs,";");
while (pch != NULL)
{
if (pm->num_dirs == MAX_MODULE_DIRS) {
GF_LOG(GF_LOG_WARNING, GF_LOG_CORE, ("Reach maximum number of module directories - check the configuration file and the \"ModulesDirectory\" key.\n"));
break;
}
pm->dirs[pm->num_dirs] = gf_strdup(pch);
pm->num_dirs++;
pch = strtok (NULL, ";");
}
*num_dirs = pm->num_dirs;
return pm->dirs;
}
GF_EXPORT
GF_BaseInterface *gf_modules_load_interface(GF_ModuleManager *pm, u32 whichplug, u32 InterfaceFamily)
{
const char *opt;
char szKey[32];
ModuleInstance *inst;
GF_BaseInterface *ifce;
if (!pm) {
GF_LOG(GF_LOG_ERROR, GF_LOG_CORE, ("[Core] gf_modules_load_interface() : No Module Manager set\n"));
return NULL;
}
gf_mx_p(pm->mutex);
inst = (ModuleInstance *) gf_list_get(pm->plug_list, whichplug);
if (!inst) {
GF_LOG(GF_LOG_ERROR, GF_LOG_CORE, ("[Core] gf_modules_load_interface() : no module %d exist.\n", whichplug));
gf_mx_v(pm->mutex);
return NULL;
}
GF_LOG(GF_LOG_DEBUG, GF_LOG_CORE, ("[Core] Load interface...%s\n", inst->name));
if (!pm->cfg) {
GF_LOG(GF_LOG_ERROR, GF_LOG_CORE, ("[Core] No pm->cfg has been set !!!\n"));
gf_mx_v(pm->mutex);
return NULL;
}
opt = gf_cfg_get_key(pm->cfg, "PluginsCache", inst->name);
if (opt) {
const char * ifce_str = gf_4cc_to_str(InterfaceFamily);
snprintf(szKey, 32, "%s:yes", ifce_str ? ifce_str : "(null)");
if (!strstr(opt, szKey)) {
gf_mx_v(pm->mutex);
return NULL;
}
}
if (!gf_modules_load_library(inst)) {
GF_LOG(GF_LOG_ERROR, GF_LOG_CORE, ("[Core] Cannot load library %s\n", inst->name));
gf_cfg_set_key(pm->cfg, "PluginsCache", inst->name, "Invalid Plugin");
gf_mx_v(pm->mutex);
return NULL;
}
if (!inst->query_func) {
GF_LOG(GF_LOG_ERROR, GF_LOG_CORE, ("[Core] Library %s missing GPAC export symbols\n", inst->name));
gf_cfg_set_key(pm->cfg, "PluginsCache", inst->name, "Invalid Plugin");
goto err_exit;
}
if (!opt) {
u32 i;
Bool found = GF_FALSE;
char *key;
const u32 *si = inst->query_func();
if (!si) {
GF_LOG(GF_LOG_WARNING, GF_LOG_CORE, ("[Core] GPAC module %s has no supported interfaces - disabling\n", inst->name));
gf_cfg_set_key(pm->cfg, "PluginsCache", inst->name, "Invalid Plugin");
goto err_exit;
}
i=0;
while (si[i]) i++;
key = (char*)gf_malloc(sizeof(char) * 10 * i);
key[0] = 0;
i=0;
while (si[i]) {
snprintf(szKey, 32, "%s:yes ", gf_4cc_to_str(si[i]));
strcat(key, szKey);
if (InterfaceFamily==si[i]) found = GF_TRUE;
i++;
}
gf_cfg_set_key(pm->cfg, "PluginsCache", inst->name, key);
gf_free(key);
if (!found) goto err_exit;
}
if (!inst->query_func || !inst->query_func(InterfaceFamily) ) goto err_exit;
ifce = (GF_BaseInterface *) inst->load_func(InterfaceFamily);
if (!ifce) goto err_exit;
if (!ifce->module_name || (ifce->InterfaceType != InterfaceFamily)) {
inst->destroy_func(ifce);
goto err_exit;
}
gf_list_add(inst->interfaces, ifce);
ifce->HPLUG = inst;
GF_LOG(GF_LOG_DEBUG, GF_LOG_CORE, ("[Core] Load interface %s DONE.\n", inst->name));
gf_mx_v(pm->mutex);
return ifce;
err_exit:
GF_LOG(GF_LOG_DEBUG, GF_LOG_CORE, ("[Core] Load interface %s exit label, freing library...\n", inst->name));
gf_modules_unload_library(inst);
GF_LOG(GF_LOG_DEBUG, GF_LOG_CORE, ("[Core] Load interface %s EXIT.\n", inst->name));
gf_mx_v(pm->mutex);
return NULL;
}
GF_EXPORT
GF_BaseInterface *gf_modules_load_interface_by_name(GF_ModuleManager *pm, const char *plug_name, u32 InterfaceFamily)
{
const char *file_name;
u32 i, count;
GF_BaseInterface *ifce;
if (!pm || !plug_name || !pm->plug_list || !pm->cfg) {
GF_LOG(GF_LOG_ERROR, GF_LOG_CORE, ("[Core] gf_modules_load_interface_by_name has bad parameters pm=%p, plug_name=%s.\n", pm, plug_name));
return NULL;
}
count = gf_list_count(pm->plug_list);
file_name = gf_cfg_get_key(pm->cfg, "PluginsCache", plug_name);
if (file_name) {
for (i=0; i<count; i++) {
ModuleInstance *inst = (ModuleInstance *) gf_list_get(pm->plug_list, i);
if (!strcmp(inst->name, file_name)) {
ifce = gf_modules_load_interface(pm, i, InterfaceFamily);
if (ifce) return ifce;
}
}
}
GF_LOG(GF_LOG_INFO, GF_LOG_CORE, ("[Core] Plugin %s of type %d not found in cache, searching for it...\n", plug_name, InterfaceFamily));
for (i=0; i<count; i++) {
const char *mod_filename;
ifce = gf_modules_load_interface(pm, i, InterfaceFamily);
if (!ifce) continue;
if (ifce->module_name && !strnicmp(ifce->module_name, plug_name, MIN(strlen(ifce->module_name), strlen(plug_name)) )) {
gf_cfg_set_key(pm->cfg, "PluginsCache", plug_name, ((ModuleInstance*)ifce->HPLUG)->name);
GF_LOG(GF_LOG_DEBUG, GF_LOG_CORE, ("[Core] Added plugin cache %s for %s\n", plug_name, ((ModuleInstance*)ifce->HPLUG)->name));
return ifce;
}
mod_filename = gf_module_get_file_name(ifce);
if (mod_filename && strstr(mod_filename, plug_name)) {
return ifce;
}
gf_modules_close_interface(ifce);
}
GF_LOG(GF_LOG_WARNING, GF_LOG_CORE, ("[Core] Plugin %s not found in %d modules.\n", plug_name, count));
return NULL;
}
GF_EXPORT
GF_Err gf_modules_close_interface(GF_BaseInterface *ifce)
{
ModuleInstance *par;
s32 i;
if (!ifce) return GF_BAD_PARAM;
par = (ModuleInstance *) ifce->HPLUG;
if (!par || !ifce->InterfaceType) return GF_BAD_PARAM;
i = gf_list_find(par->plugman->plug_list, par);
if (i<0) return GF_BAD_PARAM;
i = gf_list_find(par->interfaces, ifce);
if (i<0) return GF_BAD_PARAM;
gf_list_rem(par->interfaces, (u32) i);
par->destroy_func(ifce);
gf_modules_unload_library(par);
return GF_OK;
}
GF_EXPORT
const char *gf_modules_get_option(GF_BaseInterface *ifce, const char *secName, const char *keyName)
{
GF_Config *cfg;
if (!ifce || !ifce->HPLUG) return NULL;
cfg = ((ModuleInstance *)ifce->HPLUG)->plugman->cfg;
if (!cfg) return NULL;
return gf_cfg_get_ikey(cfg, secName, keyName);
}
GF_EXPORT
GF_Err gf_modules_set_option(GF_BaseInterface *ifce, const char *secName, const char *keyName, const char *keyValue)
{
GF_Config *cfg;
if (!ifce || !ifce->HPLUG) return GF_BAD_PARAM;
cfg = ((ModuleInstance *)ifce->HPLUG)->plugman->cfg;
if (!cfg) return GF_NOT_SUPPORTED;
return gf_cfg_set_key(cfg, secName, keyName, keyValue);
}
GF_EXPORT
GF_Config *gf_modules_get_config(GF_BaseInterface *ifce)
{
if (!ifce || !ifce->HPLUG) return NULL;
return ((ModuleInstance *)ifce->HPLUG)->plugman->cfg;
}
GF_EXPORT
const char *gf_modules_get_file_name(GF_ModuleManager *pm, u32 i)
{
ModuleInstance *inst = (ModuleInstance *) gf_list_get(pm->plug_list, i);
if (!inst) return NULL;
return inst->name;
}
GF_EXPORT
const char *gf_module_get_file_name(GF_BaseInterface *ifce)
{
ModuleInstance *inst = (ModuleInstance *) ifce->HPLUG;
if (!inst) return NULL;
return inst->name;
}