2005-11-19 20:07:43 +00:00
|
|
|
/*
|
|
|
|
* FreeSWITCH Modular Media Switching Software Library / Soft-Switch Application
|
2009-02-13 23:37:37 +00:00
|
|
|
* Copyright (C) 2005-2009, Anthony Minessale II <anthm@freeswitch.org>
|
2005-11-19 20:07:43 +00:00
|
|
|
*
|
|
|
|
* Version: MPL 1.1
|
|
|
|
*
|
|
|
|
* The contents of this file are subject to the Mozilla Public License Version
|
|
|
|
* 1.1 (the "License"); you may not use this file except in compliance with
|
|
|
|
* the License. You may obtain a copy of the License at
|
|
|
|
* http://www.mozilla.org/MPL/
|
|
|
|
*
|
|
|
|
* Software distributed under the License is distributed on an "AS IS" basis,
|
|
|
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
|
|
|
* for the specific language governing rights and limitations under the
|
|
|
|
* License.
|
|
|
|
*
|
|
|
|
* The Original Code is FreeSWITCH Modular Media Switching Software Library / Soft-Switch Application
|
|
|
|
*
|
|
|
|
* The Initial Developer of the Original Code is
|
2009-02-04 21:20:54 +00:00
|
|
|
* Anthony Minessale II <anthm@freeswitch.org>
|
2005-11-19 20:07:43 +00:00
|
|
|
* Portions created by the Initial Developer are Copyright (C)
|
|
|
|
* the Initial Developer. All Rights Reserved.
|
|
|
|
*
|
|
|
|
* Contributor(s):
|
|
|
|
*
|
2009-02-04 21:20:54 +00:00
|
|
|
* Anthony Minessale II <anthm@freeswitch.org>
|
2005-11-19 20:07:43 +00:00
|
|
|
*
|
|
|
|
*
|
|
|
|
* switch_loadable_module.c -- Loadable Modules
|
|
|
|
*
|
|
|
|
*/
|
2008-01-27 17:42:51 +00:00
|
|
|
|
2006-03-06 03:23:34 +00:00
|
|
|
#include <switch.h>
|
2007-03-09 20:44:13 +00:00
|
|
|
|
|
|
|
/* for apr_pstrcat */
|
|
|
|
#include <apr_strings.h>
|
|
|
|
|
|
|
|
/* for apr_env_get and apr_env_set */
|
|
|
|
#include <apr_env.h>
|
|
|
|
|
|
|
|
/* for apr file and directory handling */
|
|
|
|
#include <apr_file_io.h>
|
2005-11-19 20:07:43 +00:00
|
|
|
|
|
|
|
struct switch_loadable_module {
|
2007-05-03 16:28:23 +00:00
|
|
|
char *key;
|
2005-11-19 20:07:43 +00:00
|
|
|
char *filename;
|
2007-05-03 16:28:23 +00:00
|
|
|
int perm;
|
2007-06-13 20:40:06 +00:00
|
|
|
switch_loadable_module_interface_t *module_interface;
|
2008-11-06 17:29:50 +00:00
|
|
|
switch_dso_lib_t lib;
|
2006-03-30 23:02:50 +00:00
|
|
|
switch_module_load_t switch_module_load;
|
|
|
|
switch_module_runtime_t switch_module_runtime;
|
|
|
|
switch_module_shutdown_t switch_module_shutdown;
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_memory_pool_t *pool;
|
2008-08-22 17:57:15 +00:00
|
|
|
switch_status_t status;
|
2009-02-20 01:10:59 +00:00
|
|
|
switch_thread_t *thread;
|
2005-11-19 20:07:43 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
struct switch_loadable_module_container {
|
2006-04-29 01:00:52 +00:00
|
|
|
switch_hash_t *module_hash;
|
|
|
|
switch_hash_t *endpoint_hash;
|
|
|
|
switch_hash_t *codec_hash;
|
|
|
|
switch_hash_t *dialplan_hash;
|
|
|
|
switch_hash_t *timer_hash;
|
|
|
|
switch_hash_t *application_hash;
|
|
|
|
switch_hash_t *api_hash;
|
|
|
|
switch_hash_t *file_hash;
|
|
|
|
switch_hash_t *speech_hash;
|
2006-11-09 05:39:04 +00:00
|
|
|
switch_hash_t *asr_hash;
|
2006-04-29 01:00:52 +00:00
|
|
|
switch_hash_t *directory_hash;
|
2006-10-19 07:13:34 +00:00
|
|
|
switch_hash_t *chat_hash;
|
2006-12-20 21:25:14 +00:00
|
|
|
switch_hash_t *say_hash;
|
2007-03-07 18:34:22 +00:00
|
|
|
switch_hash_t *management_hash;
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_mutex_t *mutex;
|
2006-04-29 01:00:52 +00:00
|
|
|
switch_memory_pool_t *pool;
|
2005-11-19 20:07:43 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
static struct switch_loadable_module_container loadable_modules;
|
2008-11-12 17:10:20 +00:00
|
|
|
static switch_status_t do_shutdown(switch_loadable_module_t *module, switch_bool_t shutdown, switch_bool_t unload, switch_bool_t fail_if_busy, const char **err);
|
2008-11-06 17:29:50 +00:00
|
|
|
static switch_status_t switch_loadable_module_load_module_ex(char *dir, char *fname, switch_bool_t runtime, switch_bool_t global, const char **err);
|
2005-11-19 20:07:43 +00:00
|
|
|
|
2009-02-16 13:59:13 +00:00
|
|
|
static void *SWITCH_THREAD_FUNC switch_loadable_module_exec(switch_thread_t *thread, void *obj)
|
2005-11-19 20:07:43 +00:00
|
|
|
{
|
2006-03-30 23:02:50 +00:00
|
|
|
|
|
|
|
|
2006-04-29 23:43:28 +00:00
|
|
|
switch_status_t status = SWITCH_STATUS_SUCCESS;
|
2006-04-29 06:05:03 +00:00
|
|
|
switch_core_thread_session_t *ts = obj;
|
|
|
|
switch_loadable_module_t *module = ts->objs[0];
|
2005-11-19 20:07:43 +00:00
|
|
|
int restarts;
|
|
|
|
|
2007-12-11 19:23:57 +00:00
|
|
|
switch_assert(thread != NULL);
|
|
|
|
switch_assert(module != NULL);
|
2005-11-19 20:07:43 +00:00
|
|
|
|
|
|
|
for (restarts = 0; status != SWITCH_STATUS_TERM; restarts++) {
|
|
|
|
status = module->switch_module_runtime();
|
|
|
|
}
|
2007-03-30 00:13:31 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE, "Thread ended for %s\n", module->module_interface->module_name);
|
2005-12-22 18:53:33 +00:00
|
|
|
|
|
|
|
if (ts->pool) {
|
2006-04-29 01:00:52 +00:00
|
|
|
switch_memory_pool_t *pool = ts->pool;
|
2007-03-30 00:13:31 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "Destroying Pool for %s\n", module->module_interface->module_name);
|
2005-12-22 18:53:33 +00:00
|
|
|
switch_core_destroy_memory_pool(&pool);
|
|
|
|
}
|
2009-02-16 13:59:13 +00:00
|
|
|
switch_thread_exit(thread, 0);
|
2005-11-19 20:07:43 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
2006-02-22 22:36:52 +00:00
|
|
|
|
2006-03-30 23:02:50 +00:00
|
|
|
|
2007-03-17 19:51:08 +00:00
|
|
|
static void switch_loadable_module_runtime(void)
|
|
|
|
{
|
|
|
|
switch_hash_index_t *hi;
|
|
|
|
void *val;
|
|
|
|
switch_loadable_module_t *module;
|
2006-04-25 00:33:00 +00:00
|
|
|
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_mutex_lock(loadable_modules.mutex);
|
2007-09-24 19:34:25 +00:00
|
|
|
for (hi = switch_hash_first(NULL, loadable_modules.module_hash); hi; hi = switch_hash_next(hi)) {
|
2007-03-17 19:51:08 +00:00
|
|
|
switch_hash_this(hi, NULL, NULL, &val);
|
|
|
|
module = (switch_loadable_module_t *) val;
|
2007-03-29 22:31:56 +00:00
|
|
|
|
2007-03-17 19:51:08 +00:00
|
|
|
if (module->switch_module_runtime) {
|
2007-03-30 00:13:31 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CONSOLE, "Starting runtime thread for %s\n", module->module_interface->module_name);
|
2009-02-20 01:10:59 +00:00
|
|
|
module->thread = switch_core_launch_thread(switch_loadable_module_exec, module, loadable_modules.pool);
|
2007-03-17 19:51:08 +00:00
|
|
|
}
|
|
|
|
}
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_mutex_unlock(loadable_modules.mutex);
|
2007-03-17 19:51:08 +00:00
|
|
|
}
|
2006-04-25 00:33:00 +00:00
|
|
|
|
2006-04-29 23:43:28 +00:00
|
|
|
static switch_status_t switch_loadable_module_process(char *key, switch_loadable_module_t *new_module)
|
2006-04-25 00:33:00 +00:00
|
|
|
{
|
2006-05-01 19:44:21 +00:00
|
|
|
switch_event_t *event;
|
2006-04-25 00:33:00 +00:00
|
|
|
|
2007-05-03 16:28:23 +00:00
|
|
|
new_module->key = switch_core_strdup(new_module->pool, key);
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_mutex_lock(loadable_modules.mutex);
|
2006-04-25 00:33:00 +00:00
|
|
|
switch_core_hash_insert(loadable_modules.module_hash, key, new_module);
|
|
|
|
|
2006-04-30 18:24:24 +00:00
|
|
|
if (new_module->module_interface->endpoint_interface) {
|
2006-04-29 23:43:28 +00:00
|
|
|
const switch_endpoint_interface_t *ptr;
|
2006-04-30 18:24:24 +00:00
|
|
|
for (ptr = new_module->module_interface->endpoint_interface; ptr; ptr = ptr->next) {
|
2007-02-23 18:24:31 +00:00
|
|
|
if (!ptr->interface_name) {
|
2007-09-29 01:06:08 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CRIT, "Failed to load endpoint interface from %s due to no interface name.\n", key);
|
2007-02-23 18:24:31 +00:00
|
|
|
} else {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE, "Adding Endpoint '%s'\n", ptr->interface_name);
|
2007-02-23 19:15:43 +00:00
|
|
|
switch_core_hash_insert(loadable_modules.endpoint_hash, ptr->interface_name, (const void *) ptr);
|
2008-07-22 17:19:26 +00:00
|
|
|
if (switch_event_create(&event, SWITCH_EVENT_MODULE_LOAD) == SWITCH_STATUS_SUCCESS) {
|
2008-08-16 02:17:09 +00:00
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "type", "endpoint");
|
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "name", ptr->interface_name);
|
2008-07-22 17:19:26 +00:00
|
|
|
switch_event_fire(&event);
|
|
|
|
}
|
2007-02-23 18:24:31 +00:00
|
|
|
}
|
2006-04-25 00:33:00 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2006-04-30 18:24:24 +00:00
|
|
|
if (new_module->module_interface->codec_interface) {
|
2006-04-29 06:05:03 +00:00
|
|
|
const switch_codec_implementation_t *impl;
|
2006-04-29 23:43:28 +00:00
|
|
|
const switch_codec_interface_t *ptr;
|
2006-04-25 00:33:00 +00:00
|
|
|
|
2006-04-30 18:24:24 +00:00
|
|
|
for (ptr = new_module->module_interface->codec_interface; ptr; ptr = ptr->next) {
|
2007-03-29 22:31:56 +00:00
|
|
|
if (!ptr->interface_name) {
|
2007-09-29 01:06:08 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CRIT, "Failed to load codec interface from %s due to no interface name.\n", key);
|
2007-02-23 18:24:31 +00:00
|
|
|
} else {
|
2007-02-23 19:15:43 +00:00
|
|
|
unsigned load_interface = 1;
|
2007-02-23 18:24:31 +00:00
|
|
|
for (impl = ptr->implementations; impl; impl = impl->next) {
|
2007-02-23 19:15:43 +00:00
|
|
|
if (!impl->iananame) {
|
|
|
|
load_interface = 0;
|
2008-05-21 22:26:30 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CRIT,
|
2008-05-27 04:30:03 +00:00
|
|
|
"Failed to load codec interface %s from %s due to no iana name in an implementation.\n", ptr->interface_name,
|
|
|
|
key);
|
2008-05-21 22:26:30 +00:00
|
|
|
break;
|
|
|
|
}
|
2008-10-20 17:48:42 +00:00
|
|
|
if (impl->decoded_bytes_per_packet > SWITCH_RECOMMENDED_BUFFER_SIZE) {
|
2008-05-21 22:26:30 +00:00
|
|
|
load_interface = 0;
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CRIT,
|
2008-05-27 04:30:03 +00:00
|
|
|
"Failed to load codec interface %s from %s due to bytes per frame exceeding buffer size.\n", ptr->interface_name,
|
|
|
|
key);
|
2007-02-23 19:15:43 +00:00
|
|
|
break;
|
2007-02-23 18:24:31 +00:00
|
|
|
}
|
2006-06-23 20:14:29 +00:00
|
|
|
}
|
2008-05-21 22:26:30 +00:00
|
|
|
if (load_interface) {
|
2007-02-23 19:15:43 +00:00
|
|
|
for (impl = ptr->implementations; impl; impl = impl->next) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE,
|
|
|
|
"Adding Codec '%s' (%s) %dhz %dms\n",
|
2008-10-20 17:48:42 +00:00
|
|
|
impl->iananame, ptr->interface_name, impl->actual_samples_per_second, impl->microseconds_per_packet / 1000);
|
2007-02-23 19:15:43 +00:00
|
|
|
if (!switch_core_hash_find(loadable_modules.codec_hash, impl->iananame)) {
|
|
|
|
switch_core_hash_insert(loadable_modules.codec_hash, impl->iananame, (const void *) ptr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (switch_event_create(&event, SWITCH_EVENT_MODULE_LOAD) == SWITCH_STATUS_SUCCESS) {
|
2008-08-16 02:17:09 +00:00
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "type", "codec");
|
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "name", ptr->interface_name);
|
2007-02-23 19:15:43 +00:00
|
|
|
switch_event_fire(&event);
|
|
|
|
}
|
2007-02-23 18:24:31 +00:00
|
|
|
}
|
2006-04-25 00:33:00 +00:00
|
|
|
}
|
2007-03-29 22:31:56 +00:00
|
|
|
}
|
2006-04-25 00:33:00 +00:00
|
|
|
}
|
|
|
|
|
2006-04-30 18:24:24 +00:00
|
|
|
if (new_module->module_interface->dialplan_interface) {
|
2006-04-29 23:43:28 +00:00
|
|
|
const switch_dialplan_interface_t *ptr;
|
2006-04-25 00:33:00 +00:00
|
|
|
|
2006-04-30 18:24:24 +00:00
|
|
|
for (ptr = new_module->module_interface->dialplan_interface; ptr; ptr = ptr->next) {
|
2007-02-23 18:24:31 +00:00
|
|
|
if (!ptr->interface_name) {
|
2007-09-29 01:06:08 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CRIT, "Failed to load dialplan interface from %s due to no interface name.\n", key);
|
2007-02-23 18:24:31 +00:00
|
|
|
} else {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE, "Adding Dialplan '%s'\n", ptr->interface_name);
|
|
|
|
if (switch_event_create(&event, SWITCH_EVENT_MODULE_LOAD) == SWITCH_STATUS_SUCCESS) {
|
2008-08-16 02:17:09 +00:00
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "type", "dialplan");
|
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "name", ptr->interface_name);
|
2007-02-23 18:24:31 +00:00
|
|
|
switch_event_fire(&event);
|
|
|
|
}
|
2007-02-23 19:15:43 +00:00
|
|
|
switch_core_hash_insert(loadable_modules.dialplan_hash, ptr->interface_name, (const void *) ptr);
|
2006-05-01 19:44:21 +00:00
|
|
|
}
|
2006-04-25 00:33:00 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2006-04-30 18:24:24 +00:00
|
|
|
if (new_module->module_interface->timer_interface) {
|
2006-04-29 23:43:28 +00:00
|
|
|
const switch_timer_interface_t *ptr;
|
2006-04-25 00:33:00 +00:00
|
|
|
|
2006-04-30 18:24:24 +00:00
|
|
|
for (ptr = new_module->module_interface->timer_interface; ptr; ptr = ptr->next) {
|
2007-02-23 18:24:31 +00:00
|
|
|
if (!ptr->interface_name) {
|
2007-09-29 01:06:08 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CRIT, "Failed to load timer interface from %s due to no interface name.\n", key);
|
2007-02-23 18:24:31 +00:00
|
|
|
} else {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE, "Adding Timer '%s'\n", ptr->interface_name);
|
|
|
|
if (switch_event_create(&event, SWITCH_EVENT_MODULE_LOAD) == SWITCH_STATUS_SUCCESS) {
|
2008-08-16 02:17:09 +00:00
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "type", "timer");
|
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "name", ptr->interface_name);
|
2007-02-23 18:24:31 +00:00
|
|
|
switch_event_fire(&event);
|
|
|
|
}
|
2007-02-23 19:15:43 +00:00
|
|
|
switch_core_hash_insert(loadable_modules.timer_hash, ptr->interface_name, (const void *) ptr);
|
2006-05-01 19:44:21 +00:00
|
|
|
}
|
2006-04-25 00:33:00 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2006-04-30 18:24:24 +00:00
|
|
|
if (new_module->module_interface->application_interface) {
|
2006-04-29 23:43:28 +00:00
|
|
|
const switch_application_interface_t *ptr;
|
2006-04-25 00:33:00 +00:00
|
|
|
|
2006-04-30 18:24:24 +00:00
|
|
|
for (ptr = new_module->module_interface->application_interface; ptr; ptr = ptr->next) {
|
2007-02-23 18:24:31 +00:00
|
|
|
if (!ptr->interface_name) {
|
2007-09-29 01:06:08 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CRIT, "Failed to load application interface from %s due to no interface name.\n", key);
|
2007-02-23 18:24:31 +00:00
|
|
|
} else {
|
2007-03-30 00:13:31 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE, "Adding Application '%s'\n", ptr->interface_name);
|
2007-02-23 18:24:31 +00:00
|
|
|
if (switch_event_create(&event, SWITCH_EVENT_MODULE_LOAD) == SWITCH_STATUS_SUCCESS) {
|
2008-08-16 02:17:09 +00:00
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "type", "application");
|
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "name", ptr->interface_name);
|
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "description", switch_str_nil(ptr->short_desc));
|
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "syntax", switch_str_nil(ptr->syntax));
|
2007-02-23 18:24:31 +00:00
|
|
|
switch_event_fire(&event);
|
|
|
|
}
|
2007-02-23 19:15:43 +00:00
|
|
|
switch_core_hash_insert(loadable_modules.application_hash, ptr->interface_name, (const void *) ptr);
|
2006-05-01 19:44:21 +00:00
|
|
|
}
|
2006-04-25 00:33:00 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2006-04-30 18:24:24 +00:00
|
|
|
if (new_module->module_interface->api_interface) {
|
2006-04-29 23:43:28 +00:00
|
|
|
const switch_api_interface_t *ptr;
|
2006-04-25 00:33:00 +00:00
|
|
|
|
2006-04-30 18:24:24 +00:00
|
|
|
for (ptr = new_module->module_interface->api_interface; ptr; ptr = ptr->next) {
|
2007-02-23 18:24:31 +00:00
|
|
|
if (!ptr->interface_name) {
|
2007-09-29 01:06:08 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CRIT, "Failed to load api interface from %s due to no interface name.\n", key);
|
2007-02-23 18:24:31 +00:00
|
|
|
} else {
|
2007-03-30 00:13:31 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE, "Adding API Function '%s'\n", ptr->interface_name);
|
2007-02-23 18:24:31 +00:00
|
|
|
if (switch_event_create(&event, SWITCH_EVENT_MODULE_LOAD) == SWITCH_STATUS_SUCCESS) {
|
2008-08-16 02:17:09 +00:00
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "type", "api");
|
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "name", ptr->interface_name);
|
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "description", switch_str_nil(ptr->desc));
|
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "syntax", switch_str_nil(ptr->syntax));
|
2007-02-23 18:24:31 +00:00
|
|
|
switch_event_fire(&event);
|
|
|
|
}
|
2007-02-23 19:15:43 +00:00
|
|
|
switch_core_hash_insert(loadable_modules.api_hash, ptr->interface_name, (const void *) ptr);
|
2006-05-01 19:44:21 +00:00
|
|
|
}
|
2006-04-25 00:33:00 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2006-04-30 18:24:24 +00:00
|
|
|
if (new_module->module_interface->file_interface) {
|
2006-04-29 23:43:28 +00:00
|
|
|
const switch_file_interface_t *ptr;
|
2006-04-25 00:33:00 +00:00
|
|
|
|
2006-04-30 18:24:24 +00:00
|
|
|
for (ptr = new_module->module_interface->file_interface; ptr; ptr = ptr->next) {
|
2007-02-23 18:24:31 +00:00
|
|
|
if (!ptr->interface_name) {
|
2007-09-29 01:06:08 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CRIT, "Failed to load file interface from %s due to no interface name.\n", key);
|
2007-02-23 18:24:31 +00:00
|
|
|
} else {
|
|
|
|
int i;
|
|
|
|
for (i = 0; ptr->extens[i]; i++) {
|
2007-03-30 00:13:31 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE, "Adding File Format '%s'\n", ptr->extens[i]);
|
2007-03-29 22:31:56 +00:00
|
|
|
if (switch_event_create(&event, SWITCH_EVENT_MODULE_LOAD) == SWITCH_STATUS_SUCCESS) {
|
2008-08-16 02:17:09 +00:00
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "type", "file");
|
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "name", ptr->extens[i]);
|
2007-03-29 22:31:56 +00:00
|
|
|
switch_event_fire(&event);
|
|
|
|
}
|
2007-02-23 19:15:43 +00:00
|
|
|
switch_core_hash_insert(loadable_modules.file_hash, ptr->extens[i], (const void *) ptr);
|
2007-02-23 18:24:31 +00:00
|
|
|
}
|
2006-04-25 00:33:00 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2006-04-30 18:24:24 +00:00
|
|
|
if (new_module->module_interface->speech_interface) {
|
2006-04-29 23:43:28 +00:00
|
|
|
const switch_speech_interface_t *ptr;
|
2006-04-25 00:33:00 +00:00
|
|
|
|
2006-04-30 18:24:24 +00:00
|
|
|
for (ptr = new_module->module_interface->speech_interface; ptr; ptr = ptr->next) {
|
2007-02-23 18:24:31 +00:00
|
|
|
if (!ptr->interface_name) {
|
2007-09-29 01:06:08 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CRIT, "Failed to load speech interface from %s due to no interface name.\n", key);
|
2007-02-23 18:24:31 +00:00
|
|
|
} else {
|
2007-03-30 00:13:31 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE, "Adding Speech interface '%s'\n", ptr->interface_name);
|
2007-02-23 18:24:31 +00:00
|
|
|
if (switch_event_create(&event, SWITCH_EVENT_MODULE_LOAD) == SWITCH_STATUS_SUCCESS) {
|
2008-08-16 02:17:09 +00:00
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "type", "speech");
|
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "name", ptr->interface_name);
|
2007-02-23 18:24:31 +00:00
|
|
|
switch_event_fire(&event);
|
|
|
|
}
|
2007-02-23 19:15:43 +00:00
|
|
|
switch_core_hash_insert(loadable_modules.speech_hash, ptr->interface_name, (const void *) ptr);
|
2006-05-01 19:44:21 +00:00
|
|
|
}
|
2006-04-25 00:33:00 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2006-11-09 05:39:04 +00:00
|
|
|
if (new_module->module_interface->asr_interface) {
|
|
|
|
const switch_asr_interface_t *ptr;
|
|
|
|
|
|
|
|
for (ptr = new_module->module_interface->asr_interface; ptr; ptr = ptr->next) {
|
2007-02-23 18:24:31 +00:00
|
|
|
if (!ptr->interface_name) {
|
2007-09-29 01:06:08 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CRIT, "Failed to load asr interface from %s due to no interface name.\n", key);
|
2007-02-23 18:24:31 +00:00
|
|
|
} else {
|
2008-06-17 22:09:59 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE, "Adding ASR interface '%s'\n", ptr->interface_name);
|
2007-02-23 18:24:31 +00:00
|
|
|
if (switch_event_create(&event, SWITCH_EVENT_MODULE_LOAD) == SWITCH_STATUS_SUCCESS) {
|
2008-08-16 02:17:09 +00:00
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "type", "asr");
|
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "name", ptr->interface_name);
|
2007-02-23 18:24:31 +00:00
|
|
|
switch_event_fire(&event);
|
|
|
|
}
|
2007-02-23 19:15:43 +00:00
|
|
|
switch_core_hash_insert(loadable_modules.asr_hash, ptr->interface_name, (const void *) ptr);
|
2006-11-09 05:39:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2006-04-30 18:24:24 +00:00
|
|
|
if (new_module->module_interface->directory_interface) {
|
2006-04-29 23:43:28 +00:00
|
|
|
const switch_directory_interface_t *ptr;
|
2006-04-25 00:33:00 +00:00
|
|
|
|
2006-04-30 18:24:24 +00:00
|
|
|
for (ptr = new_module->module_interface->directory_interface; ptr; ptr = ptr->next) {
|
2007-02-23 18:24:31 +00:00
|
|
|
if (!ptr->interface_name) {
|
2007-09-29 01:06:08 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CRIT, "Failed to load directory interface from %s due to no interface name.\n", key);
|
2007-02-23 18:24:31 +00:00
|
|
|
} else {
|
2007-03-30 00:13:31 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE, "Adding Directory interface '%s'\n", ptr->interface_name);
|
2007-02-23 18:24:31 +00:00
|
|
|
if (switch_event_create(&event, SWITCH_EVENT_MODULE_LOAD) == SWITCH_STATUS_SUCCESS) {
|
2008-08-16 02:17:09 +00:00
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "type", "directory");
|
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "name", ptr->interface_name);
|
2007-02-23 18:24:31 +00:00
|
|
|
switch_event_fire(&event);
|
|
|
|
}
|
2007-02-23 19:15:43 +00:00
|
|
|
switch_core_hash_insert(loadable_modules.directory_hash, ptr->interface_name, (const void *) ptr);
|
2006-05-01 19:44:21 +00:00
|
|
|
}
|
2006-04-25 00:33:00 +00:00
|
|
|
}
|
|
|
|
}
|
2006-10-19 07:13:34 +00:00
|
|
|
|
|
|
|
if (new_module->module_interface->chat_interface) {
|
|
|
|
const switch_chat_interface_t *ptr;
|
|
|
|
|
|
|
|
for (ptr = new_module->module_interface->chat_interface; ptr; ptr = ptr->next) {
|
2007-02-23 18:24:31 +00:00
|
|
|
if (!ptr->interface_name) {
|
2007-09-29 01:06:08 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CRIT, "Failed to load chat interface from %s due to no interface name.\n", key);
|
2007-02-23 18:24:31 +00:00
|
|
|
} else {
|
2007-03-30 00:13:31 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE, "Adding Chat interface '%s'\n", ptr->interface_name);
|
2007-02-23 18:24:31 +00:00
|
|
|
if (switch_event_create(&event, SWITCH_EVENT_MODULE_LOAD) == SWITCH_STATUS_SUCCESS) {
|
2008-08-16 02:17:09 +00:00
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "type", "chat");
|
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "name", ptr->interface_name);
|
2007-02-23 18:24:31 +00:00
|
|
|
switch_event_fire(&event);
|
|
|
|
}
|
2007-02-23 19:15:43 +00:00
|
|
|
switch_core_hash_insert(loadable_modules.chat_hash, ptr->interface_name, (const void *) ptr);
|
2006-10-19 07:13:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2006-12-20 21:25:14 +00:00
|
|
|
|
|
|
|
if (new_module->module_interface->say_interface) {
|
|
|
|
const switch_say_interface_t *ptr;
|
|
|
|
|
|
|
|
for (ptr = new_module->module_interface->say_interface; ptr; ptr = ptr->next) {
|
2007-02-23 18:24:31 +00:00
|
|
|
if (!ptr->interface_name) {
|
2007-09-29 01:06:08 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CRIT, "Failed to load say interface from %s due to no interface name.\n", key);
|
2007-02-23 18:24:31 +00:00
|
|
|
} else {
|
2007-03-30 00:13:31 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE, "Adding Say interface '%s'\n", ptr->interface_name);
|
2007-02-23 18:24:31 +00:00
|
|
|
if (switch_event_create(&event, SWITCH_EVENT_MODULE_LOAD) == SWITCH_STATUS_SUCCESS) {
|
2008-08-16 02:17:09 +00:00
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "type", "say");
|
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "name", ptr->interface_name);
|
2007-02-23 18:24:31 +00:00
|
|
|
switch_event_fire(&event);
|
|
|
|
}
|
2007-02-23 19:15:43 +00:00
|
|
|
switch_core_hash_insert(loadable_modules.say_hash, ptr->interface_name, (const void *) ptr);
|
2006-12-20 21:25:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2007-03-07 18:34:22 +00:00
|
|
|
|
|
|
|
if (new_module->module_interface->management_interface) {
|
|
|
|
const switch_management_interface_t *ptr;
|
|
|
|
|
|
|
|
for (ptr = new_module->module_interface->management_interface; ptr; ptr = ptr->next) {
|
|
|
|
if (!ptr->relative_oid) {
|
2007-09-29 01:06:08 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CRIT, "Failed to load management interface from %s due to no interface name.\n", key);
|
2007-03-07 18:34:22 +00:00
|
|
|
} else {
|
2007-03-17 19:51:08 +00:00
|
|
|
if (switch_core_hash_find(loadable_modules.management_hash, ptr->relative_oid)) {
|
2007-09-29 01:06:08 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CRIT,
|
2007-03-30 00:13:31 +00:00
|
|
|
"Failed to load management interface %s. OID %s already exists\n", key, ptr->relative_oid);
|
2007-03-17 19:51:08 +00:00
|
|
|
} else {
|
2007-03-29 22:31:56 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE,
|
2007-03-30 00:13:31 +00:00
|
|
|
"Adding Management interface '%s' OID[%s.%s]\n", key, FREESWITCH_OID_PREFIX, ptr->relative_oid);
|
2007-03-17 19:51:08 +00:00
|
|
|
switch_core_hash_insert(loadable_modules.management_hash, ptr->relative_oid, (const void *) ptr);
|
|
|
|
if (switch_event_create(&event, SWITCH_EVENT_MODULE_LOAD) == SWITCH_STATUS_SUCCESS) {
|
2008-08-16 02:17:09 +00:00
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "type", "management");
|
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "name", ptr->relative_oid);
|
2007-03-17 19:51:08 +00:00
|
|
|
switch_event_fire(&event);
|
|
|
|
}
|
2007-03-07 18:34:22 +00:00
|
|
|
}
|
2007-03-29 22:31:56 +00:00
|
|
|
|
2007-03-07 18:34:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2007-03-29 22:31:56 +00:00
|
|
|
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_mutex_unlock(loadable_modules.mutex);
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static switch_status_t switch_loadable_module_unprocess(switch_loadable_module_t *old_module)
|
|
|
|
{
|
|
|
|
switch_event_t *event;
|
|
|
|
|
|
|
|
switch_mutex_lock(loadable_modules.mutex);
|
2008-11-12 17:10:20 +00:00
|
|
|
|
2007-05-03 16:28:23 +00:00
|
|
|
if (old_module->module_interface->endpoint_interface) {
|
|
|
|
const switch_endpoint_interface_t *ptr;
|
2008-07-01 23:41:09 +00:00
|
|
|
|
2007-05-03 16:28:23 +00:00
|
|
|
for (ptr = old_module->module_interface->endpoint_interface; ptr; ptr = ptr->next) {
|
|
|
|
if (ptr->interface_name) {
|
2008-07-01 23:41:09 +00:00
|
|
|
|
2008-11-12 20:45:18 +00:00
|
|
|
switch_core_session_hupall_endpoint(ptr, SWITCH_CAUSE_MANAGER_REQUEST);
|
2008-07-01 23:41:09 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "Write lock interface '%s' to wait for existing references.\n", ptr->interface_name);
|
2008-09-09 15:25:31 +00:00
|
|
|
if (switch_thread_rwlock_trywrlock_timeout(ptr->rwlock, 10) == SWITCH_STATUS_SUCCESS) {
|
|
|
|
switch_thread_rwlock_unlock(ptr->rwlock);
|
|
|
|
} else {
|
2008-11-19 19:22:20 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Giving up on '%s' waiting for existing references.\n", ptr->interface_name);
|
2008-09-09 15:25:31 +00:00
|
|
|
}
|
2008-07-01 23:41:09 +00:00
|
|
|
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE, "Deleting Endpoint '%s'\n", ptr->interface_name);
|
|
|
|
switch_core_hash_delete(loadable_modules.endpoint_hash, ptr->interface_name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (old_module->module_interface->codec_interface) {
|
|
|
|
const switch_codec_implementation_t *impl;
|
|
|
|
const switch_codec_interface_t *ptr;
|
|
|
|
|
|
|
|
for (ptr = old_module->module_interface->codec_interface; ptr; ptr = ptr->next) {
|
|
|
|
if (ptr->interface_name) {
|
|
|
|
unsigned load_interface = 1;
|
|
|
|
for (impl = ptr->implementations; impl; impl = impl->next) {
|
|
|
|
if (!impl->iananame) {
|
|
|
|
load_interface = 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (load_interface) {
|
|
|
|
for (impl = ptr->implementations; impl; impl = impl->next) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE,
|
|
|
|
"Deleting Codec '%s' (%s) %dhz %dms\n",
|
2008-10-20 17:48:42 +00:00
|
|
|
impl->iananame, ptr->interface_name, impl->actual_samples_per_second, impl->microseconds_per_packet / 1000);
|
2008-11-12 20:45:18 +00:00
|
|
|
switch_core_session_hupall_matching_var("read_codec", impl->iananame, SWITCH_CAUSE_MANAGER_REQUEST);
|
|
|
|
switch_core_session_hupall_matching_var("write_codec", impl->iananame, SWITCH_CAUSE_MANAGER_REQUEST);
|
2007-05-03 16:28:23 +00:00
|
|
|
if (switch_core_hash_find(loadable_modules.codec_hash, impl->iananame)) {
|
|
|
|
switch_core_hash_delete(loadable_modules.codec_hash, impl->iananame);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (switch_event_create(&event, SWITCH_EVENT_MODULE_UNLOAD) == SWITCH_STATUS_SUCCESS) {
|
2008-08-16 02:17:09 +00:00
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "type", "codec");
|
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "name", ptr->interface_name);
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_event_fire(&event);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (old_module->module_interface->dialplan_interface) {
|
|
|
|
const switch_dialplan_interface_t *ptr;
|
|
|
|
|
|
|
|
for (ptr = old_module->module_interface->dialplan_interface; ptr; ptr = ptr->next) {
|
|
|
|
if (ptr->interface_name) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE, "Deleting Dialplan '%s'\n", ptr->interface_name);
|
|
|
|
if (switch_event_create(&event, SWITCH_EVENT_MODULE_UNLOAD) == SWITCH_STATUS_SUCCESS) {
|
2008-08-16 02:17:09 +00:00
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "type", "dialplan");
|
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "name", ptr->interface_name);
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_event_fire(&event);
|
|
|
|
}
|
|
|
|
switch_core_hash_delete(loadable_modules.dialplan_hash, ptr->interface_name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (old_module->module_interface->timer_interface) {
|
|
|
|
const switch_timer_interface_t *ptr;
|
|
|
|
|
|
|
|
for (ptr = old_module->module_interface->timer_interface; ptr; ptr = ptr->next) {
|
|
|
|
if (ptr->interface_name) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE, "Deleting Timer '%s'\n", ptr->interface_name);
|
|
|
|
if (switch_event_create(&event, SWITCH_EVENT_MODULE_UNLOAD) == SWITCH_STATUS_SUCCESS) {
|
2008-08-16 02:17:09 +00:00
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "type", "timer");
|
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "name", ptr->interface_name);
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_event_fire(&event);
|
|
|
|
}
|
|
|
|
switch_core_hash_delete(loadable_modules.timer_hash, ptr->interface_name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (old_module->module_interface->application_interface) {
|
|
|
|
const switch_application_interface_t *ptr;
|
|
|
|
for (ptr = old_module->module_interface->application_interface; ptr; ptr = ptr->next) {
|
|
|
|
if (ptr->interface_name) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE, "Deleting Application '%s'\n", ptr->interface_name);
|
2008-11-12 20:45:18 +00:00
|
|
|
switch_core_session_hupall_matching_var(SWITCH_CURRENT_APPLICATION_VARIABLE, ptr->interface_name, SWITCH_CAUSE_MANAGER_REQUEST);
|
2008-07-01 23:41:09 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "Write lock interface '%s' to wait for existing references.\n", ptr->interface_name);
|
2008-09-09 15:25:31 +00:00
|
|
|
if (switch_thread_rwlock_trywrlock_timeout(ptr->rwlock, 10) == SWITCH_STATUS_SUCCESS) {
|
|
|
|
switch_thread_rwlock_unlock(ptr->rwlock);
|
|
|
|
} else {
|
2008-11-19 19:22:20 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Giving up on '%s' waiting for existing references.\n", ptr->interface_name);
|
2008-09-09 15:25:31 +00:00
|
|
|
}
|
|
|
|
|
2007-05-03 16:28:23 +00:00
|
|
|
if (switch_event_create(&event, SWITCH_EVENT_MODULE_UNLOAD) == SWITCH_STATUS_SUCCESS) {
|
2008-08-16 02:17:09 +00:00
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "type", "application");
|
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "name", ptr->interface_name);
|
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "description", switch_str_nil(ptr->short_desc));
|
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "syntax", switch_str_nil(ptr->syntax));
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_event_fire(&event);
|
|
|
|
}
|
|
|
|
switch_core_hash_delete(loadable_modules.application_hash, ptr->interface_name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (old_module->module_interface->api_interface) {
|
|
|
|
const switch_api_interface_t *ptr;
|
|
|
|
|
|
|
|
for (ptr = old_module->module_interface->api_interface; ptr; ptr = ptr->next) {
|
|
|
|
if (ptr->interface_name) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE, "Deleting API Function '%s'\n", ptr->interface_name);
|
2008-07-01 23:41:09 +00:00
|
|
|
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "Write lock interface '%s' to wait for existing references.\n", ptr->interface_name);
|
2008-09-09 15:25:31 +00:00
|
|
|
|
|
|
|
if (switch_thread_rwlock_trywrlock_timeout(ptr->rwlock, 10) == SWITCH_STATUS_SUCCESS) {
|
|
|
|
switch_thread_rwlock_unlock(ptr->rwlock);
|
|
|
|
} else {
|
2008-11-19 19:22:20 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Giving up on '%s' waiting for existing references.\n", ptr->interface_name);
|
2008-09-09 15:25:31 +00:00
|
|
|
}
|
|
|
|
|
2008-07-01 23:41:09 +00:00
|
|
|
|
2007-05-03 16:28:23 +00:00
|
|
|
if (switch_event_create(&event, SWITCH_EVENT_MODULE_UNLOAD) == SWITCH_STATUS_SUCCESS) {
|
2008-08-16 02:17:09 +00:00
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "type", "api");
|
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "name", ptr->interface_name);
|
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "description", switch_str_nil(ptr->desc));
|
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "syntax", switch_str_nil(ptr->syntax));
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_event_fire(&event);
|
|
|
|
}
|
|
|
|
switch_core_hash_delete(loadable_modules.api_hash, ptr->interface_name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (old_module->module_interface->file_interface) {
|
|
|
|
const switch_file_interface_t *ptr;
|
|
|
|
|
|
|
|
for (ptr = old_module->module_interface->file_interface; ptr; ptr = ptr->next) {
|
|
|
|
if (ptr->interface_name) {
|
|
|
|
int i;
|
2008-11-12 19:28:05 +00:00
|
|
|
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "Write lock interface '%s' to wait for existing references.\n", ptr->interface_name);
|
|
|
|
|
|
|
|
if (switch_thread_rwlock_trywrlock_timeout(ptr->rwlock, 10) == SWITCH_STATUS_SUCCESS) {
|
|
|
|
switch_thread_rwlock_unlock(ptr->rwlock);
|
|
|
|
} else {
|
2008-11-19 19:22:20 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Giving up on '%s' waiting for existing references.\n", ptr->interface_name);
|
2008-11-12 19:28:05 +00:00
|
|
|
}
|
|
|
|
|
2007-05-03 16:28:23 +00:00
|
|
|
for (i = 0; ptr->extens[i]; i++) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE, "Deleting File Format '%s'\n", ptr->extens[i]);
|
|
|
|
if (switch_event_create(&event, SWITCH_EVENT_MODULE_UNLOAD) == SWITCH_STATUS_SUCCESS) {
|
2008-08-16 02:17:09 +00:00
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "type", "file");
|
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "name", ptr->extens[i]);
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_event_fire(&event);
|
|
|
|
}
|
|
|
|
switch_core_hash_delete(loadable_modules.file_hash, ptr->extens[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (old_module->module_interface->speech_interface) {
|
|
|
|
const switch_speech_interface_t *ptr;
|
|
|
|
|
|
|
|
for (ptr = old_module->module_interface->speech_interface; ptr; ptr = ptr->next) {
|
2008-11-12 19:28:05 +00:00
|
|
|
|
2007-05-03 16:28:23 +00:00
|
|
|
if (ptr->interface_name) {
|
2008-11-12 19:28:05 +00:00
|
|
|
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "Write lock interface '%s' to wait for existing references.\n", ptr->interface_name);
|
|
|
|
|
|
|
|
if (switch_thread_rwlock_trywrlock_timeout(ptr->rwlock, 10) == SWITCH_STATUS_SUCCESS) {
|
|
|
|
switch_thread_rwlock_unlock(ptr->rwlock);
|
|
|
|
} else {
|
2008-11-19 19:22:20 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Giving up on '%s' waiting for existing references.\n", ptr->interface_name);
|
2008-11-12 19:28:05 +00:00
|
|
|
}
|
|
|
|
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE, "Deleting Speech interface '%s'\n", ptr->interface_name);
|
|
|
|
if (switch_event_create(&event, SWITCH_EVENT_MODULE_UNLOAD) == SWITCH_STATUS_SUCCESS) {
|
2008-08-16 02:17:09 +00:00
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "type", "speech");
|
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "name", ptr->interface_name);
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_event_fire(&event);
|
|
|
|
}
|
|
|
|
switch_core_hash_delete(loadable_modules.speech_hash, ptr->interface_name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (old_module->module_interface->asr_interface) {
|
|
|
|
const switch_asr_interface_t *ptr;
|
|
|
|
|
|
|
|
for (ptr = old_module->module_interface->asr_interface; ptr; ptr = ptr->next) {
|
|
|
|
if (ptr->interface_name) {
|
2008-11-12 19:28:05 +00:00
|
|
|
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "Write lock interface '%s' to wait for existing references.\n", ptr->interface_name);
|
|
|
|
|
|
|
|
if (switch_thread_rwlock_trywrlock_timeout(ptr->rwlock, 10) == SWITCH_STATUS_SUCCESS) {
|
|
|
|
switch_thread_rwlock_unlock(ptr->rwlock);
|
|
|
|
} else {
|
2008-11-19 19:22:20 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Giving up on '%s' waiting for existing references.\n", ptr->interface_name);
|
2008-11-12 19:28:05 +00:00
|
|
|
}
|
|
|
|
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE, "Deleting Asr interface '%s'\n", ptr->interface_name);
|
|
|
|
if (switch_event_create(&event, SWITCH_EVENT_MODULE_UNLOAD) == SWITCH_STATUS_SUCCESS) {
|
2008-08-16 02:17:09 +00:00
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "type", "asr");
|
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "name", ptr->interface_name);
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_event_fire(&event);
|
|
|
|
}
|
|
|
|
switch_core_hash_delete(loadable_modules.asr_hash, ptr->interface_name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (old_module->module_interface->directory_interface) {
|
|
|
|
const switch_directory_interface_t *ptr;
|
|
|
|
|
|
|
|
for (ptr = old_module->module_interface->directory_interface; ptr; ptr = ptr->next) {
|
|
|
|
if (ptr->interface_name) {
|
2008-11-12 19:28:05 +00:00
|
|
|
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "Write lock interface '%s' to wait for existing references.\n", ptr->interface_name);
|
|
|
|
|
|
|
|
if (switch_thread_rwlock_trywrlock_timeout(ptr->rwlock, 10) == SWITCH_STATUS_SUCCESS) {
|
|
|
|
switch_thread_rwlock_unlock(ptr->rwlock);
|
|
|
|
} else {
|
2008-11-19 19:22:20 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Giving up on '%s' waiting for existing references.\n", ptr->interface_name);
|
2008-11-12 19:28:05 +00:00
|
|
|
}
|
|
|
|
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE, "Deleting Directory interface '%s'\n", ptr->interface_name);
|
|
|
|
if (switch_event_create(&event, SWITCH_EVENT_MODULE_UNLOAD) == SWITCH_STATUS_SUCCESS) {
|
2008-08-16 02:17:09 +00:00
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "type", "directory");
|
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "name", ptr->interface_name);
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_event_fire(&event);
|
|
|
|
}
|
|
|
|
switch_core_hash_delete(loadable_modules.directory_hash, ptr->interface_name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (old_module->module_interface->chat_interface) {
|
|
|
|
const switch_chat_interface_t *ptr;
|
|
|
|
|
|
|
|
for (ptr = old_module->module_interface->chat_interface; ptr; ptr = ptr->next) {
|
2008-05-27 04:30:03 +00:00
|
|
|
if (ptr->interface_name) {
|
2008-11-12 19:28:05 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "Write lock interface '%s' to wait for existing references.\n", ptr->interface_name);
|
|
|
|
|
|
|
|
if (switch_thread_rwlock_trywrlock_timeout(ptr->rwlock, 10) == SWITCH_STATUS_SUCCESS) {
|
|
|
|
switch_thread_rwlock_unlock(ptr->rwlock);
|
|
|
|
} else {
|
2008-11-19 19:22:20 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Giving up on '%s' waiting for existing references.\n", ptr->interface_name);
|
2008-11-12 19:28:05 +00:00
|
|
|
}
|
|
|
|
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE, "Deleting Chat interface '%s'\n", ptr->interface_name);
|
|
|
|
if (switch_event_create(&event, SWITCH_EVENT_MODULE_UNLOAD) == SWITCH_STATUS_SUCCESS) {
|
2008-08-16 02:17:09 +00:00
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "type", "chat");
|
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "name", ptr->interface_name);
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_event_fire(&event);
|
|
|
|
}
|
|
|
|
switch_core_hash_delete(loadable_modules.chat_hash, ptr->interface_name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (old_module->module_interface->say_interface) {
|
|
|
|
const switch_say_interface_t *ptr;
|
|
|
|
|
|
|
|
for (ptr = old_module->module_interface->say_interface; ptr; ptr = ptr->next) {
|
|
|
|
if (ptr->interface_name) {
|
2008-11-12 19:28:05 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "Write lock interface '%s' to wait for existing references.\n", ptr->interface_name);
|
|
|
|
|
|
|
|
if (switch_thread_rwlock_trywrlock_timeout(ptr->rwlock, 10) == SWITCH_STATUS_SUCCESS) {
|
|
|
|
switch_thread_rwlock_unlock(ptr->rwlock);
|
|
|
|
} else {
|
2008-11-19 19:22:20 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Giving up on '%s' waiting for existing references.\n", ptr->interface_name);
|
2008-11-12 19:28:05 +00:00
|
|
|
}
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE, "Deleting Say interface '%s'\n", ptr->interface_name);
|
|
|
|
if (switch_event_create(&event, SWITCH_EVENT_MODULE_UNLOAD) == SWITCH_STATUS_SUCCESS) {
|
2008-08-16 02:17:09 +00:00
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "type", "say");
|
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "name", ptr->interface_name);
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_event_fire(&event);
|
|
|
|
}
|
|
|
|
switch_core_hash_delete(loadable_modules.say_hash, ptr->interface_name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (old_module->module_interface->management_interface) {
|
|
|
|
const switch_management_interface_t *ptr;
|
|
|
|
|
|
|
|
for (ptr = old_module->module_interface->management_interface; ptr; ptr = ptr->next) {
|
|
|
|
if (ptr->relative_oid) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE,
|
|
|
|
"Deleting Management interface '%s' OID[%s.%s]\n", old_module->key, FREESWITCH_OID_PREFIX, ptr->relative_oid);
|
|
|
|
switch_core_hash_delete(loadable_modules.management_hash, ptr->relative_oid);
|
|
|
|
if (switch_event_create(&event, SWITCH_EVENT_MODULE_UNLOAD) == SWITCH_STATUS_SUCCESS) {
|
2008-08-16 02:17:09 +00:00
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "type", "management");
|
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "name", ptr->relative_oid);
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_event_fire(&event);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2008-11-12 17:10:20 +00:00
|
|
|
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_mutex_unlock(loadable_modules.mutex);
|
2006-04-25 00:33:00 +00:00
|
|
|
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
|
|
|
|
}
|
2006-03-30 23:02:50 +00:00
|
|
|
|
2007-05-03 16:28:23 +00:00
|
|
|
|
2008-11-06 17:29:50 +00:00
|
|
|
static switch_status_t switch_loadable_module_load_file(char *path, char *filename, switch_bool_t global, switch_loadable_module_t **new_module)
|
2005-11-19 20:07:43 +00:00
|
|
|
{
|
2006-04-29 06:05:03 +00:00
|
|
|
switch_loadable_module_t *module = NULL;
|
2008-11-06 17:29:50 +00:00
|
|
|
switch_dso_lib_t dso = NULL;
|
2006-02-13 18:34:39 +00:00
|
|
|
apr_status_t status = SWITCH_STATUS_SUCCESS;
|
2008-11-06 17:29:50 +00:00
|
|
|
switch_loadable_module_function_table_t *interface_struct_handle = NULL;
|
2007-06-13 14:35:55 +00:00
|
|
|
switch_loadable_module_function_table_t *mod_interface_functions = NULL;
|
|
|
|
char *struct_name = NULL;
|
2006-04-25 00:33:00 +00:00
|
|
|
switch_module_load_t load_func_ptr = NULL;
|
2005-11-19 20:07:43 +00:00
|
|
|
int loading = 1;
|
2007-06-13 20:40:06 +00:00
|
|
|
switch_loadable_module_interface_t *module_interface = NULL;
|
2008-11-06 17:29:50 +00:00
|
|
|
char *derr = NULL;
|
|
|
|
const char *err = NULL;
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_memory_pool_t *pool;
|
2008-11-08 11:21:54 +00:00
|
|
|
switch_bool_t load_global = global;
|
2005-11-19 20:07:43 +00:00
|
|
|
|
2007-12-11 19:23:57 +00:00
|
|
|
switch_assert(path != NULL);
|
2005-11-19 20:07:43 +00:00
|
|
|
|
2007-10-03 16:44:11 +00:00
|
|
|
switch_core_new_memory_pool(&pool);
|
2005-11-19 20:07:43 +00:00
|
|
|
*new_module = NULL;
|
|
|
|
|
2007-10-03 16:44:11 +00:00
|
|
|
struct_name = switch_core_sprintf(pool, "%s_module_interface", filename);
|
|
|
|
|
2007-10-12 22:54:18 +00:00
|
|
|
#ifdef WIN32
|
2008-11-08 11:21:54 +00:00
|
|
|
dso = switch_dso_open("FreeSwitch.dll", load_global, &derr);
|
2007-10-12 22:56:15 +00:00
|
|
|
#elif defined (MACOSX) || defined(DARWIN)
|
2008-11-08 11:21:54 +00:00
|
|
|
dso = switch_dso_open(SWITCH_PREFIX_DIR "/lib/libfreeswitch.dylib", load_global, &derr);
|
2007-10-12 22:54:18 +00:00
|
|
|
#else
|
2008-11-08 11:21:54 +00:00
|
|
|
dso = switch_dso_open(NULL, load_global, &derr);
|
2007-10-12 22:54:18 +00:00
|
|
|
#endif
|
2008-11-06 17:29:50 +00:00
|
|
|
if (!derr && dso) {
|
|
|
|
interface_struct_handle = switch_dso_data_sym(dso, struct_name, &derr);
|
|
|
|
}
|
|
|
|
|
|
|
|
switch_safe_free(derr)
|
2007-10-03 16:44:11 +00:00
|
|
|
|
|
|
|
if (!interface_struct_handle) {
|
2008-11-08 11:21:54 +00:00
|
|
|
dso = switch_dso_open(path, load_global, &derr);
|
2007-05-03 16:28:23 +00:00
|
|
|
}
|
|
|
|
|
2005-11-19 20:07:43 +00:00
|
|
|
while (loading) {
|
2008-11-06 17:29:50 +00:00
|
|
|
if (derr) {
|
2005-11-19 20:07:43 +00:00
|
|
|
err = derr;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2007-10-03 16:44:11 +00:00
|
|
|
if (!interface_struct_handle) {
|
2008-11-06 17:29:50 +00:00
|
|
|
interface_struct_handle = switch_dso_data_sym(dso, struct_name, &derr);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (derr) {
|
|
|
|
err = derr;
|
|
|
|
break;
|
2007-10-03 16:44:11 +00:00
|
|
|
}
|
|
|
|
|
2008-11-08 11:21:54 +00:00
|
|
|
if (interface_struct_handle && interface_struct_handle->switch_api_version != SWITCH_API_VERSION) {
|
|
|
|
err = "Trying to load an out of date module, please rebuild the module.";
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!load_global && interface_struct_handle && switch_test_flag(interface_struct_handle, SMODF_GLOBAL_SYMBOLS)) {
|
|
|
|
load_global = SWITCH_TRUE;
|
|
|
|
switch_dso_destroy(&dso);
|
|
|
|
interface_struct_handle = NULL;
|
|
|
|
dso = switch_dso_open(path, load_global, &derr);
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_DEBUG, "Loading module with global namespace at request of module\n");
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2007-06-13 14:41:49 +00:00
|
|
|
if (interface_struct_handle) {
|
|
|
|
mod_interface_functions = interface_struct_handle;
|
2007-06-13 14:35:55 +00:00
|
|
|
load_func_ptr = mod_interface_functions->load;
|
|
|
|
}
|
2005-11-19 20:07:43 +00:00
|
|
|
|
|
|
|
if (load_func_ptr == NULL) {
|
2008-01-07 11:21:24 +00:00
|
|
|
err = "Cannot locate symbol 'switch_module_load' please make sure this is a valid module.";
|
2005-11-19 20:07:43 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2007-06-13 20:40:06 +00:00
|
|
|
status = load_func_ptr(&module_interface, pool);
|
2007-05-03 16:28:23 +00:00
|
|
|
|
|
|
|
if (status != SWITCH_STATUS_SUCCESS && status != SWITCH_STATUS_NOUNLOAD) {
|
2005-11-19 20:07:43 +00:00
|
|
|
err = "Module load routine returned an error";
|
2006-04-30 18:24:24 +00:00
|
|
|
module_interface = NULL;
|
2005-11-19 20:07:43 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2007-05-03 16:28:23 +00:00
|
|
|
if ((module = switch_core_alloc(pool, sizeof(switch_loadable_module_t))) == 0) {
|
2005-11-19 20:07:43 +00:00
|
|
|
err = "Could not allocate memory\n";
|
2007-05-03 16:28:23 +00:00
|
|
|
abort();
|
2005-11-19 20:07:43 +00:00
|
|
|
}
|
|
|
|
|
2007-05-03 16:28:23 +00:00
|
|
|
if (status == SWITCH_STATUS_NOUNLOAD) {
|
2008-05-27 04:30:03 +00:00
|
|
|
module->perm++;
|
|
|
|
}
|
|
|
|
|
2005-11-19 20:07:43 +00:00
|
|
|
loading = 0;
|
|
|
|
}
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2005-11-19 20:07:43 +00:00
|
|
|
|
|
|
|
if (err) {
|
2007-05-03 16:28:23 +00:00
|
|
|
if (pool) {
|
|
|
|
switch_core_destroy_memory_pool(&pool);
|
|
|
|
}
|
2007-09-29 01:06:08 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CRIT, "Error Loading module %s\n**%s**\n", path, err);
|
2008-11-06 17:29:50 +00:00
|
|
|
switch_safe_free(derr);
|
2005-11-19 20:07:43 +00:00
|
|
|
return SWITCH_STATUS_GENERR;
|
|
|
|
}
|
|
|
|
|
2007-05-03 16:28:23 +00:00
|
|
|
module->pool = pool;
|
2007-06-13 14:35:55 +00:00
|
|
|
module->filename = switch_core_strdup(module->pool, path);
|
2006-04-30 18:24:24 +00:00
|
|
|
module->module_interface = module_interface;
|
2005-11-19 20:07:43 +00:00
|
|
|
module->switch_module_load = load_func_ptr;
|
|
|
|
|
2007-06-13 14:35:55 +00:00
|
|
|
if (mod_interface_functions) {
|
|
|
|
module->switch_module_shutdown = mod_interface_functions->shutdown;
|
|
|
|
module->switch_module_runtime = mod_interface_functions->runtime;
|
|
|
|
}
|
2005-11-19 20:07:43 +00:00
|
|
|
|
|
|
|
module->lib = dso;
|
|
|
|
|
|
|
|
*new_module = module;
|
2007-03-30 00:13:31 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CONSOLE, "Successfully Loaded [%s]\n", module_interface->module_name);
|
2005-11-19 20:07:43 +00:00
|
|
|
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
|
|
|
|
}
|
2007-05-03 16:28:23 +00:00
|
|
|
SWITCH_DECLARE(switch_status_t) switch_loadable_module_load_module(char *dir, char *fname, switch_bool_t runtime, const char **err)
|
2008-11-06 17:29:50 +00:00
|
|
|
{
|
|
|
|
return switch_loadable_module_load_module_ex(dir, fname, runtime, SWITCH_FALSE, err);
|
|
|
|
}
|
|
|
|
|
|
|
|
static switch_status_t switch_loadable_module_load_module_ex(char *dir, char *fname, switch_bool_t runtime, switch_bool_t global, const char **err)
|
2006-02-14 16:26:24 +00:00
|
|
|
{
|
2006-03-30 23:02:50 +00:00
|
|
|
switch_size_t len = 0;
|
2006-02-14 16:26:24 +00:00
|
|
|
char *path;
|
2007-06-13 14:35:55 +00:00
|
|
|
char *file, *dot;
|
2006-04-29 06:05:03 +00:00
|
|
|
switch_loadable_module_t *new_module = NULL;
|
2008-11-15 00:32:58 +00:00
|
|
|
switch_status_t status = SWITCH_STATUS_SUCCESS;
|
2006-02-14 16:26:24 +00:00
|
|
|
|
2006-02-14 17:31:20 +00:00
|
|
|
#ifdef WIN32
|
|
|
|
const char *ext = ".dll";
|
|
|
|
#else
|
|
|
|
const char *ext = ".so";
|
|
|
|
#endif
|
|
|
|
|
2008-11-15 00:32:58 +00:00
|
|
|
*err = "";
|
2006-02-14 17:31:20 +00:00
|
|
|
|
2006-02-20 00:23:25 +00:00
|
|
|
if ((file = switch_core_strdup(loadable_modules.pool, fname)) == 0) {
|
2008-11-15 00:32:58 +00:00
|
|
|
*err = "allocation error";
|
2006-04-20 00:58:06 +00:00
|
|
|
return SWITCH_STATUS_FALSE;
|
2006-02-14 16:26:24 +00:00
|
|
|
}
|
|
|
|
|
2007-12-06 20:03:27 +00:00
|
|
|
if (switch_is_file_path(file)) {
|
2006-02-14 16:26:24 +00:00
|
|
|
path = switch_core_strdup(loadable_modules.pool, file);
|
2008-05-27 04:30:03 +00:00
|
|
|
file = (char *) switch_cut_path(file);
|
2007-06-13 14:35:55 +00:00
|
|
|
if ((dot = strchr(file, '.'))) {
|
2008-10-13 06:42:50 +00:00
|
|
|
*dot = '\0';
|
2007-06-13 14:35:55 +00:00
|
|
|
}
|
2006-02-14 16:26:24 +00:00
|
|
|
} else {
|
2007-06-13 14:35:55 +00:00
|
|
|
if ((dot = strchr(file, '.'))) {
|
2008-10-13 06:42:50 +00:00
|
|
|
*dot = '\0';
|
2006-02-14 17:31:20 +00:00
|
|
|
}
|
2007-06-13 14:35:55 +00:00
|
|
|
len = strlen(dir);
|
|
|
|
len += strlen(file);
|
|
|
|
len += 8;
|
|
|
|
path = (char *) switch_core_alloc(loadable_modules.pool, len);
|
2007-12-12 21:53:32 +00:00
|
|
|
switch_snprintf(path, len, "%s%s%s%s", dir, SWITCH_PATH_SEPARATOR, file, ext);
|
2006-02-14 16:26:24 +00:00
|
|
|
}
|
2006-12-01 16:08:56 +00:00
|
|
|
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_mutex_lock(loadable_modules.mutex);
|
2007-03-29 22:31:56 +00:00
|
|
|
if (switch_core_hash_find(loadable_modules.module_hash, file)) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_WARNING, "Module %s Already Loaded!\n", file);
|
2007-06-13 14:48:13 +00:00
|
|
|
*err = "Module already loaded";
|
2007-05-03 16:28:23 +00:00
|
|
|
status = SWITCH_STATUS_FALSE;
|
2008-11-06 17:29:50 +00:00
|
|
|
} else if ((status = switch_loadable_module_load_file(path, file, global, &new_module)) == SWITCH_STATUS_SUCCESS) {
|
2007-06-13 14:35:55 +00:00
|
|
|
if ((status = switch_loadable_module_process(file, new_module)) == SWITCH_STATUS_SUCCESS && runtime) {
|
2007-03-17 19:51:08 +00:00
|
|
|
if (new_module->switch_module_runtime) {
|
2009-02-20 01:10:59 +00:00
|
|
|
new_module->thread = switch_core_launch_thread(switch_loadable_module_exec, new_module, new_module->pool);
|
2007-03-17 19:51:08 +00:00
|
|
|
}
|
2008-11-15 00:32:58 +00:00
|
|
|
} else if (status != SWITCH_STATUS_SUCCESS) {
|
|
|
|
*err = "module load routine returned an error";
|
2007-03-17 19:51:08 +00:00
|
|
|
}
|
2008-11-15 00:32:58 +00:00
|
|
|
} else {
|
|
|
|
*err = "module load file routine returned an error";
|
2006-02-14 16:26:24 +00:00
|
|
|
}
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_mutex_unlock(loadable_modules.mutex);
|
|
|
|
|
|
|
|
return status;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2008-07-10 14:41:31 +00:00
|
|
|
SWITCH_DECLARE(switch_status_t) switch_loadable_module_exists(const char *mod)
|
|
|
|
{
|
|
|
|
switch_status_t status;
|
|
|
|
|
|
|
|
if (switch_strlen_zero(mod)) {
|
|
|
|
return SWITCH_STATUS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch_mutex_lock(loadable_modules.mutex);
|
|
|
|
if (switch_core_hash_find(loadable_modules.module_hash, mod)) {
|
|
|
|
status = SWITCH_STATUS_SUCCESS;
|
|
|
|
} else {
|
|
|
|
status = SWITCH_STATUS_FALSE;
|
|
|
|
}
|
|
|
|
switch_mutex_unlock(loadable_modules.mutex);
|
2007-05-03 16:28:23 +00:00
|
|
|
|
2008-07-10 14:41:31 +00:00
|
|
|
return status;
|
|
|
|
}
|
2007-05-03 16:28:23 +00:00
|
|
|
|
2008-11-12 17:10:20 +00:00
|
|
|
SWITCH_DECLARE(switch_status_t) switch_loadable_module_unload_module(char *dir, char *fname, switch_bool_t force, const char **err)
|
2007-05-03 16:28:23 +00:00
|
|
|
{
|
|
|
|
switch_loadable_module_t *module = NULL;
|
|
|
|
switch_status_t status = SWITCH_STATUS_SUCCESS;
|
|
|
|
|
2008-11-12 17:10:20 +00:00
|
|
|
if (force) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_WARNING, "Spin the barrel and pull the trigger.......!\n");
|
|
|
|
}
|
|
|
|
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_mutex_lock(loadable_modules.mutex);
|
2007-06-13 14:35:55 +00:00
|
|
|
if ((module = switch_core_hash_find(loadable_modules.module_hash, fname))) {
|
2007-05-03 16:28:23 +00:00
|
|
|
if (module->perm) {
|
2007-09-29 01:06:08 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CRIT, "Module is not unloadable.\n");
|
2007-05-03 16:28:23 +00:00
|
|
|
*err = "Module is not unloadable";
|
|
|
|
status = SWITCH_STATUS_NOUNLOAD;
|
2008-07-21 17:33:46 +00:00
|
|
|
goto end;
|
2007-05-03 16:28:23 +00:00
|
|
|
} else {
|
2008-11-12 17:10:20 +00:00
|
|
|
if ((status = do_shutdown(module, SWITCH_TRUE, SWITCH_TRUE, !force, err) != SWITCH_STATUS_SUCCESS)) {
|
|
|
|
goto end;
|
|
|
|
}
|
2007-05-03 16:28:23 +00:00
|
|
|
}
|
2007-09-29 01:06:08 +00:00
|
|
|
switch_core_hash_delete(loadable_modules.module_hash, fname);
|
2007-05-03 16:28:23 +00:00
|
|
|
} else {
|
2007-05-24 15:30:31 +00:00
|
|
|
*err = "No such module!";
|
2007-05-03 16:28:23 +00:00
|
|
|
status = SWITCH_STATUS_FALSE;
|
|
|
|
}
|
2008-07-21 17:33:46 +00:00
|
|
|
end:
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_mutex_unlock(loadable_modules.mutex);
|
|
|
|
|
2008-11-12 17:10:20 +00:00
|
|
|
if (force) {
|
2008-11-12 19:28:05 +00:00
|
|
|
switch_yield(1000000);
|
2008-11-12 17:10:20 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_WARNING, "PHEW!\n");
|
|
|
|
}
|
|
|
|
|
2007-03-17 19:51:08 +00:00
|
|
|
return status;
|
2007-03-29 22:31:56 +00:00
|
|
|
|
2006-02-14 16:26:24 +00:00
|
|
|
}
|
|
|
|
|
2006-04-29 23:43:28 +00:00
|
|
|
SWITCH_DECLARE(switch_status_t) switch_loadable_module_build_dynamic(char *filename,
|
2007-03-29 22:31:56 +00:00
|
|
|
switch_module_load_t switch_module_load,
|
|
|
|
switch_module_runtime_t switch_module_runtime,
|
2008-05-27 04:30:03 +00:00
|
|
|
switch_module_shutdown_t switch_module_shutdown, switch_bool_t runtime)
|
2007-03-29 22:31:56 +00:00
|
|
|
{
|
|
|
|
switch_loadable_module_t *module = NULL;
|
|
|
|
switch_module_load_t load_func_ptr = NULL;
|
|
|
|
int loading = 1;
|
|
|
|
const char *err = NULL;
|
2007-06-13 20:40:06 +00:00
|
|
|
switch_loadable_module_interface_t *module_interface = NULL;
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_memory_pool_t *pool;
|
|
|
|
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2007-05-03 16:28:23 +00:00
|
|
|
if (switch_core_new_memory_pool(&pool) != SWITCH_STATUS_SUCCESS) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CRIT, "OH OH no pool\n");
|
|
|
|
abort();
|
|
|
|
}
|
2007-03-29 22:31:56 +00:00
|
|
|
|
2007-05-03 16:28:23 +00:00
|
|
|
if ((module = switch_core_alloc(pool, sizeof(switch_loadable_module_t))) == 0) {
|
2007-03-29 22:31:56 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CONSOLE, "Couldn't allocate memory\n");
|
2007-05-03 16:28:23 +00:00
|
|
|
abort();
|
2007-03-29 22:31:56 +00:00
|
|
|
}
|
|
|
|
|
2007-05-03 16:28:23 +00:00
|
|
|
|
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
while (loading) {
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_status_t status;
|
2007-05-04 00:14:12 +00:00
|
|
|
load_func_ptr = (switch_module_load_t) switch_module_load;
|
2007-03-29 22:31:56 +00:00
|
|
|
|
|
|
|
if (load_func_ptr == NULL) {
|
|
|
|
err = "Cannot Load";
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2007-06-13 20:40:06 +00:00
|
|
|
status = load_func_ptr(&module_interface, pool);
|
2007-05-03 16:28:23 +00:00
|
|
|
|
|
|
|
if (status != SWITCH_STATUS_SUCCESS && status != SWITCH_STATUS_NOUNLOAD) {
|
2007-03-29 22:31:56 +00:00
|
|
|
err = "Module load routine returned an error";
|
|
|
|
module_interface = NULL;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2007-05-03 16:28:23 +00:00
|
|
|
if ((module = switch_core_alloc(pool, sizeof(switch_loadable_module_t))) == 0) {
|
2007-03-29 22:31:56 +00:00
|
|
|
err = "Could not allocate memory\n";
|
2007-05-03 16:28:23 +00:00
|
|
|
abort();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (status == SWITCH_STATUS_NOUNLOAD) {
|
|
|
|
module->perm++;
|
2007-03-29 22:31:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
loading = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (err) {
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_core_destroy_memory_pool(&pool);
|
2007-03-29 22:31:56 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CONSOLE, "Error Loading module %s\n**%s**\n", filename, err);
|
|
|
|
return SWITCH_STATUS_GENERR;
|
|
|
|
}
|
|
|
|
|
2007-05-03 16:28:23 +00:00
|
|
|
module->pool = pool;
|
|
|
|
module->filename = switch_core_strdup(module->pool, filename);
|
2007-03-29 22:31:56 +00:00
|
|
|
module->module_interface = module_interface;
|
|
|
|
module->switch_module_load = load_func_ptr;
|
|
|
|
|
|
|
|
if (switch_module_shutdown) {
|
|
|
|
module->switch_module_shutdown = switch_module_shutdown;
|
|
|
|
}
|
|
|
|
if (switch_module_runtime) {
|
|
|
|
module->switch_module_runtime = switch_module_runtime;
|
|
|
|
}
|
2007-10-03 16:44:11 +00:00
|
|
|
if (runtime && module->switch_module_runtime) {
|
2009-02-20 01:10:59 +00:00
|
|
|
module->thread = switch_core_launch_thread(switch_loadable_module_exec, module, module->pool);
|
2007-03-29 22:31:56 +00:00
|
|
|
}
|
2007-03-30 00:13:31 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CONSOLE, "Successfully Loaded [%s]\n", module_interface->module_name);
|
2006-04-25 00:33:00 +00:00
|
|
|
return switch_loadable_module_process((char *) module->filename, module);
|
2007-03-29 22:31:56 +00:00
|
|
|
}
|
2006-04-25 00:33:00 +00:00
|
|
|
|
2006-03-06 03:23:34 +00:00
|
|
|
#ifdef WIN32
|
|
|
|
static void switch_loadable_module_path_init()
|
|
|
|
{
|
2007-03-29 22:31:56 +00:00
|
|
|
char *path = NULL, *working = NULL;
|
2006-03-06 03:23:34 +00:00
|
|
|
apr_dir_t *perl_dir_handle = NULL;
|
|
|
|
|
2006-03-06 20:35:23 +00:00
|
|
|
apr_env_get(&path, "path", loadable_modules.pool);
|
2007-03-29 22:31:56 +00:00
|
|
|
apr_filepath_get(&working, APR_FILEPATH_NATIVE, loadable_modules.pool);
|
2006-03-06 20:35:23 +00:00
|
|
|
|
|
|
|
if (apr_dir_open(&perl_dir_handle, ".\\perl", loadable_modules.pool) == APR_SUCCESS) {
|
2007-03-29 22:31:56 +00:00
|
|
|
apr_dir_close(perl_dir_handle);
|
2007-03-30 00:13:31 +00:00
|
|
|
apr_env_set("path", apr_pstrcat(loadable_modules.pool, path, ";", working, "\\perl", NULL), loadable_modules.pool);
|
2006-03-06 20:35:23 +00:00
|
|
|
}
|
2006-03-06 03:23:34 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2006-04-29 23:43:28 +00:00
|
|
|
SWITCH_DECLARE(switch_status_t) switch_loadable_module_init()
|
2005-11-19 20:07:43 +00:00
|
|
|
{
|
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
apr_finfo_t finfo = { 0 };
|
2006-02-13 18:34:39 +00:00
|
|
|
apr_dir_t *module_dir_handle = NULL;
|
2006-01-20 15:05:05 +00:00
|
|
|
apr_int32_t finfo_flags = APR_FINFO_DIRENT | APR_FINFO_TYPE | APR_FINFO_NAME;
|
2006-02-14 16:26:24 +00:00
|
|
|
char *cf = "modules.conf";
|
2006-05-10 03:23:05 +00:00
|
|
|
char *pcf = "post_load_modules.conf";
|
|
|
|
switch_xml_t cfg, xml;
|
2006-02-14 16:26:24 +00:00
|
|
|
unsigned char all = 0;
|
|
|
|
unsigned int count = 0;
|
2007-05-03 16:28:23 +00:00
|
|
|
const char *err;
|
|
|
|
|
2006-02-14 16:26:24 +00:00
|
|
|
|
2005-11-19 20:07:43 +00:00
|
|
|
#ifdef WIN32
|
|
|
|
const char *ext = ".dll";
|
|
|
|
const char *EXT = ".DLL";
|
2006-03-05 23:44:04 +00:00
|
|
|
#elif defined (MACOSX) || defined (DARWIN)
|
|
|
|
const char *ext = ".dylib";
|
|
|
|
const char *EXT = ".DYLIB";
|
2005-11-19 20:07:43 +00:00
|
|
|
#else
|
|
|
|
const char *ext = ".so";
|
|
|
|
const char *EXT = ".SO";
|
|
|
|
#endif
|
|
|
|
|
|
|
|
memset(&loadable_modules, 0, sizeof(loadable_modules));
|
|
|
|
switch_core_new_memory_pool(&loadable_modules.pool);
|
|
|
|
|
|
|
|
|
2006-03-06 03:23:34 +00:00
|
|
|
#ifdef WIN32
|
2006-03-06 20:35:23 +00:00
|
|
|
switch_loadable_module_path_init();
|
2006-03-06 03:23:34 +00:00
|
|
|
#endif
|
|
|
|
|
2005-11-19 20:07:43 +00:00
|
|
|
switch_core_hash_init(&loadable_modules.module_hash, loadable_modules.pool);
|
2008-11-05 00:20:30 +00:00
|
|
|
switch_core_hash_init_nocase(&loadable_modules.endpoint_hash, loadable_modules.pool);
|
|
|
|
switch_core_hash_init_nocase(&loadable_modules.codec_hash, loadable_modules.pool);
|
|
|
|
switch_core_hash_init_nocase(&loadable_modules.timer_hash, loadable_modules.pool);
|
|
|
|
switch_core_hash_init_nocase(&loadable_modules.application_hash, loadable_modules.pool);
|
|
|
|
switch_core_hash_init_nocase(&loadable_modules.api_hash, loadable_modules.pool);
|
2005-12-26 21:01:22 +00:00
|
|
|
switch_core_hash_init(&loadable_modules.file_hash, loadable_modules.pool);
|
2008-11-05 00:20:30 +00:00
|
|
|
switch_core_hash_init_nocase(&loadable_modules.speech_hash, loadable_modules.pool);
|
|
|
|
switch_core_hash_init_nocase(&loadable_modules.asr_hash, loadable_modules.pool);
|
|
|
|
switch_core_hash_init_nocase(&loadable_modules.directory_hash, loadable_modules.pool);
|
|
|
|
switch_core_hash_init_nocase(&loadable_modules.chat_hash, loadable_modules.pool);
|
|
|
|
switch_core_hash_init_nocase(&loadable_modules.say_hash, loadable_modules.pool);
|
|
|
|
switch_core_hash_init_nocase(&loadable_modules.management_hash, loadable_modules.pool);
|
|
|
|
switch_core_hash_init_nocase(&loadable_modules.dialplan_hash, loadable_modules.pool);
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_mutex_init(&loadable_modules.mutex, SWITCH_MUTEX_NESTED, loadable_modules.pool);
|
2005-11-19 20:07:43 +00:00
|
|
|
|
2008-02-20 17:11:37 +00:00
|
|
|
switch_loadable_module_load_module("", "CORE_SOFTTIMER_MODULE", SWITCH_FALSE, &err);
|
|
|
|
switch_loadable_module_load_module("", "CORE_PCM_MODULE", SWITCH_FALSE, &err);
|
2007-10-03 16:44:11 +00:00
|
|
|
|
2006-05-10 15:47:54 +00:00
|
|
|
if ((xml = switch_xml_open_cfg(cf, &cfg, NULL))) {
|
2006-05-10 03:23:05 +00:00
|
|
|
switch_xml_t mods, ld;
|
|
|
|
if ((mods = switch_xml_child(cfg, "modules"))) {
|
|
|
|
for (ld = switch_xml_child(mods, "load"); ld; ld = ld->next) {
|
2008-11-06 17:29:50 +00:00
|
|
|
switch_bool_t global = SWITCH_FALSE;
|
2006-05-17 00:58:21 +00:00
|
|
|
const char *val = switch_xml_attr_soft(ld, "module");
|
2008-11-06 17:29:50 +00:00
|
|
|
const char *sglobal = switch_xml_attr_soft(ld, "global");
|
2007-12-30 00:22:51 +00:00
|
|
|
if (switch_strlen_zero(val) || (strchr(val, '.') && !strstr(val, ext) && !strstr(val, EXT))) {
|
2006-05-10 03:23:05 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CONSOLE, "Invalid extension for %s\n", val);
|
|
|
|
continue;
|
2005-11-19 20:07:43 +00:00
|
|
|
}
|
2008-11-06 17:29:50 +00:00
|
|
|
global = switch_true(sglobal);
|
|
|
|
switch_loadable_module_load_module_ex((char *) SWITCH_GLOBAL_dirs.mod_dir, (char *) val, SWITCH_FALSE, global, &err);
|
2006-05-10 03:23:05 +00:00
|
|
|
count++;
|
2005-11-19 20:07:43 +00:00
|
|
|
}
|
2006-02-14 16:26:24 +00:00
|
|
|
}
|
2006-05-10 03:23:05 +00:00
|
|
|
switch_xml_free(xml);
|
2007-03-29 22:31:56 +00:00
|
|
|
|
2006-02-14 16:26:24 +00:00
|
|
|
} else {
|
2006-04-11 21:13:44 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CONSOLE, "open of %s failed\n", cf);
|
2006-02-14 16:26:24 +00:00
|
|
|
}
|
2005-11-19 20:07:43 +00:00
|
|
|
|
2006-05-10 15:47:54 +00:00
|
|
|
if ((xml = switch_xml_open_cfg(pcf, &cfg, NULL))) {
|
2006-05-10 03:23:05 +00:00
|
|
|
switch_xml_t mods, ld;
|
|
|
|
|
|
|
|
if ((mods = switch_xml_child(cfg, "modules"))) {
|
|
|
|
for (ld = switch_xml_child(mods, "load"); ld; ld = ld->next) {
|
2008-11-06 17:29:50 +00:00
|
|
|
switch_bool_t global = SWITCH_FALSE;
|
2006-05-17 00:58:21 +00:00
|
|
|
const char *val = switch_xml_attr_soft(ld, "module");
|
2008-11-06 17:29:50 +00:00
|
|
|
const char *sglobal = switch_xml_attr_soft(ld, "global");
|
2007-12-30 00:22:51 +00:00
|
|
|
if (switch_strlen_zero(val) || (strchr(val, '.') && !strstr(val, ext) && !strstr(val, EXT))) {
|
2006-05-10 03:23:05 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CONSOLE, "Invalid extension for %s\n", val);
|
|
|
|
continue;
|
|
|
|
}
|
2008-11-06 17:29:50 +00:00
|
|
|
global = switch_true(sglobal);
|
|
|
|
switch_loadable_module_load_module_ex((char *) SWITCH_GLOBAL_dirs.mod_dir, (char *) val, SWITCH_FALSE, global, &err);
|
2006-05-10 03:23:05 +00:00
|
|
|
count++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
switch_xml_free(xml);
|
2007-03-29 22:31:56 +00:00
|
|
|
|
2006-05-10 03:23:05 +00:00
|
|
|
} else {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CONSOLE, "open of %s failed\n", pcf);
|
2006-02-14 16:26:24 +00:00
|
|
|
}
|
2005-11-19 20:07:43 +00:00
|
|
|
|
2006-05-10 03:23:05 +00:00
|
|
|
if (!count) {
|
2008-01-07 11:21:24 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CONSOLE, "No modules loaded, assuming 'load all'\n");
|
2006-05-10 03:23:05 +00:00
|
|
|
all = 1;
|
|
|
|
}
|
2007-03-29 22:31:56 +00:00
|
|
|
|
2006-02-14 16:26:24 +00:00
|
|
|
if (all) {
|
2006-02-28 21:21:48 +00:00
|
|
|
if (apr_dir_open(&module_dir_handle, SWITCH_GLOBAL_dirs.mod_dir, loadable_modules.pool) != APR_SUCCESS) {
|
2007-03-30 00:13:31 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CONSOLE, "Can't open directory: %s\n", SWITCH_GLOBAL_dirs.mod_dir);
|
2006-02-14 16:26:24 +00:00
|
|
|
return SWITCH_STATUS_GENERR;
|
|
|
|
}
|
2005-11-19 20:07:43 +00:00
|
|
|
|
2006-02-14 16:26:24 +00:00
|
|
|
while (apr_dir_read(&finfo, finfo_flags, module_dir_handle) == APR_SUCCESS) {
|
|
|
|
const char *fname = finfo.fname;
|
2005-11-19 20:07:43 +00:00
|
|
|
|
2006-02-14 16:26:24 +00:00
|
|
|
if (finfo.filetype != APR_REG) {
|
|
|
|
continue;
|
2005-11-19 20:07:43 +00:00
|
|
|
}
|
|
|
|
|
2006-02-14 16:26:24 +00:00
|
|
|
if (!fname) {
|
|
|
|
fname = finfo.name;
|
2005-12-09 19:33:21 +00:00
|
|
|
}
|
|
|
|
|
2006-03-30 23:02:50 +00:00
|
|
|
if (!fname) {
|
2006-02-14 16:26:24 +00:00
|
|
|
continue;
|
2005-12-26 21:01:22 +00:00
|
|
|
}
|
|
|
|
|
2007-12-30 00:22:51 +00:00
|
|
|
if (switch_strlen_zero(fname) || (!strstr(fname, ext) && !strstr(fname, EXT))) {
|
2006-02-14 16:26:24 +00:00
|
|
|
continue;
|
2006-01-27 16:43:57 +00:00
|
|
|
}
|
2006-02-09 16:28:49 +00:00
|
|
|
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_loadable_module_load_module((char *) SWITCH_GLOBAL_dirs.mod_dir, (char *) fname, SWITCH_FALSE, &err);
|
2005-11-19 20:07:43 +00:00
|
|
|
}
|
2006-02-14 16:26:24 +00:00
|
|
|
apr_dir_close(module_dir_handle);
|
2005-11-19 20:07:43 +00:00
|
|
|
}
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2007-03-17 19:51:08 +00:00
|
|
|
switch_loadable_module_runtime();
|
2006-05-10 03:23:05 +00:00
|
|
|
|
2005-11-19 20:07:43 +00:00
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2008-11-12 17:10:20 +00:00
|
|
|
static switch_status_t do_shutdown(switch_loadable_module_t *module, switch_bool_t shutdown, switch_bool_t unload, switch_bool_t fail_if_busy, const char **err)
|
2007-05-03 16:28:23 +00:00
|
|
|
{
|
2008-08-22 19:00:56 +00:00
|
|
|
int32_t flags = switch_core_flags();
|
2007-12-11 19:23:57 +00:00
|
|
|
switch_assert(module != NULL);
|
2008-08-22 19:00:56 +00:00
|
|
|
|
2008-11-12 17:10:20 +00:00
|
|
|
if (fail_if_busy && module->module_interface->rwlock && switch_thread_rwlock_trywrlock(module->module_interface->rwlock) != SWITCH_STATUS_SUCCESS) {
|
|
|
|
if (err) {
|
|
|
|
*err = "Module in use.";
|
|
|
|
}
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_WARNING, "Module %s is in use, cannot unload.\n", module->module_interface->module_name);
|
|
|
|
return SWITCH_STATUS_FALSE;
|
|
|
|
}
|
|
|
|
|
2008-08-22 17:57:15 +00:00
|
|
|
if (shutdown) {
|
|
|
|
switch_loadable_module_unprocess(module);
|
|
|
|
if (module->switch_module_shutdown) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CONSOLE, "Stopping: %s\n", module->module_interface->module_name);
|
|
|
|
module->status = module->switch_module_shutdown();
|
2007-05-03 16:28:23 +00:00
|
|
|
} else {
|
2008-08-22 17:57:15 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CONSOLE, "%s has no shutdown routine\n", module->module_interface->module_name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-11-12 17:10:20 +00:00
|
|
|
if (fail_if_busy && module->module_interface->rwlock) {
|
|
|
|
switch_thread_rwlock_unlock(module->module_interface->rwlock);
|
|
|
|
}
|
|
|
|
|
2008-08-22 19:00:56 +00:00
|
|
|
if (unload && module->status != SWITCH_STATUS_NOUNLOAD && !(flags & SCF_VG)) {
|
2008-08-22 17:57:15 +00:00
|
|
|
switch_memory_pool_t *pool;
|
2009-02-20 01:10:59 +00:00
|
|
|
switch_status_t st;
|
|
|
|
|
|
|
|
if (module->thread) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CONSOLE, "%s stopping runtime thread.\n", module->module_interface->module_name);
|
|
|
|
switch_thread_join(&st, module->thread);
|
|
|
|
}
|
|
|
|
|
2008-08-22 17:57:15 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CONSOLE, "%s unloaded.\n", module->module_interface->module_name);
|
2008-11-06 17:29:50 +00:00
|
|
|
switch_dso_destroy(&module->lib);
|
2008-08-22 17:57:15 +00:00
|
|
|
if ((pool = module->pool)) {
|
|
|
|
module = NULL;
|
|
|
|
switch_core_destroy_memory_pool(&pool);
|
2007-05-03 16:28:23 +00:00
|
|
|
}
|
|
|
|
}
|
2008-08-22 17:57:15 +00:00
|
|
|
|
2008-11-12 17:10:20 +00:00
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
|
2007-05-03 16:28:23 +00:00
|
|
|
}
|
|
|
|
|
2006-01-05 21:03:22 +00:00
|
|
|
SWITCH_DECLARE(void) switch_loadable_module_shutdown(void)
|
2005-11-19 20:07:43 +00:00
|
|
|
{
|
2006-01-20 15:05:05 +00:00
|
|
|
switch_hash_index_t *hi;
|
2005-11-19 20:07:43 +00:00
|
|
|
void *val;
|
2006-04-29 06:05:03 +00:00
|
|
|
switch_loadable_module_t *module;
|
2009-03-05 02:46:00 +00:00
|
|
|
|
|
|
|
if (!loadable_modules.module_hash) {
|
|
|
|
return;
|
|
|
|
}
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2007-09-24 19:34:25 +00:00
|
|
|
for (hi = switch_hash_first(NULL, loadable_modules.module_hash); hi; hi = switch_hash_next(hi)) {
|
2005-12-14 22:46:09 +00:00
|
|
|
switch_hash_this(hi, NULL, NULL, &val);
|
2006-04-29 06:05:03 +00:00
|
|
|
module = (switch_loadable_module_t *) val;
|
2008-08-22 17:57:15 +00:00
|
|
|
if (!module->perm) {
|
2008-11-12 17:10:20 +00:00
|
|
|
do_shutdown(module, SWITCH_TRUE, SWITCH_FALSE, SWITCH_FALSE, NULL);
|
2008-08-22 17:57:15 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-08-24 01:52:27 +00:00
|
|
|
switch_yield(1000000);
|
|
|
|
|
2008-08-22 17:57:15 +00:00
|
|
|
for (hi = switch_hash_first(NULL, loadable_modules.module_hash); hi; hi = switch_hash_next(hi)) {
|
|
|
|
switch_hash_this(hi, NULL, NULL, &val);
|
|
|
|
module = (switch_loadable_module_t *) val;
|
|
|
|
if (!module->perm) {
|
2008-11-12 17:10:20 +00:00
|
|
|
do_shutdown(module, SWITCH_FALSE, SWITCH_TRUE, SWITCH_FALSE, NULL);
|
2008-08-22 17:57:15 +00:00
|
|
|
}
|
2005-11-19 20:07:43 +00:00
|
|
|
}
|
|
|
|
|
2007-09-29 01:06:08 +00:00
|
|
|
switch_core_hash_destroy(&loadable_modules.module_hash);
|
|
|
|
switch_core_hash_destroy(&loadable_modules.endpoint_hash);
|
|
|
|
switch_core_hash_destroy(&loadable_modules.codec_hash);
|
|
|
|
switch_core_hash_destroy(&loadable_modules.timer_hash);
|
|
|
|
switch_core_hash_destroy(&loadable_modules.application_hash);
|
|
|
|
switch_core_hash_destroy(&loadable_modules.api_hash);
|
|
|
|
switch_core_hash_destroy(&loadable_modules.file_hash);
|
|
|
|
switch_core_hash_destroy(&loadable_modules.speech_hash);
|
|
|
|
switch_core_hash_destroy(&loadable_modules.asr_hash);
|
|
|
|
switch_core_hash_destroy(&loadable_modules.directory_hash);
|
|
|
|
switch_core_hash_destroy(&loadable_modules.chat_hash);
|
|
|
|
switch_core_hash_destroy(&loadable_modules.say_hash);
|
|
|
|
switch_core_hash_destroy(&loadable_modules.management_hash);
|
|
|
|
switch_core_hash_destroy(&loadable_modules.dialplan_hash);
|
|
|
|
|
2005-11-19 20:07:43 +00:00
|
|
|
}
|
|
|
|
|
2007-05-12 14:48:14 +00:00
|
|
|
SWITCH_DECLARE(switch_endpoint_interface_t *) switch_loadable_module_get_endpoint_interface(const char *name)
|
2005-11-19 20:07:43 +00:00
|
|
|
{
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_endpoint_interface_t *ptr;
|
|
|
|
|
|
|
|
switch_mutex_lock(loadable_modules.mutex);
|
|
|
|
ptr = switch_core_hash_find(loadable_modules.endpoint_hash, name);
|
2008-11-12 19:28:05 +00:00
|
|
|
if (ptr) {
|
|
|
|
PROTECT_INTERFACE(ptr);
|
|
|
|
}
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_mutex_unlock(loadable_modules.mutex);
|
|
|
|
|
2008-11-12 19:28:05 +00:00
|
|
|
|
2007-05-03 16:28:23 +00:00
|
|
|
return ptr;
|
2005-11-19 20:07:43 +00:00
|
|
|
}
|
|
|
|
|
2007-05-12 14:48:14 +00:00
|
|
|
SWITCH_DECLARE(switch_codec_interface_t *) switch_loadable_module_get_codec_interface(const char *name)
|
2005-11-19 20:07:43 +00:00
|
|
|
{
|
2006-05-03 17:15:17 +00:00
|
|
|
char altname[256] = "";
|
|
|
|
switch_codec_interface_t *codec;
|
2006-05-04 16:01:38 +00:00
|
|
|
switch_size_t x;
|
2007-03-29 22:31:56 +00:00
|
|
|
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_mutex_lock(loadable_modules.mutex);
|
2006-05-03 17:15:17 +00:00
|
|
|
if (!(codec = switch_core_hash_find(loadable_modules.codec_hash, name))) {
|
2007-03-29 22:31:56 +00:00
|
|
|
for (x = 0; x < strlen(name); x++) {
|
|
|
|
altname[x] = (char) toupper((int) name[x]);
|
2006-05-03 17:15:17 +00:00
|
|
|
}
|
|
|
|
if (!(codec = switch_core_hash_find(loadable_modules.codec_hash, altname))) {
|
2007-03-29 22:31:56 +00:00
|
|
|
for (x = 0; x < strlen(name); x++) {
|
|
|
|
altname[x] = (char) tolower((int) name[x]);
|
2006-05-03 17:15:17 +00:00
|
|
|
}
|
|
|
|
codec = switch_core_hash_find(loadable_modules.codec_hash, altname);
|
|
|
|
}
|
2006-05-03 16:55:30 +00:00
|
|
|
}
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_mutex_unlock(loadable_modules.mutex);
|
2005-11-19 20:07:43 +00:00
|
|
|
|
2008-11-12 19:28:05 +00:00
|
|
|
if (codec) {
|
|
|
|
PROTECT_INTERFACE(codec);
|
|
|
|
}
|
2005-12-26 21:01:22 +00:00
|
|
|
|
2008-11-12 19:28:05 +00:00
|
|
|
return codec;
|
2006-01-27 16:43:57 +00:00
|
|
|
}
|
|
|
|
|
2008-11-12 19:28:05 +00:00
|
|
|
#define HASH_FUNC(_kind_) SWITCH_DECLARE(switch_##_kind_##_interface_t *) switch_loadable_module_get_##_kind_##_interface(const char *name) \
|
|
|
|
{ \
|
|
|
|
switch_##_kind_##_interface_t *i; \
|
|
|
|
if ((i = switch_core_hash_find_locked(loadable_modules._kind_##_hash, name, loadable_modules.mutex))) { \
|
|
|
|
PROTECT_INTERFACE(i); \
|
|
|
|
} \
|
|
|
|
return i; \
|
|
|
|
}
|
|
|
|
|
|
|
|
HASH_FUNC(dialplan)
|
|
|
|
HASH_FUNC(timer)
|
|
|
|
HASH_FUNC(application)
|
|
|
|
HASH_FUNC(api)
|
|
|
|
HASH_FUNC(file)
|
|
|
|
HASH_FUNC(speech)
|
|
|
|
HASH_FUNC(asr)
|
|
|
|
HASH_FUNC(directory)
|
2009-01-20 20:49:47 +00:00
|
|
|
HASH_FUNC(chat)
|
2006-11-09 05:39:04 +00:00
|
|
|
|
2006-02-09 16:28:49 +00:00
|
|
|
|
2007-05-12 14:48:14 +00:00
|
|
|
SWITCH_DECLARE(switch_say_interface_t *) switch_loadable_module_get_say_interface(const char *name)
|
2006-12-20 21:25:14 +00:00
|
|
|
{
|
2008-11-12 19:28:05 +00:00
|
|
|
return switch_core_hash_find_locked(loadable_modules.say_hash, name, loadable_modules.mutex);
|
2006-12-20 21:25:14 +00:00
|
|
|
}
|
|
|
|
|
2007-05-12 14:48:14 +00:00
|
|
|
SWITCH_DECLARE(switch_management_interface_t *) switch_loadable_module_get_management_interface(const char *relative_oid)
|
2007-03-07 18:34:22 +00:00
|
|
|
{
|
2007-05-03 16:28:23 +00:00
|
|
|
return switch_core_hash_find_locked(loadable_modules.management_hash, relative_oid, loadable_modules.mutex);
|
2007-03-07 18:34:22 +00:00
|
|
|
}
|
|
|
|
|
2007-09-24 19:34:25 +00:00
|
|
|
SWITCH_DECLARE(int) switch_loadable_module_get_codecs(const switch_codec_implementation_t **array, int arraylen)
|
2005-11-19 20:07:43 +00:00
|
|
|
{
|
2006-01-20 15:05:05 +00:00
|
|
|
switch_hash_index_t *hi;
|
2005-11-19 20:07:43 +00:00
|
|
|
void *val;
|
2006-07-10 22:08:02 +00:00
|
|
|
switch_codec_interface_t *codec_interface;
|
2005-11-19 20:07:43 +00:00
|
|
|
int i = 0;
|
2007-03-29 22:31:56 +00:00
|
|
|
const switch_codec_implementation_t *imp;
|
2005-11-19 20:07:43 +00:00
|
|
|
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_mutex_lock(loadable_modules.mutex);
|
2007-09-24 19:34:25 +00:00
|
|
|
for (hi = switch_hash_first(NULL, loadable_modules.codec_hash); hi; hi = switch_hash_next(hi)) {
|
2005-12-14 22:46:09 +00:00
|
|
|
switch_hash_this(hi, NULL, NULL, &val);
|
2006-07-10 22:08:02 +00:00
|
|
|
codec_interface = (switch_codec_interface_t *) val;
|
2007-03-29 22:31:56 +00:00
|
|
|
/* Look for a 20ms implementation because it's the safest choice */
|
|
|
|
for (imp = codec_interface->implementations; imp; imp = imp->next) {
|
2008-10-20 17:48:42 +00:00
|
|
|
if (imp->microseconds_per_packet / 1000 == 20) {
|
2007-03-29 22:31:56 +00:00
|
|
|
array[i++] = imp;
|
|
|
|
goto found;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/* oh well we will use what we have */
|
2006-07-10 22:08:02 +00:00
|
|
|
array[i++] = codec_interface->implementations;
|
2007-01-29 18:31:35 +00:00
|
|
|
|
2008-05-27 04:30:03 +00:00
|
|
|
found:
|
2007-01-29 18:31:35 +00:00
|
|
|
|
2005-11-19 20:07:43 +00:00
|
|
|
if (i > arraylen) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_mutex_unlock(loadable_modules.mutex);
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2005-11-19 20:07:43 +00:00
|
|
|
return i;
|
|
|
|
|
|
|
|
}
|
2006-02-22 22:36:52 +00:00
|
|
|
|
2007-03-30 00:13:31 +00:00
|
|
|
SWITCH_DECLARE(int) switch_loadable_module_get_codecs_sorted(const switch_codec_implementation_t **array, int arraylen, char **prefs, int preflen)
|
2005-12-06 17:18:56 +00:00
|
|
|
{
|
|
|
|
int x, i = 0;
|
2006-04-29 23:43:28 +00:00
|
|
|
switch_codec_interface_t *codec_interface;
|
2006-07-10 22:08:02 +00:00
|
|
|
const switch_codec_implementation_t *imp;
|
|
|
|
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_mutex_lock(loadable_modules.mutex);
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
for (x = 0; x < preflen; x++) {
|
2006-07-14 19:42:09 +00:00
|
|
|
char *cur, *last = NULL, *next = NULL, *name, *p, buf[256];
|
2008-12-16 16:40:38 +00:00
|
|
|
uint32_t interval = 0, rate = 0;
|
2006-07-14 19:42:09 +00:00
|
|
|
|
|
|
|
switch_copy_string(buf, prefs[x], sizeof(buf));
|
|
|
|
last = name = next = cur = buf;
|
|
|
|
|
|
|
|
for (;;) {
|
|
|
|
if (!next) {
|
|
|
|
break;
|
|
|
|
}
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2006-07-14 19:42:09 +00:00
|
|
|
if ((p = strchr(next, '@'))) {
|
|
|
|
*p++ = '\0';
|
|
|
|
}
|
|
|
|
next = p;
|
2007-06-28 14:47:58 +00:00
|
|
|
|
2006-07-14 19:42:09 +00:00
|
|
|
if (cur != name) {
|
|
|
|
if (strchr(cur, 'i')) {
|
|
|
|
interval = atoi(cur);
|
2007-06-28 14:47:58 +00:00
|
|
|
} else if ((strchr(cur, 'k') || strchr(cur, 'h'))) {
|
2006-07-14 19:42:09 +00:00
|
|
|
rate = atoi(cur);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
cur = next;
|
|
|
|
}
|
2006-07-10 22:08:02 +00:00
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
if ((codec_interface = switch_loadable_module_get_codec_interface(name)) != 0) {
|
|
|
|
/* If no specific codec interval is requested opt for 20ms above all else because lots of stuff assumes it */
|
2008-12-16 16:40:38 +00:00
|
|
|
for (imp = codec_interface->implementations; imp; imp = imp->next) {
|
|
|
|
uint8_t match = 1;
|
|
|
|
|
|
|
|
if (imp->codec_type != SWITCH_CODEC_TYPE_VIDEO) {
|
2007-01-29 15:43:41 +00:00
|
|
|
|
2008-12-16 16:40:38 +00:00
|
|
|
if ((!interval && (uint32_t) (imp->microseconds_per_packet / 1000) != 20) ||
|
|
|
|
(interval && (uint32_t) (imp->microseconds_per_packet / 1000) != interval)) {
|
|
|
|
match = 0;
|
2007-03-29 22:31:56 +00:00
|
|
|
}
|
2007-01-29 15:43:41 +00:00
|
|
|
|
2008-12-16 16:40:38 +00:00
|
|
|
if (match && ((!rate && (uint32_t) imp->samples_per_second != 8000) || (rate && (uint32_t) imp->samples_per_second != rate))) {
|
|
|
|
match = 0;
|
2007-03-29 22:31:56 +00:00
|
|
|
}
|
|
|
|
}
|
2008-12-16 16:40:38 +00:00
|
|
|
|
|
|
|
if (match) {
|
|
|
|
array[i++] = imp;
|
|
|
|
goto found;
|
|
|
|
}
|
2007-03-29 22:31:56 +00:00
|
|
|
}
|
2007-01-29 15:43:41 +00:00
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
/* Either looking for a specific interval or there was no interval specified and there wasn't one @20ms available */
|
2006-07-10 22:08:02 +00:00
|
|
|
for (imp = codec_interface->implementations; imp; imp = imp->next) {
|
2006-07-14 19:42:09 +00:00
|
|
|
uint8_t match = 1;
|
|
|
|
|
2008-05-06 00:02:37 +00:00
|
|
|
if (imp->codec_type != SWITCH_CODEC_TYPE_VIDEO) {
|
2008-12-16 16:40:38 +00:00
|
|
|
|
2008-10-20 17:48:42 +00:00
|
|
|
if (interval && (uint32_t) (imp->microseconds_per_packet / 1000) != interval) {
|
2008-05-06 00:02:37 +00:00
|
|
|
match = 0;
|
|
|
|
}
|
2008-12-16 16:40:38 +00:00
|
|
|
|
2008-05-06 00:02:37 +00:00
|
|
|
if (match && rate && (uint32_t) imp->samples_per_second != rate) {
|
|
|
|
match = 0;
|
|
|
|
}
|
2006-07-14 19:42:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (match) {
|
2006-07-10 22:08:02 +00:00
|
|
|
array[i++] = imp;
|
2007-03-29 22:31:56 +00:00
|
|
|
goto found;
|
2006-07-10 22:08:02 +00:00
|
|
|
}
|
2006-02-21 15:05:18 +00:00
|
|
|
}
|
2007-01-29 15:43:41 +00:00
|
|
|
|
2008-05-27 04:30:03 +00:00
|
|
|
found:
|
2007-01-29 15:43:41 +00:00
|
|
|
|
2008-11-13 23:07:03 +00:00
|
|
|
UNPROTECT_INTERFACE(codec_interface);
|
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
if (i > arraylen) {
|
|
|
|
break;
|
|
|
|
}
|
2008-11-13 23:07:03 +00:00
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
}
|
|
|
|
}
|
2005-12-06 17:18:56 +00:00
|
|
|
|
2007-05-03 16:28:23 +00:00
|
|
|
switch_mutex_unlock(loadable_modules.mutex);
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2007-05-03 16:28:23 +00:00
|
|
|
|
2005-12-06 17:18:56 +00:00
|
|
|
return i;
|
|
|
|
}
|
2005-12-23 02:55:25 +00:00
|
|
|
|
2007-05-12 14:48:14 +00:00
|
|
|
SWITCH_DECLARE(switch_status_t) switch_api_execute(const char *cmd, const char *arg, switch_core_session_t *session, switch_stream_handle_t *stream)
|
2005-12-23 02:55:25 +00:00
|
|
|
{
|
2006-04-29 23:43:28 +00:00
|
|
|
switch_api_interface_t *api;
|
|
|
|
switch_status_t status;
|
2005-12-23 02:55:25 +00:00
|
|
|
|
2007-12-11 19:23:57 +00:00
|
|
|
switch_assert(stream != NULL);
|
|
|
|
switch_assert(stream->data != NULL);
|
|
|
|
switch_assert(stream->write_function != NULL);
|
2006-05-10 15:47:54 +00:00
|
|
|
|
2008-05-24 03:46:19 +00:00
|
|
|
if (!stream->param_event) {
|
|
|
|
switch_event_create(&stream->param_event, SWITCH_EVENT_API);
|
2006-05-10 19:07:38 +00:00
|
|
|
}
|
|
|
|
|
2008-05-24 03:46:19 +00:00
|
|
|
if (stream->param_event) {
|
2006-05-10 19:07:38 +00:00
|
|
|
if (cmd) {
|
2008-08-16 02:17:09 +00:00
|
|
|
switch_event_add_header_string(stream->param_event, SWITCH_STACK_BOTTOM, "API-Command", cmd);
|
2006-05-10 19:07:38 +00:00
|
|
|
}
|
|
|
|
if (arg) {
|
2008-08-16 02:17:09 +00:00
|
|
|
switch_event_add_header_string(stream->param_event, SWITCH_STACK_BOTTOM, "API-Command-Argument", arg);
|
2006-05-10 19:07:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-05-24 05:09:50 +00:00
|
|
|
|
2008-05-16 16:15:37 +00:00
|
|
|
if (cmd && (api = switch_loadable_module_get_api_interface(cmd)) != 0) {
|
2008-05-24 05:09:50 +00:00
|
|
|
if ((status = api->function(arg, session, stream)) != SWITCH_STATUS_SUCCESS) {
|
|
|
|
stream->write_function(stream, "COMMAND RETURNED ERROR!\n");
|
|
|
|
}
|
2008-11-12 19:28:05 +00:00
|
|
|
UNPROTECT_INTERFACE(api);
|
2005-12-23 02:55:25 +00:00
|
|
|
} else {
|
2005-12-23 03:12:39 +00:00
|
|
|
status = SWITCH_STATUS_FALSE;
|
2007-04-29 03:21:56 +00:00
|
|
|
stream->write_function(stream, "INVALID COMMAND!\n");
|
2005-12-23 02:55:25 +00:00
|
|
|
}
|
|
|
|
|
2008-05-24 03:46:19 +00:00
|
|
|
if (stream->param_event) {
|
|
|
|
switch_event_fire(&stream->param_event);
|
2005-12-23 03:12:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return status;
|
2005-12-23 02:55:25 +00:00
|
|
|
}
|
2006-11-27 22:30:48 +00:00
|
|
|
|
2007-06-13 20:40:06 +00:00
|
|
|
|
|
|
|
SWITCH_DECLARE(switch_loadable_module_interface_t *) switch_loadable_module_create_module_interface(switch_memory_pool_t *pool, const char *name)
|
|
|
|
{
|
|
|
|
switch_loadable_module_interface_t *mod;
|
|
|
|
|
|
|
|
mod = switch_core_alloc(pool, sizeof(switch_loadable_module_interface_t));
|
2007-12-11 19:23:57 +00:00
|
|
|
switch_assert(mod != NULL);
|
2007-06-13 20:40:06 +00:00
|
|
|
|
|
|
|
mod->pool = pool;
|
|
|
|
|
|
|
|
mod->module_name = switch_core_strdup(mod->pool, name);
|
2008-11-12 17:10:20 +00:00
|
|
|
switch_thread_rwlock_create(&mod->rwlock, mod->pool);
|
2007-06-13 20:40:06 +00:00
|
|
|
return mod;
|
|
|
|
}
|
|
|
|
|
2007-11-16 05:47:00 +00:00
|
|
|
#define ALLOC_INTERFACE(_TYPE_) { \
|
2007-06-13 20:40:06 +00:00
|
|
|
switch_##_TYPE_##_interface_t *i, *ptr; \
|
|
|
|
i = switch_core_alloc(mod->pool, sizeof(switch_##_TYPE_##_interface_t)); \
|
2007-12-11 19:23:57 +00:00
|
|
|
switch_assert(i != NULL); \
|
2007-06-13 20:40:06 +00:00
|
|
|
for (ptr = mod->_TYPE_##_interface; ptr && ptr->next; ptr = ptr->next); \
|
|
|
|
if (ptr) { \
|
|
|
|
ptr->next = i; \
|
|
|
|
} else { \
|
|
|
|
mod->_TYPE_##_interface = i; \
|
|
|
|
} \
|
2008-07-01 23:41:09 +00:00
|
|
|
switch_thread_rwlock_create(&i->rwlock, mod->pool); \
|
2008-11-19 19:22:20 +00:00
|
|
|
switch_mutex_init(&i->reflock, SWITCH_MUTEX_NESTED, mod->pool); \
|
2008-11-12 17:10:20 +00:00
|
|
|
i->parent = mod; \
|
2007-06-13 20:49:45 +00:00
|
|
|
return i; }
|
2007-06-13 20:40:06 +00:00
|
|
|
|
|
|
|
|
|
|
|
SWITCH_DECLARE(void *) switch_loadable_module_create_interface(switch_loadable_module_interface_t *mod, switch_module_interface_name_t iname)
|
|
|
|
{
|
|
|
|
|
2008-05-27 04:30:03 +00:00
|
|
|
switch (iname) {
|
2007-06-13 20:40:06 +00:00
|
|
|
case SWITCH_ENDPOINT_INTERFACE:
|
2007-11-16 05:47:00 +00:00
|
|
|
ALLOC_INTERFACE(endpoint)
|
2007-06-13 20:40:06 +00:00
|
|
|
|
|
|
|
case SWITCH_TIMER_INTERFACE:
|
2007-11-16 05:47:00 +00:00
|
|
|
ALLOC_INTERFACE(timer)
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2007-06-13 20:40:06 +00:00
|
|
|
case SWITCH_DIALPLAN_INTERFACE:
|
2007-11-16 05:47:00 +00:00
|
|
|
ALLOC_INTERFACE(dialplan)
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2007-06-13 20:40:06 +00:00
|
|
|
case SWITCH_CODEC_INTERFACE:
|
2007-11-16 05:47:00 +00:00
|
|
|
ALLOC_INTERFACE(codec)
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2007-06-13 20:40:06 +00:00
|
|
|
case SWITCH_APPLICATION_INTERFACE:
|
2007-11-16 05:47:00 +00:00
|
|
|
ALLOC_INTERFACE(application)
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2007-06-13 20:40:06 +00:00
|
|
|
case SWITCH_API_INTERFACE:
|
2007-11-16 05:47:00 +00:00
|
|
|
ALLOC_INTERFACE(api)
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2007-06-13 20:40:06 +00:00
|
|
|
case SWITCH_FILE_INTERFACE:
|
2007-11-16 05:47:00 +00:00
|
|
|
ALLOC_INTERFACE(file)
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2007-06-13 20:40:06 +00:00
|
|
|
case SWITCH_SPEECH_INTERFACE:
|
2007-11-16 05:47:00 +00:00
|
|
|
ALLOC_INTERFACE(speech)
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2007-06-13 20:40:06 +00:00
|
|
|
case SWITCH_DIRECTORY_INTERFACE:
|
2007-11-16 05:47:00 +00:00
|
|
|
ALLOC_INTERFACE(directory)
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2007-06-13 20:40:06 +00:00
|
|
|
case SWITCH_CHAT_INTERFACE:
|
2007-11-16 05:47:00 +00:00
|
|
|
ALLOC_INTERFACE(chat)
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2007-06-13 20:40:06 +00:00
|
|
|
case SWITCH_SAY_INTERFACE:
|
2007-11-16 05:47:00 +00:00
|
|
|
ALLOC_INTERFACE(say)
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2007-06-13 20:40:06 +00:00
|
|
|
case SWITCH_ASR_INTERFACE:
|
2007-11-16 05:47:00 +00:00
|
|
|
ALLOC_INTERFACE(asr)
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2007-06-13 20:40:06 +00:00
|
|
|
case SWITCH_MANAGEMENT_INTERFACE:
|
2007-11-16 05:47:00 +00:00
|
|
|
ALLOC_INTERFACE(management)
|
2008-05-27 04:30:03 +00:00
|
|
|
|
2007-06-13 20:40:06 +00:00
|
|
|
default:
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_WARNING, "Invalid Module Type!\n");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2006-11-27 22:30:48 +00:00
|
|
|
/* For Emacs:
|
|
|
|
* Local Variables:
|
|
|
|
* mode:c
|
2008-02-03 22:14:57 +00:00
|
|
|
* indent-tabs-mode:t
|
2006-11-27 22:30:48 +00:00
|
|
|
* tab-width:4
|
|
|
|
* c-basic-offset:4
|
|
|
|
* End:
|
|
|
|
* For VIM:
|
2008-07-03 19:12:26 +00:00
|
|
|
* vim:set softtabstop=4 shiftwidth=4 tabstop=4:
|
2006-11-27 22:30:48 +00:00
|
|
|
*/
|