2005-11-19 20:07:43 +00:00
|
|
|
/*
|
|
|
|
* FreeSWITCH Modular Media Switching Software Library / Soft-Switch Application
|
|
|
|
* Copyright (C) 2005/2006, Anthony Minessale II <anthmct@yahoo.com>
|
|
|
|
*
|
|
|
|
* 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
|
|
|
|
* Anthony Minessale II <anthmct@yahoo.com>
|
|
|
|
* Portions created by the Initial Developer are Copyright (C)
|
|
|
|
* the Initial Developer. All Rights Reserved.
|
|
|
|
*
|
|
|
|
* Contributor(s):
|
|
|
|
*
|
|
|
|
* Anthony Minessale II <anthmct@yahoo.com>
|
2006-09-18 05:08:55 +00:00
|
|
|
* Michael Jerris <mike@jerris.com>
|
|
|
|
* Paul D. Tinsley <pdt at jackhammer.org>
|
2005-11-19 20:07:43 +00:00
|
|
|
*
|
|
|
|
*
|
|
|
|
* switch_core.c -- Main Core Library
|
|
|
|
*
|
|
|
|
*/
|
2006-04-12 15:25:26 +00:00
|
|
|
|
2005-11-19 20:07:43 +00:00
|
|
|
#include <switch.h>
|
2006-08-19 18:51:22 +00:00
|
|
|
#include <switch_version.h>
|
2007-03-29 22:31:56 +00:00
|
|
|
#include "private/switch_core.h"
|
2006-02-28 21:21:48 +00:00
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
SWITCH_DECLARE_DATA switch_directories SWITCH_GLOBAL_dirs = { 0 };
|
2007-03-28 23:37:12 +00:00
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
/* The main runtime obj we keep this hidden for ourselves */
|
|
|
|
static struct {
|
2006-05-10 03:23:05 +00:00
|
|
|
switch_time_t initiated;
|
Ringback (sponsored by Front Logic)
This addition lets you set artifical ringback on a channel
that is waiting for an originated call to be answered.
the syntax is
<action application="set" data="ringback=[data]"/>
where data is either the full path to an audio file
or a teletone generation script..
syntax of teletone scripts
LEGEND:
0-9,a-d,*,# (standard dtmf tones)
variables: c,r,d,v,>,<,+,w,l,L,%
c (channels) - Sets the number of channels.
r (rate) - Sets the sample rate.
d (duration) - Sets the default tone duration.
v (volume) - Sets the default volume.
> (decrease vol) - factor to decrease volume by per frame (0 for even decrease across duration).
< (increase vol) - factor to increase volume by per frame (0 for even increase across duration).
+ (step) - factor to step by used by < and >.
w (wait) - default silence after each tone.
l (loops) - number of times to repeat each tone in the script.
L (LOOPS) - number of times to repeat the the whole script.
% (manual tone) - a generic tone specified by a duration, a wait and a list of frequencies.
standard tones can have custom duration per use with the () modifier
7(1000, 500) to generate DTMF 7 for 1 second then pause .5 seconds
EXAMPLES
UK Ring Tone [400+450 hz on for 400ms off for 200ms then 400+450 hz on for 400ms off for 2200ms]
%(400,200,400,450);%(400,2200,400,450)
US Ring Tone [440+480 hz on for 2000ms off for 4000ms]
%(2000,4000,440,480)
ATT BONG [volume level 4000, even decay, step by 2, # key for 60ms with no wait, volume level 2000, 350+440hz {us dialtone} for 940ms
v=4000;>=0;+=2;#(60,0);v=2000;%(940,0,350,440)
SIT Tone 913.8 hz for 274 ms with no wait, 1370.6 hz for 274 ms with no wait, 1776.7 hz for 380ms with no wait
%(274,0,913.8);%(274,0,1370.6);%(380,0,1776.7)
ATTN TONE (phone's off the hook!) 1400+2060+2450+2600 hz for 100ms with 100ms wait
%(100,100,1400,2060,2450,2600)
git-svn-id: http://svn.freeswitch.org/svn/freeswitch/trunk@3408 d0543943-73ff-0310-b7d9-9358b9ac24b2
2006-11-19 01:05:06 +00:00
|
|
|
switch_hash_t *global_vars;
|
2007-03-29 22:31:56 +00:00
|
|
|
switch_memory_pool_t *memory_pool;
|
2006-05-05 13:35:33 +00:00
|
|
|
#ifdef CRASH_PROT
|
2006-04-29 01:00:52 +00:00
|
|
|
switch_hash_t *stack_table;
|
2006-05-05 13:35:33 +00:00
|
|
|
#endif
|
2006-04-29 06:05:03 +00:00
|
|
|
const switch_state_handler_table_t *state_handlers[SWITCH_MAX_STATE_HANDLERS];
|
2007-03-29 22:31:56 +00:00
|
|
|
int state_handler_index;
|
2005-11-19 20:07:43 +00:00
|
|
|
FILE *console;
|
2006-09-20 20:25:26 +00:00
|
|
|
uint8_t running;
|
2007-02-22 17:38:34 +00:00
|
|
|
char uuid_str[SWITCH_UUID_FORMATTED_LENGTH + 1];
|
2007-03-29 22:31:56 +00:00
|
|
|
uint32_t no_new_sessions;
|
|
|
|
uint32_t shutting_down;
|
|
|
|
} runtime;
|
2005-12-21 22:25:22 +00:00
|
|
|
|
|
|
|
|
2007-03-28 23:37:12 +00:00
|
|
|
static void send_heartbeat(void)
|
|
|
|
{
|
|
|
|
switch_event_t *event;
|
|
|
|
switch_core_time_duration_t duration;
|
|
|
|
|
|
|
|
switch_core_measure_time(switch_core_uptime(), &duration);
|
2007-03-29 22:31:56 +00:00
|
|
|
|
2007-03-28 23:37:12 +00:00
|
|
|
if (switch_event_create(&event, SWITCH_EVENT_HEARTBEAT) == SWITCH_STATUS_SUCCESS) {
|
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "Event-Info", "System Ready");
|
2007-03-29 22:31:56 +00:00
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "Up-Time",
|
2007-03-28 23:37:12 +00:00
|
|
|
"%u year%s, "
|
|
|
|
"%u day%s, "
|
|
|
|
"%u hour%s, "
|
|
|
|
"%u minute%s, "
|
|
|
|
"%u second%s, "
|
|
|
|
"%u millisecond%s, "
|
|
|
|
"%u microsecond%s\n",
|
|
|
|
duration.yr, duration.yr == 1 ? "" : "s",
|
|
|
|
duration.day, duration.day == 1 ? "" : "s",
|
2007-03-29 22:31:56 +00:00
|
|
|
duration.hr, duration.hr == 1 ? "" : "s",
|
|
|
|
duration.min, duration.min == 1 ? "" : "s",
|
|
|
|
duration.sec, duration.sec == 1 ? "" : "s",
|
|
|
|
duration.ms, duration.ms == 1 ? "" : "s", duration.mms, duration.mms == 1 ? "" : "s");
|
2005-11-19 20:07:43 +00:00
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "Session-Count", "%u", switch_core_session_count());
|
|
|
|
switch_event_fire(&event);
|
2005-11-19 20:07:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
static void heartbeat_callback(switch_scheduler_task_t *task)
|
2005-11-19 20:07:43 +00:00
|
|
|
{
|
2007-03-29 22:31:56 +00:00
|
|
|
send_heartbeat();
|
2005-11-19 20:07:43 +00:00
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
/* reschedule this task */
|
|
|
|
task->runtime += 20;
|
2005-11-19 20:07:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
SWITCH_DECLARE(switch_status_t) switch_core_set_console(char *console)
|
2005-11-19 20:07:43 +00:00
|
|
|
{
|
2007-03-29 22:31:56 +00:00
|
|
|
if ((runtime.console = fopen(console, "a")) == 0) {
|
|
|
|
fprintf(stderr, "Cannot open output file %s.\n", console);
|
|
|
|
return SWITCH_STATUS_FALSE;
|
|
|
|
}
|
2006-03-27 17:42:59 +00:00
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
2005-11-19 20:07:43 +00:00
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
SWITCH_DECLARE(FILE *) switch_core_get_console(void)
|
|
|
|
{
|
|
|
|
return runtime.console;
|
2005-11-19 20:07:43 +00:00
|
|
|
}
|
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
SWITCH_DECLARE(FILE *) switch_core_data_channel(switch_text_channel_t channel)
|
2005-11-19 20:07:43 +00:00
|
|
|
{
|
2007-03-29 22:31:56 +00:00
|
|
|
FILE *handle = stdout;
|
2005-11-19 20:07:43 +00:00
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
switch (channel) {
|
|
|
|
case SWITCH_CHANNEL_ID_LOG:
|
|
|
|
case SWITCH_CHANNEL_ID_LOG_CLEAN:
|
|
|
|
handle = runtime.console;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
handle = runtime.console;
|
|
|
|
break;
|
|
|
|
}
|
2005-11-19 20:07:43 +00:00
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
return handle;
|
|
|
|
}
|
2006-07-07 18:59:14 +00:00
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
SWITCH_DECLARE(int) switch_core_add_state_handler(const switch_state_handler_table_t *state_handler)
|
|
|
|
{
|
|
|
|
int index = runtime.state_handler_index++;
|
2006-07-07 18:59:14 +00:00
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
if (runtime.state_handler_index >= SWITCH_MAX_STATE_HANDLERS) {
|
|
|
|
return -1;
|
2006-09-20 20:25:26 +00:00
|
|
|
}
|
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
runtime.state_handlers[index] = state_handler;
|
|
|
|
return index;
|
|
|
|
}
|
2005-11-19 20:07:43 +00:00
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
SWITCH_DECLARE(const switch_state_handler_table_t *) switch_core_get_state_handler(int index)
|
|
|
|
{
|
2005-11-19 20:07:43 +00:00
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
if (index > SWITCH_MAX_STATE_HANDLERS || index > runtime.state_handler_index) {
|
2005-11-19 20:07:43 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
return runtime.state_handlers[index];
|
2005-11-19 20:07:43 +00:00
|
|
|
}
|
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
|
|
|
|
SWITCH_DECLARE(char *) switch_core_get_variable(char *varname)
|
2006-09-18 22:22:25 +00:00
|
|
|
{
|
2007-03-29 22:31:56 +00:00
|
|
|
return (char *) switch_core_hash_find(runtime.global_vars, varname);
|
2006-09-18 22:22:25 +00:00
|
|
|
}
|
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
SWITCH_DECLARE(void) switch_core_set_variable(char *varname, char *value)
|
2005-11-19 20:07:43 +00:00
|
|
|
{
|
2007-03-30 00:13:31 +00:00
|
|
|
switch_core_hash_insert(runtime.global_vars, switch_core_strdup(runtime.memory_pool, varname), switch_core_strdup(runtime.memory_pool, value));
|
2005-11-19 20:07:43 +00:00
|
|
|
}
|
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
SWITCH_DECLARE(char *) switch_core_get_uuid(void)
|
2007-02-23 16:42:40 +00:00
|
|
|
{
|
2007-03-29 22:31:56 +00:00
|
|
|
return runtime.uuid_str;
|
|
|
|
}
|
2007-02-23 16:42:40 +00:00
|
|
|
|
|
|
|
|
2007-03-30 00:13:31 +00:00
|
|
|
static void *switch_core_service_thread(switch_thread_t * thread, void *obj)
|
2007-03-29 22:31:56 +00:00
|
|
|
{
|
|
|
|
switch_core_thread_session_t *data = obj;
|
|
|
|
switch_core_session_t *session = data->objs[0];
|
|
|
|
int *stream_id_p = data->objs[1];
|
|
|
|
switch_channel_t *channel;
|
|
|
|
switch_frame_t *read_frame;
|
|
|
|
int stream_id = *stream_id_p;
|
2007-02-23 16:42:40 +00:00
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
assert(thread != NULL);
|
|
|
|
assert(session != NULL);
|
|
|
|
channel = switch_core_session_get_channel(session);
|
|
|
|
assert(channel != NULL);
|
2007-02-23 16:42:40 +00:00
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
switch_channel_set_flag(channel, CF_SERVICE);
|
|
|
|
while (data->running > 0) {
|
|
|
|
switch (switch_core_session_read_frame(session, &read_frame, -1, stream_id)) {
|
|
|
|
case SWITCH_STATUS_SUCCESS:
|
|
|
|
case SWITCH_STATUS_TIMEOUT:
|
|
|
|
case SWITCH_STATUS_BREAK:
|
2007-02-23 16:42:40 +00:00
|
|
|
break;
|
2007-03-29 22:31:56 +00:00
|
|
|
default:
|
|
|
|
data->running = -1;
|
|
|
|
continue;
|
2007-02-23 16:42:40 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
switch_channel_clear_flag(channel, CF_SERVICE);
|
|
|
|
data->running = 0;
|
|
|
|
return NULL;
|
2007-02-23 16:42:40 +00:00
|
|
|
}
|
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
/* Either add a timeout here or make damn sure the thread cannot get hung somehow (my preference) */
|
|
|
|
SWITCH_DECLARE(void) switch_core_thread_session_end(switch_core_thread_session_t *thread_session)
|
2005-12-21 22:25:22 +00:00
|
|
|
{
|
2007-03-29 22:31:56 +00:00
|
|
|
if (thread_session->running > 0) {
|
|
|
|
thread_session->running = -1;
|
2006-04-26 17:18:33 +00:00
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
while (thread_session->running) {
|
|
|
|
switch_yield(1000);
|
2006-04-26 17:18:33 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2006-04-04 21:26:21 +00:00
|
|
|
|
2007-03-30 00:13:31 +00:00
|
|
|
SWITCH_DECLARE(void) switch_core_service_session(switch_core_session_t *session, switch_core_thread_session_t *thread_session, int stream_id)
|
2006-04-26 17:18:33 +00:00
|
|
|
{
|
2007-03-29 22:31:56 +00:00
|
|
|
thread_session->running = 1;
|
|
|
|
thread_session->objs[0] = session;
|
|
|
|
thread_session->objs[1] = &stream_id;
|
|
|
|
switch_core_session_launch_thread(session, switch_core_service_thread, thread_session);
|
|
|
|
}
|
2006-09-18 22:22:25 +00:00
|
|
|
|
2007-02-22 17:38:34 +00:00
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
/* This function abstracts the thread creation for modules by allowing you to pass a function ptr and
|
|
|
|
a void object and trust that that the function will be run in a thread with arg This lets
|
|
|
|
you request and activate a thread without giving up any knowledge about what is in the thread
|
|
|
|
neither the core nor the calling module know anything about each other.
|
2006-09-18 22:22:25 +00:00
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
This thread is expected to never exit until the application exits so the func is responsible
|
|
|
|
to make sure that is the case.
|
2006-09-18 22:22:25 +00:00
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
The typical use for this is so switch_loadable_module.c can start up a thread for each module
|
|
|
|
passing the table of module methods as a session obj into the core without actually allowing
|
|
|
|
the core to have any clue and keeping switch_loadable_module.c from needing any thread code.
|
2006-04-26 17:18:33 +00:00
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
*/
|
2006-04-26 17:18:33 +00:00
|
|
|
|
2007-03-30 00:13:31 +00:00
|
|
|
SWITCH_DECLARE(void) switch_core_launch_thread(switch_thread_start_t func, void *obj, switch_memory_pool_t * pool)
|
2006-04-26 17:18:33 +00:00
|
|
|
{
|
2006-04-29 01:00:52 +00:00
|
|
|
switch_thread_t *thread;
|
2007-03-29 22:31:56 +00:00
|
|
|
switch_threadattr_t *thd_attr = NULL;
|
|
|
|
switch_core_thread_session_t *ts;
|
|
|
|
int mypool;
|
2006-04-26 17:18:33 +00:00
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
mypool = pool ? 0 : 1;
|
2006-04-26 17:18:33 +00:00
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
if (!pool && switch_core_new_memory_pool(&pool) != SWITCH_STATUS_SUCCESS) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CRIT, "Could not allocate memory pool\n");
|
|
|
|
return;
|
|
|
|
}
|
2006-04-26 17:18:33 +00:00
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
switch_threadattr_create(&thd_attr, pool);
|
|
|
|
switch_threadattr_detach_set(thd_attr, 1);
|
2006-04-04 21:26:21 +00:00
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
if ((ts = switch_core_alloc(pool, sizeof(*ts))) == 0) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CRIT, "Could not allocate memory\n");
|
|
|
|
} else {
|
|
|
|
if (mypool) {
|
|
|
|
ts->pool = pool;
|
2007-02-23 20:13:15 +00:00
|
|
|
}
|
2007-03-29 22:31:56 +00:00
|
|
|
ts->objs[0] = obj;
|
|
|
|
switch_threadattr_stacksize_set(thd_attr, SWITCH_THREAD_STACKSIZE);
|
|
|
|
switch_thread_create(&thread, thd_attr, func, ts, pool);
|
2005-12-21 22:25:22 +00:00
|
|
|
}
|
2006-04-04 21:26:21 +00:00
|
|
|
|
2005-12-21 22:25:22 +00:00
|
|
|
}
|
2007-03-11 01:07:47 +00:00
|
|
|
|
2006-03-01 17:06:10 +00:00
|
|
|
SWITCH_DECLARE(void) switch_core_set_globals(void)
|
2005-11-19 20:07:43 +00:00
|
|
|
{
|
2006-08-20 03:04:55 +00:00
|
|
|
#define BUFSIZE 1024
|
2007-03-11 01:07:47 +00:00
|
|
|
#ifdef WIN32
|
2007-03-29 22:31:56 +00:00
|
|
|
char lpPathBuffer[BUFSIZE];
|
|
|
|
DWORD dwBufSize = BUFSIZE;
|
2007-03-11 01:07:47 +00:00
|
|
|
char base_dir[1024];
|
2006-08-20 03:04:55 +00:00
|
|
|
char *lastbacklash;
|
2007-03-29 22:31:56 +00:00
|
|
|
GetModuleFileName(NULL, base_dir, BUFSIZE);
|
|
|
|
lastbacklash = strrchr(base_dir, '\\');
|
2007-03-11 01:07:47 +00:00
|
|
|
base_dir[(lastbacklash - base_dir)] = '\0';
|
|
|
|
#else
|
|
|
|
char base_dir[1024] = SWITCH_PREFIX_DIR;
|
|
|
|
#endif
|
|
|
|
|
2006-08-20 03:36:14 +00:00
|
|
|
if (!SWITCH_GLOBAL_dirs.base_dir && (SWITCH_GLOBAL_dirs.base_dir = (char *) malloc(BUFSIZE))) {
|
2007-03-11 01:07:47 +00:00
|
|
|
switch_snprintf(SWITCH_GLOBAL_dirs.base_dir, BUFSIZE, "%s", base_dir);
|
2006-08-20 03:04:55 +00:00
|
|
|
}
|
2007-03-11 01:07:47 +00:00
|
|
|
|
2006-08-20 03:36:14 +00:00
|
|
|
if (!SWITCH_GLOBAL_dirs.mod_dir && (SWITCH_GLOBAL_dirs.mod_dir = (char *) malloc(BUFSIZE))) {
|
2007-03-11 01:07:47 +00:00
|
|
|
#ifdef SWITCH_MOD_DIR
|
|
|
|
switch_snprintf(SWITCH_GLOBAL_dirs.mod_dir, BUFSIZE, "%s", SWITCH_MOD_DIR);
|
|
|
|
#else
|
|
|
|
switch_snprintf(SWITCH_GLOBAL_dirs.mod_dir, BUFSIZE, "%s%smod", base_dir, SWITCH_PATH_SEPARATOR);
|
|
|
|
#endif
|
2006-08-20 03:04:55 +00:00
|
|
|
}
|
2007-03-11 01:07:47 +00:00
|
|
|
|
2006-08-20 03:36:14 +00:00
|
|
|
if (!SWITCH_GLOBAL_dirs.conf_dir && (SWITCH_GLOBAL_dirs.conf_dir = (char *) malloc(BUFSIZE))) {
|
2007-03-11 01:07:47 +00:00
|
|
|
#ifdef SWITCH_CONF_DIR
|
|
|
|
switch_snprintf(SWITCH_GLOBAL_dirs.conf_dir, BUFSIZE, "%s", SWITCH_CONF_DIR);
|
|
|
|
#else
|
|
|
|
switch_snprintf(SWITCH_GLOBAL_dirs.conf_dir, BUFSIZE, "%s%sconf", base_dir, SWITCH_PATH_SEPARATOR);
|
|
|
|
#endif
|
2006-08-20 03:04:55 +00:00
|
|
|
}
|
2007-03-11 01:07:47 +00:00
|
|
|
|
2006-08-20 03:36:14 +00:00
|
|
|
if (!SWITCH_GLOBAL_dirs.log_dir && (SWITCH_GLOBAL_dirs.log_dir = (char *) malloc(BUFSIZE))) {
|
2007-03-11 01:07:47 +00:00
|
|
|
#ifdef SWITCH_LOG_DIR
|
|
|
|
switch_snprintf(SWITCH_GLOBAL_dirs.log_dir, BUFSIZE, "%s", SWITCH_LOG_DIR);
|
|
|
|
#else
|
|
|
|
switch_snprintf(SWITCH_GLOBAL_dirs.log_dir, BUFSIZE, "%s%slog", base_dir, SWITCH_PATH_SEPARATOR);
|
|
|
|
#endif
|
2006-08-20 03:04:55 +00:00
|
|
|
}
|
2007-03-11 01:07:47 +00:00
|
|
|
|
2006-08-20 03:36:14 +00:00
|
|
|
if (!SWITCH_GLOBAL_dirs.db_dir && (SWITCH_GLOBAL_dirs.db_dir = (char *) malloc(BUFSIZE))) {
|
2007-03-11 01:07:47 +00:00
|
|
|
#ifdef SWITCH_DB_DIR
|
|
|
|
switch_snprintf(SWITCH_GLOBAL_dirs.db_dir, BUFSIZE, "%s", SWITCH_DB_DIR);
|
|
|
|
#else
|
|
|
|
switch_snprintf(SWITCH_GLOBAL_dirs.db_dir, BUFSIZE, "%s%sdb", base_dir, SWITCH_PATH_SEPARATOR);
|
|
|
|
#endif
|
2006-08-20 03:04:55 +00:00
|
|
|
}
|
2007-03-11 01:07:47 +00:00
|
|
|
|
2006-08-20 03:36:14 +00:00
|
|
|
if (!SWITCH_GLOBAL_dirs.script_dir && (SWITCH_GLOBAL_dirs.script_dir = (char *) malloc(BUFSIZE))) {
|
2007-03-11 01:07:47 +00:00
|
|
|
#ifdef SWITCH_SCRIPT_DIR
|
|
|
|
switch_snprintf(SWITCH_GLOBAL_dirs.script_dir, BUFSIZE, "%s", SWITCH_SCRIPT_DIR);
|
|
|
|
#else
|
|
|
|
switch_snprintf(SWITCH_GLOBAL_dirs.script_dir, BUFSIZE, "%s%sscripts", base_dir, SWITCH_PATH_SEPARATOR);
|
|
|
|
#endif
|
2006-08-20 03:04:55 +00:00
|
|
|
}
|
2007-03-11 01:07:47 +00:00
|
|
|
|
2006-08-20 03:36:14 +00:00
|
|
|
if (!SWITCH_GLOBAL_dirs.htdocs_dir && (SWITCH_GLOBAL_dirs.htdocs_dir = (char *) malloc(BUFSIZE))) {
|
2007-03-11 01:07:47 +00:00
|
|
|
#ifdef SWITCH_HTDOCS_DIR
|
|
|
|
switch_snprintf(SWITCH_GLOBAL_dirs.htdocs_dir, BUFSIZE, "%s", SWITCH_HTDOCS_DIR);
|
|
|
|
#else
|
|
|
|
switch_snprintf(SWITCH_GLOBAL_dirs.htdocs_dir, BUFSIZE, "%s%shtdocs", base_dir, SWITCH_PATH_SEPARATOR);
|
|
|
|
#endif
|
2006-11-09 05:39:04 +00:00
|
|
|
}
|
2007-03-11 01:07:47 +00:00
|
|
|
|
|
|
|
if (!SWITCH_GLOBAL_dirs.grammar_dir && (SWITCH_GLOBAL_dirs.grammar_dir = (char *) malloc(BUFSIZE))) {
|
|
|
|
#ifdef SWITCH_GRAMMAR_DIR
|
|
|
|
switch_snprintf(SWITCH_GLOBAL_dirs.grammar_dir, BUFSIZE, "%s", SWITCH_GRAMMAR_DIR);
|
2006-08-20 03:04:55 +00:00
|
|
|
#else
|
2007-03-11 01:07:47 +00:00
|
|
|
switch_snprintf(SWITCH_GLOBAL_dirs.grammar_dir, BUFSIZE, "%s%sgrammar", base_dir, SWITCH_PATH_SEPARATOR);
|
2006-08-20 03:04:55 +00:00
|
|
|
#endif
|
2007-03-11 01:07:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!SWITCH_GLOBAL_dirs.temp_dir && (SWITCH_GLOBAL_dirs.temp_dir = (char *) malloc(BUFSIZE))) {
|
2006-03-01 06:25:56 +00:00
|
|
|
#ifdef SWITCH_TEMP_DIR
|
2007-03-11 01:07:47 +00:00
|
|
|
switch_snprintf(SWITCH_GLOBAL_dirs.temp_dir, BUFSIZE, "%s", SWITCH_TEMP_DIR);
|
2006-03-01 06:25:56 +00:00
|
|
|
#else
|
|
|
|
#ifdef WIN32
|
2007-03-11 01:07:47 +00:00
|
|
|
GetTempPath(dwBufSize, lpPathBuffer);
|
|
|
|
switch_snprintf(SWITCH_GLOBAL_dirs.temp_dir, BUFSIZE, "%s", lpPathBuffer);
|
2006-03-01 06:25:56 +00:00
|
|
|
#else
|
2007-03-11 01:07:47 +00:00
|
|
|
switch_snprintf(SWITCH_GLOBAL_dirs.temp_dir, BUFSIZE, "%s", "/tmp/");
|
2006-03-01 06:25:56 +00:00
|
|
|
#endif
|
|
|
|
#endif
|
2007-03-11 01:07:47 +00:00
|
|
|
}
|
|
|
|
|
2006-03-01 17:06:10 +00:00
|
|
|
}
|
|
|
|
|
2006-07-07 18:59:14 +00:00
|
|
|
|
2006-09-20 20:25:26 +00:00
|
|
|
SWITCH_DECLARE(int32_t) set_high_priority(void)
|
|
|
|
{
|
|
|
|
#ifdef __linux__
|
2007-03-29 22:31:56 +00:00
|
|
|
struct sched_param sched = { 0 };
|
2006-09-20 20:25:26 +00:00
|
|
|
sched.sched_priority = 1;
|
|
|
|
if (sched_setscheduler(0, SCHED_RR, &sched)) {
|
2007-03-29 22:31:56 +00:00
|
|
|
sched.sched_priority = 0;
|
|
|
|
if (sched_setscheduler(0, SCHED_OTHER, &sched)) {
|
2006-09-20 20:25:26 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef WIN32
|
|
|
|
SetPriorityClass(GetCurrentProcess(), HIGH_PRIORITY_CLASS);
|
|
|
|
#else
|
|
|
|
nice(-10);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#define USE_MLOCKALL
|
|
|
|
#ifdef HAVE_MLOCKALL
|
|
|
|
#ifdef USE_MLOCKALL
|
2007-03-29 22:31:56 +00:00
|
|
|
mlockall(MCL_CURRENT | MCL_FUTURE);
|
2006-09-20 20:25:26 +00:00
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(void) switch_core_runtime_loop(int bg)
|
|
|
|
{
|
2006-09-20 20:47:28 +00:00
|
|
|
#ifdef WIN32
|
|
|
|
HANDLE shutdown_event;
|
|
|
|
char path[256] = "";
|
|
|
|
#endif
|
2006-09-20 20:25:26 +00:00
|
|
|
if (bg) {
|
|
|
|
bg = 0;
|
|
|
|
#ifdef WIN32
|
2006-09-20 20:47:28 +00:00
|
|
|
snprintf(path, sizeof(path), "Global\\Freeswitch.%d", getpid());
|
2007-03-29 22:31:56 +00:00
|
|
|
shutdown_event = CreateEvent(NULL, FALSE, FALSE, path);
|
2006-09-20 20:25:26 +00:00
|
|
|
WaitForSingleObject(shutdown_event, INFINITE);
|
|
|
|
#else
|
|
|
|
runtime.running = 1;
|
2007-03-29 22:31:56 +00:00
|
|
|
while (runtime.running) {
|
2006-09-20 20:25:26 +00:00
|
|
|
switch_yield(1000000);
|
|
|
|
}
|
|
|
|
#endif
|
2007-03-29 22:31:56 +00:00
|
|
|
} else {
|
2006-09-20 20:25:26 +00:00
|
|
|
/* wait for console input */
|
|
|
|
switch_console_loop();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-06-05 18:36:02 +00:00
|
|
|
SWITCH_DECLARE(switch_status_t) switch_core_init(char *console, const char **err)
|
2006-03-01 17:06:10 +00:00
|
|
|
{
|
2006-07-11 20:59:07 +00:00
|
|
|
switch_xml_t xml = NULL, cfg = NULL;
|
2007-02-22 23:12:58 +00:00
|
|
|
switch_uuid_t uuid;
|
2006-03-01 17:06:10 +00:00
|
|
|
memset(&runtime, 0, sizeof(runtime));
|
|
|
|
|
2006-04-11 21:13:44 +00:00
|
|
|
/* INIT APR and Create the pool context */
|
|
|
|
if (apr_initialize() != SWITCH_STATUS_SUCCESS) {
|
2006-06-05 18:36:02 +00:00
|
|
|
*err = "FATAL ERROR! Could not initilize APR\n";
|
2006-04-11 21:13:44 +00:00
|
|
|
return SWITCH_STATUS_MEMERR;
|
|
|
|
}
|
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
if (!(runtime.memory_pool = switch_core_memory_init())) {
|
|
|
|
*err = "FATAL ERROR! Could noat allocate memory pool\n";
|
2006-04-11 21:13:44 +00:00
|
|
|
return SWITCH_STATUS_MEMERR;
|
|
|
|
}
|
2007-03-29 22:31:56 +00:00
|
|
|
assert(runtime.memory_pool != NULL);
|
2006-04-11 21:13:44 +00:00
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
switch_core_set_globals();
|
|
|
|
switch_core_session_init(runtime.memory_pool);
|
Ringback (sponsored by Front Logic)
This addition lets you set artifical ringback on a channel
that is waiting for an originated call to be answered.
the syntax is
<action application="set" data="ringback=[data]"/>
where data is either the full path to an audio file
or a teletone generation script..
syntax of teletone scripts
LEGEND:
0-9,a-d,*,# (standard dtmf tones)
variables: c,r,d,v,>,<,+,w,l,L,%
c (channels) - Sets the number of channels.
r (rate) - Sets the sample rate.
d (duration) - Sets the default tone duration.
v (volume) - Sets the default volume.
> (decrease vol) - factor to decrease volume by per frame (0 for even decrease across duration).
< (increase vol) - factor to increase volume by per frame (0 for even increase across duration).
+ (step) - factor to step by used by < and >.
w (wait) - default silence after each tone.
l (loops) - number of times to repeat each tone in the script.
L (LOOPS) - number of times to repeat the the whole script.
% (manual tone) - a generic tone specified by a duration, a wait and a list of frequencies.
standard tones can have custom duration per use with the () modifier
7(1000, 500) to generate DTMF 7 for 1 second then pause .5 seconds
EXAMPLES
UK Ring Tone [400+450 hz on for 400ms off for 200ms then 400+450 hz on for 400ms off for 2200ms]
%(400,200,400,450);%(400,2200,400,450)
US Ring Tone [440+480 hz on for 2000ms off for 4000ms]
%(2000,4000,440,480)
ATT BONG [volume level 4000, even decay, step by 2, # key for 60ms with no wait, volume level 2000, 350+440hz {us dialtone} for 940ms
v=4000;>=0;+=2;#(60,0);v=2000;%(940,0,350,440)
SIT Tone 913.8 hz for 274 ms with no wait, 1370.6 hz for 274 ms with no wait, 1776.7 hz for 380ms with no wait
%(274,0,913.8);%(274,0,1370.6);%(380,0,1776.7)
ATTN TONE (phone's off the hook!) 1400+2060+2450+2600 hz for 100ms with 100ms wait
%(100,100,1400,2060,2450,2600)
git-svn-id: http://svn.freeswitch.org/svn/freeswitch/trunk@3408 d0543943-73ff-0310-b7d9-9358b9ac24b2
2006-11-19 01:05:06 +00:00
|
|
|
switch_core_hash_init(&runtime.global_vars, runtime.memory_pool);
|
|
|
|
|
2006-06-05 18:36:02 +00:00
|
|
|
if (switch_xml_init(runtime.memory_pool, err) != SWITCH_STATUS_SUCCESS) {
|
2006-05-15 18:16:43 +00:00
|
|
|
apr_terminate();
|
2006-05-10 03:23:05 +00:00
|
|
|
return SWITCH_STATUS_MEMERR;
|
|
|
|
}
|
|
|
|
|
2006-07-07 18:59:14 +00:00
|
|
|
|
|
|
|
if ((xml = switch_xml_open_cfg("switch.conf", &cfg, NULL))) {
|
|
|
|
switch_xml_t settings, param;
|
2007-03-29 22:31:56 +00:00
|
|
|
|
2006-07-07 18:59:14 +00:00
|
|
|
if ((settings = switch_xml_child(cfg, "settings"))) {
|
|
|
|
for (param = switch_xml_child(settings, "param"); param; param = param->next) {
|
2007-02-16 23:36:10 +00:00
|
|
|
const char *var = switch_xml_attr_soft(param, "name");
|
|
|
|
const char *val = switch_xml_attr_soft(param, "value");
|
2007-03-29 22:31:56 +00:00
|
|
|
|
2006-07-07 18:59:14 +00:00
|
|
|
if (!strcasecmp(var, "max-sessions")) {
|
2007-03-29 22:31:56 +00:00
|
|
|
switch_core_session_limit(atoi(val));
|
2006-07-07 18:59:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
Ringback (sponsored by Front Logic)
This addition lets you set artifical ringback on a channel
that is waiting for an originated call to be answered.
the syntax is
<action application="set" data="ringback=[data]"/>
where data is either the full path to an audio file
or a teletone generation script..
syntax of teletone scripts
LEGEND:
0-9,a-d,*,# (standard dtmf tones)
variables: c,r,d,v,>,<,+,w,l,L,%
c (channels) - Sets the number of channels.
r (rate) - Sets the sample rate.
d (duration) - Sets the default tone duration.
v (volume) - Sets the default volume.
> (decrease vol) - factor to decrease volume by per frame (0 for even decrease across duration).
< (increase vol) - factor to increase volume by per frame (0 for even increase across duration).
+ (step) - factor to step by used by < and >.
w (wait) - default silence after each tone.
l (loops) - number of times to repeat each tone in the script.
L (LOOPS) - number of times to repeat the the whole script.
% (manual tone) - a generic tone specified by a duration, a wait and a list of frequencies.
standard tones can have custom duration per use with the () modifier
7(1000, 500) to generate DTMF 7 for 1 second then pause .5 seconds
EXAMPLES
UK Ring Tone [400+450 hz on for 400ms off for 200ms then 400+450 hz on for 400ms off for 2200ms]
%(400,200,400,450);%(400,2200,400,450)
US Ring Tone [440+480 hz on for 2000ms off for 4000ms]
%(2000,4000,440,480)
ATT BONG [volume level 4000, even decay, step by 2, # key for 60ms with no wait, volume level 2000, 350+440hz {us dialtone} for 940ms
v=4000;>=0;+=2;#(60,0);v=2000;%(940,0,350,440)
SIT Tone 913.8 hz for 274 ms with no wait, 1370.6 hz for 274 ms with no wait, 1776.7 hz for 380ms with no wait
%(274,0,913.8);%(274,0,1370.6);%(380,0,1776.7)
ATTN TONE (phone's off the hook!) 1400+2060+2450+2600 hz for 100ms with 100ms wait
%(100,100,1400,2060,2450,2600)
git-svn-id: http://svn.freeswitch.org/svn/freeswitch/trunk@3408 d0543943-73ff-0310-b7d9-9358b9ac24b2
2006-11-19 01:05:06 +00:00
|
|
|
|
|
|
|
if ((settings = switch_xml_child(cfg, "variables"))) {
|
|
|
|
for (param = switch_xml_child(settings, "variable"); param; param = param->next) {
|
2007-02-16 23:36:10 +00:00
|
|
|
const char *var = switch_xml_attr_soft(param, "name");
|
|
|
|
const char *val = switch_xml_attr_soft(param, "value");
|
Ringback (sponsored by Front Logic)
This addition lets you set artifical ringback on a channel
that is waiting for an originated call to be answered.
the syntax is
<action application="set" data="ringback=[data]"/>
where data is either the full path to an audio file
or a teletone generation script..
syntax of teletone scripts
LEGEND:
0-9,a-d,*,# (standard dtmf tones)
variables: c,r,d,v,>,<,+,w,l,L,%
c (channels) - Sets the number of channels.
r (rate) - Sets the sample rate.
d (duration) - Sets the default tone duration.
v (volume) - Sets the default volume.
> (decrease vol) - factor to decrease volume by per frame (0 for even decrease across duration).
< (increase vol) - factor to increase volume by per frame (0 for even increase across duration).
+ (step) - factor to step by used by < and >.
w (wait) - default silence after each tone.
l (loops) - number of times to repeat each tone in the script.
L (LOOPS) - number of times to repeat the the whole script.
% (manual tone) - a generic tone specified by a duration, a wait and a list of frequencies.
standard tones can have custom duration per use with the () modifier
7(1000, 500) to generate DTMF 7 for 1 second then pause .5 seconds
EXAMPLES
UK Ring Tone [400+450 hz on for 400ms off for 200ms then 400+450 hz on for 400ms off for 2200ms]
%(400,200,400,450);%(400,2200,400,450)
US Ring Tone [440+480 hz on for 2000ms off for 4000ms]
%(2000,4000,440,480)
ATT BONG [volume level 4000, even decay, step by 2, # key for 60ms with no wait, volume level 2000, 350+440hz {us dialtone} for 940ms
v=4000;>=0;+=2;#(60,0);v=2000;%(940,0,350,440)
SIT Tone 913.8 hz for 274 ms with no wait, 1370.6 hz for 274 ms with no wait, 1776.7 hz for 380ms with no wait
%(274,0,913.8);%(274,0,1370.6);%(380,0,1776.7)
ATTN TONE (phone's off the hook!) 1400+2060+2450+2600 hz for 100ms with 100ms wait
%(100,100,1400,2060,2450,2600)
git-svn-id: http://svn.freeswitch.org/svn/freeswitch/trunk@3408 d0543943-73ff-0310-b7d9-9358b9ac24b2
2006-11-19 01:05:06 +00:00
|
|
|
char *varr = NULL, *vall = NULL;
|
|
|
|
|
|
|
|
varr = switch_core_strdup(runtime.memory_pool, var);
|
|
|
|
vall = switch_core_strdup(runtime.memory_pool, val);
|
|
|
|
switch_core_hash_insert(runtime.global_vars, varr, vall);
|
|
|
|
}
|
|
|
|
}
|
2006-07-07 18:59:14 +00:00
|
|
|
switch_xml_free(xml);
|
|
|
|
}
|
|
|
|
|
2006-06-05 18:36:02 +00:00
|
|
|
*err = NULL;
|
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
if (console) {
|
2006-03-01 17:06:10 +00:00
|
|
|
if (*console != '/') {
|
|
|
|
char path[265];
|
|
|
|
snprintf(path, sizeof(path), "%s%s%s", SWITCH_GLOBAL_dirs.log_dir, SWITCH_PATH_SEPARATOR, console);
|
|
|
|
console = path;
|
|
|
|
}
|
2006-06-05 18:36:02 +00:00
|
|
|
if (switch_core_set_console(console) != SWITCH_STATUS_SUCCESS) {
|
|
|
|
*err = "FATAL ERROR! Could not open console\n";
|
|
|
|
apr_terminate();
|
|
|
|
return SWITCH_STATUS_GENERR;
|
|
|
|
}
|
2006-02-26 03:13:01 +00:00
|
|
|
} else {
|
|
|
|
runtime.console = stdout;
|
|
|
|
}
|
2005-11-19 20:07:43 +00:00
|
|
|
|
2006-04-26 19:11:49 +00:00
|
|
|
assert(runtime.memory_pool != NULL);
|
|
|
|
switch_log_init(runtime.memory_pool);
|
2005-12-13 19:53:29 +00:00
|
|
|
switch_event_init(runtime.memory_pool);
|
2007-03-29 22:31:56 +00:00
|
|
|
switch_core_sqldb_start(runtime.memory_pool);
|
2006-04-05 20:17:22 +00:00
|
|
|
switch_rtp_init(runtime.memory_pool);
|
2007-03-29 22:31:56 +00:00
|
|
|
runtime.running = 1;
|
2005-12-28 05:17:21 +00:00
|
|
|
|
2005-12-21 22:25:22 +00:00
|
|
|
|
2006-05-05 13:35:33 +00:00
|
|
|
#ifdef CRASH_PROT
|
2006-04-27 21:09:58 +00:00
|
|
|
switch_core_hash_init(&runtime.stack_table, runtime.memory_pool);
|
2006-05-05 13:35:33 +00:00
|
|
|
#endif
|
2007-03-28 23:37:12 +00:00
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
|
|
|
|
switch_scheduler_task_thread_start();
|
2006-05-10 03:23:05 +00:00
|
|
|
runtime.initiated = switch_time_now();
|
2007-02-22 17:38:34 +00:00
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
switch_scheduler_add_task(time(NULL), heartbeat_callback, "heartbeat", "core", 0, NULL, SSHF_NONE);
|
|
|
|
|
2007-02-22 17:38:34 +00:00
|
|
|
|
|
|
|
switch_uuid_get(&uuid);
|
|
|
|
switch_uuid_format(runtime.uuid_str, &uuid);
|
|
|
|
|
2005-11-19 20:07:43 +00:00
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
2006-11-20 21:43:44 +00:00
|
|
|
|
2006-08-19 18:51:22 +00:00
|
|
|
#ifdef SIGPIPE
|
2006-10-02 16:48:00 +00:00
|
|
|
static void handle_SIGPIPE(int sig)
|
2006-08-19 18:51:22 +00:00
|
|
|
{
|
2007-03-29 22:31:56 +00:00
|
|
|
if (sig);
|
2006-08-19 18:51:22 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CONSOLE, "Sig Pipe!\n");
|
2006-10-02 16:48:00 +00:00
|
|
|
return;
|
2006-08-19 18:51:22 +00:00
|
|
|
}
|
|
|
|
#endif
|
2006-11-20 21:43:44 +00:00
|
|
|
|
|
|
|
#ifdef SIGPOLL
|
|
|
|
static void handle_SIGPOLL(int sig)
|
|
|
|
{
|
2007-03-29 22:31:56 +00:00
|
|
|
if (sig);
|
2006-11-20 21:43:44 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CONSOLE, "Sig Poll!\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef SIGIO
|
|
|
|
static void handle_SIGIO(int sig)
|
|
|
|
{
|
2007-03-29 22:31:56 +00:00
|
|
|
if (sig);
|
2006-11-20 21:43:44 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CONSOLE, "Sig I/O!\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2006-08-19 18:51:22 +00:00
|
|
|
#ifdef TRAP_BUS
|
2006-10-02 16:48:00 +00:00
|
|
|
static void handle_SIGBUS(int sig)
|
2006-08-19 18:51:22 +00:00
|
|
|
{
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CONSOLE, "Sig BUS!\n");
|
2006-10-02 16:48:00 +00:00
|
|
|
return;
|
2006-08-19 18:51:22 +00:00
|
|
|
}
|
|
|
|
#endif
|
2005-11-19 20:07:43 +00:00
|
|
|
|
2006-08-19 18:51:22 +00:00
|
|
|
/* no ctl-c mofo */
|
2006-10-02 16:48:00 +00:00
|
|
|
static void handle_SIGINT(int sig)
|
2006-08-19 18:51:22 +00:00
|
|
|
{
|
|
|
|
if (sig);
|
2006-10-02 16:48:00 +00:00
|
|
|
return;
|
2006-08-19 18:51:22 +00:00
|
|
|
}
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_core_init_and_modload(char *console, const char **err)
|
|
|
|
{
|
|
|
|
switch_event_t *event;
|
|
|
|
if (switch_core_init(console, err) != SWITCH_STATUS_SUCCESS) {
|
|
|
|
return SWITCH_STATUS_GENERR;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* set signal handlers */
|
2006-10-02 16:48:00 +00:00
|
|
|
signal(SIGINT, handle_SIGINT);
|
2006-08-19 18:51:22 +00:00
|
|
|
#ifdef SIGPIPE
|
2006-10-02 16:48:00 +00:00
|
|
|
signal(SIGPIPE, handle_SIGPIPE);
|
2006-08-19 18:51:22 +00:00
|
|
|
#endif
|
2006-11-20 21:43:44 +00:00
|
|
|
#ifdef SIGPOLL
|
|
|
|
signal(SIGPIPE, handle_SIGPOLL);
|
|
|
|
#endif
|
|
|
|
#ifdef SIGIO
|
|
|
|
signal(SIGPIPE, handle_SIGIO);
|
|
|
|
#endif
|
2006-08-19 18:51:22 +00:00
|
|
|
#ifdef TRAP_BUS
|
2006-10-02 16:48:00 +00:00
|
|
|
signal(SIGBUS, handle_SIGBUS);
|
2006-08-19 18:51:22 +00:00
|
|
|
#endif
|
2007-03-29 22:31:56 +00:00
|
|
|
|
2006-08-19 18:51:22 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CONSOLE, "Bringing up environment.\n");
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CONSOLE, "Loading Modules.\n");
|
|
|
|
if (switch_loadable_module_init() != SWITCH_STATUS_SUCCESS) {
|
|
|
|
*err = "Cannot load modules";
|
2007-02-13 02:48:50 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CONSOLE, "Error: %s", *err);
|
2006-08-19 18:51:22 +00:00
|
|
|
return SWITCH_STATUS_GENERR;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (switch_event_create(&event, SWITCH_EVENT_STARTUP) == SWITCH_STATUS_SUCCESS) {
|
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "Event-Info", "System Ready");
|
|
|
|
switch_event_fire(&event);
|
|
|
|
}
|
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CONSOLE,
|
2007-03-30 00:13:31 +00:00
|
|
|
"\nFreeSWITCH Version %s Started.\nCrash Protection [%s]\nMax Sessions[%u]\n\n", SWITCH_VERSION_FULL, __CP, switch_core_session_limit(0));
|
2006-08-19 18:51:22 +00:00
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
|
|
|
|
}
|
2005-11-19 20:07:43 +00:00
|
|
|
|
2006-05-10 03:23:05 +00:00
|
|
|
SWITCH_DECLARE(void) switch_core_measure_time(switch_time_t total_ms, switch_core_time_duration_t *duration)
|
|
|
|
{
|
2007-03-29 22:31:56 +00:00
|
|
|
switch_time_t temp = total_ms / 1000;
|
2006-05-10 04:05:07 +00:00
|
|
|
memset(duration, 0, sizeof(*duration));
|
2007-03-29 22:31:56 +00:00
|
|
|
duration->mms = (uint32_t) (total_ms % 1000);
|
|
|
|
duration->ms = (uint32_t) (temp % 1000);
|
|
|
|
temp = temp / 1000;
|
|
|
|
duration->sec = (uint32_t) (temp % 60);
|
2006-05-10 04:42:31 +00:00
|
|
|
temp = temp / 60;
|
2007-03-29 22:31:56 +00:00
|
|
|
duration->min = (uint32_t) (temp % 60);
|
|
|
|
temp = temp / 60;
|
|
|
|
duration->hr = (uint32_t) (temp % 24);
|
2006-05-10 04:42:31 +00:00
|
|
|
temp = temp / 24;
|
2007-03-29 22:31:56 +00:00
|
|
|
duration->day = (uint32_t) (temp % 365);
|
|
|
|
duration->yr = (uint32_t) (temp / 365);
|
2006-05-10 03:23:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(switch_time_t) switch_core_uptime(void)
|
|
|
|
{
|
|
|
|
return switch_time_now() - runtime.initiated;
|
|
|
|
}
|
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
SWITCH_DECLARE(int32_t) switch_core_session_ctl(switch_session_ctl_t cmd, uint32_t * val)
|
2006-09-20 20:25:26 +00:00
|
|
|
{
|
|
|
|
|
|
|
|
if (runtime.shutting_down) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (cmd) {
|
|
|
|
case SCSC_PAUSE_INBOUND:
|
|
|
|
runtime.no_new_sessions = *val;
|
|
|
|
break;
|
|
|
|
case SCSC_HUPALL:
|
|
|
|
switch_core_session_hupall(SWITCH_CAUSE_MANAGER_REQUEST);
|
|
|
|
break;
|
|
|
|
case SCSC_SHUTDOWN:
|
|
|
|
runtime.running = 0;
|
|
|
|
break;
|
|
|
|
case SCSC_CHECK_RUNNING:
|
|
|
|
*val = runtime.running;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
SWITCH_DECLARE(switch_bool_t) switch_core_ready(void)
|
|
|
|
{
|
|
|
|
return (runtime.shutting_down || runtime.no_new_sessions) ? SWITCH_FALSE : SWITCH_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-03-01 17:23:02 +00:00
|
|
|
SWITCH_DECLARE(switch_status_t) switch_core_destroy(void)
|
2005-11-19 20:07:43 +00:00
|
|
|
{
|
2006-08-19 18:51:22 +00:00
|
|
|
switch_event_t *event;
|
|
|
|
if (switch_event_create(&event, SWITCH_EVENT_SHUTDOWN) == SWITCH_STATUS_SUCCESS) {
|
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "Event-Info", "System Shutting Down");
|
|
|
|
switch_event_fire(&event);
|
|
|
|
}
|
2006-09-20 20:25:26 +00:00
|
|
|
runtime.shutting_down = 1;
|
|
|
|
runtime.no_new_sessions = 1;
|
2006-08-19 18:51:22 +00:00
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
|
2006-08-19 18:51:22 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CONSOLE, "End existing sessions\n");
|
2006-09-20 20:25:26 +00:00
|
|
|
switch_core_session_hupall(SWITCH_CAUSE_SYSTEM_SHUTDOWN);
|
2006-08-19 18:51:22 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CONSOLE, "Clean up modules.\n");
|
|
|
|
switch_loadable_module_shutdown();
|
2005-12-14 20:22:19 +00:00
|
|
|
|
2006-08-15 19:02:06 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CONSOLE, "Closing Event Engine.\n");
|
|
|
|
switch_event_shutdown();
|
2007-03-28 23:37:12 +00:00
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
switch_core_sqldb_stop();
|
|
|
|
switch_scheduler_task_thread_stop();
|
2007-03-28 23:37:12 +00:00
|
|
|
|
2006-05-10 03:23:05 +00:00
|
|
|
switch_xml_destroy();
|
2006-08-15 19:02:06 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CONSOLE, "Finalizing Shutdown.\n");
|
2006-08-15 17:52:12 +00:00
|
|
|
switch_log_shutdown();
|
2007-03-28 23:37:12 +00:00
|
|
|
|
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
|
|
|
|
if (runtime.console != stdout && runtime.console != stderr) {
|
2006-02-26 03:13:01 +00:00
|
|
|
fclose(runtime.console);
|
|
|
|
runtime.console = NULL;
|
|
|
|
}
|
2005-12-14 21:29:46 +00:00
|
|
|
|
2007-03-29 22:31:56 +00:00
|
|
|
switch_yield(1000000);
|
2006-12-29 16:39:56 +00:00
|
|
|
|
2007-03-11 01:07:47 +00:00
|
|
|
switch_safe_free(SWITCH_GLOBAL_dirs.base_dir);
|
|
|
|
switch_safe_free(SWITCH_GLOBAL_dirs.mod_dir);
|
|
|
|
switch_safe_free(SWITCH_GLOBAL_dirs.conf_dir);
|
|
|
|
switch_safe_free(SWITCH_GLOBAL_dirs.log_dir);
|
|
|
|
switch_safe_free(SWITCH_GLOBAL_dirs.db_dir);
|
|
|
|
switch_safe_free(SWITCH_GLOBAL_dirs.script_dir);
|
|
|
|
switch_safe_free(SWITCH_GLOBAL_dirs.htdocs_dir);
|
|
|
|
switch_safe_free(SWITCH_GLOBAL_dirs.grammar_dir);
|
|
|
|
switch_safe_free(SWITCH_GLOBAL_dirs.temp_dir);
|
2006-08-20 03:36:14 +00:00
|
|
|
|
2007-01-22 17:16:25 +00:00
|
|
|
if (runtime.memory_pool) {
|
|
|
|
apr_pool_destroy(runtime.memory_pool);
|
2007-03-04 01:47:08 +00:00
|
|
|
/* apr_terminate(); */
|
2007-01-22 17:16:25 +00:00
|
|
|
}
|
|
|
|
|
2005-11-19 20:07:43 +00:00
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
2006-11-27 22:30:48 +00:00
|
|
|
|
2007-02-21 21:46:32 +00:00
|
|
|
|
2006-11-27 22:30:48 +00:00
|
|
|
/* For Emacs:
|
|
|
|
* Local Variables:
|
|
|
|
* mode:c
|
2007-02-09 02:36:03 +00:00
|
|
|
* indent-tabs-mode:t
|
2006-11-27 22:30:48 +00:00
|
|
|
* tab-width:4
|
|
|
|
* c-basic-offset:4
|
|
|
|
* End:
|
|
|
|
* For VIM:
|
|
|
|
* vim:set softtabstop=4 shiftwidth=4 tabstop=4 expandtab:
|
|
|
|
*/
|