/* * FreeSWITCH Modular Media Switching Software Library / Soft-Switch Application * Copyright (C) 2005/2006, Anthony Minessale II * * 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 * Portions created by the Initial Developer are Copyright (C) * the Initial Developer. All Rights Reserved. * * Contributor(s): * * Anthony Minessale II * * * mod_softtimer.c -- Software Timer Module * */ #include #include static switch_memory_pool_t *module_pool = NULL; static struct { int32_t RUNNING; switch_mutex_t *mutex; } globals; static const char modname[] = "mod_softtimer"; #define MAX_ELEMENTS 1000 struct timer_private { uint64_t reference; }; typedef struct timer_private timer_private_t; struct timer_matrix { uint64_t tick; uint32_t count; }; typedef struct timer_matrix timer_matrix_t; static timer_matrix_t TIMER_MATRIX[MAX_ELEMENTS+1]; #define IDLE_SPEED 100 static inline switch_status_t timer_init(switch_timer_t *timer) { timer_private_t *private_info; if ((private_info = switch_core_alloc(timer->memory_pool, sizeof(*private_info)))) { switch_mutex_lock(globals.mutex); TIMER_MATRIX[timer->interval].count++; switch_mutex_unlock(globals.mutex); timer->private_info = private_info; private_info->reference = TIMER_MATRIX[timer->interval].tick; return SWITCH_STATUS_SUCCESS; } return SWITCH_STATUS_MEMERR; } static inline switch_status_t timer_step(switch_timer_t *timer) { timer_private_t *private_info = timer->private_info; private_info->reference += timer->interval; return SWITCH_STATUS_SUCCESS; } static inline switch_status_t timer_next(switch_timer_t *timer) { timer_private_t *private_info = timer->private_info; timer_step(timer); while (TIMER_MATRIX[timer->interval].tick < private_info->reference) { switch_yield(1000); } timer->samplecount += timer->samples; return SWITCH_STATUS_SUCCESS; } static inline switch_status_t timer_check(switch_timer_t *timer) { timer_private_t *private_info = timer->private_info; switch_status_t status; if (TIMER_MATRIX[timer->interval].tick < private_info->reference) { status = SWITCH_STATUS_FALSE; } else { private_info->reference += timer->interval; status = SWITCH_STATUS_SUCCESS; } return status; } static inline switch_status_t timer_destroy(switch_timer_t *timer) { switch_mutex_lock(globals.mutex); TIMER_MATRIX[timer->interval].count--; switch_mutex_unlock(globals.mutex); timer->private_info = NULL; return SWITCH_STATUS_SUCCESS; } static const switch_timer_interface_t timer_interface = { /*.interface_name */ "soft", /*.timer_init */ timer_init, /*.timer_next */ timer_next, /*.timer_step */ timer_step, /*.timer_check */ timer_check, /*.timer_destroy */ timer_destroy }; static const switch_loadable_module_interface_t mod_softtimer_module_interface = { /*.module_name */ modname, /*.endpoint_interface */ NULL, /*.timer_interface */ &timer_interface, /*.switch_dialplan_interface */ NULL, /*.switch_codec_interface */ NULL, /*.switch_application_interface */ NULL }; SWITCH_MOD_DECLARE(switch_status_t) switch_module_load(const switch_loadable_module_interface_t **module_interface, char *filename) { if (switch_core_new_memory_pool(&module_pool) != SWITCH_STATUS_SUCCESS) { switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CRIT, "OH OH no pool\n"); return SWITCH_STATUS_MEMERR; } /* connect my internal structure to the blank pointer passed to me */ *module_interface = &mod_softtimer_module_interface; /* indicate that the module should continue to be loaded */ return SWITCH_STATUS_SUCCESS; } /* I cant resist setting this to 10ms, we dont even run anything smaller than 20ms so this is already twice the granularity we need, we'll change it if we need anything smaller */ #define STEP_MS 10 #define STEP_MIC 10000 SWITCH_MOD_DECLARE(switch_status_t) switch_module_runtime(void) { switch_time_t reference = switch_time_now(); uint32_t current_ms = 0; uint32_t x; memset(&globals, 0, sizeof(globals)); switch_mutex_init(&globals.mutex, SWITCH_MUTEX_NESTED, module_pool); globals.RUNNING = 1; while(globals.RUNNING == 1) { reference += STEP_MIC; while (switch_time_now() < reference) { switch_yield(STEP_MIC); } current_ms += STEP_MS; for (x = 0; x < MAX_ELEMENTS; x++) { int i = x, index; if (i == 0) { i = 1; } index = (current_ms % i == 0) ? i : 0; if (TIMER_MATRIX[index].count) { TIMER_MATRIX[index].tick += index; } } if (current_ms == MAX_ELEMENTS) { current_ms = 0; } } switch_mutex_lock(globals.mutex); globals.RUNNING = 0; switch_mutex_unlock(globals.mutex); return SWITCH_STATUS_TERM; } SWITCH_MOD_DECLARE(switch_status_t) switch_module_shutdown(void) { if (globals.RUNNING) { switch_mutex_lock(globals.mutex); globals.RUNNING = -1; switch_mutex_unlock(globals.mutex); while (globals.RUNNING) { switch_yield(10000); } } return SWITCH_STATUS_SUCCESS; } /* For Emacs: * Local Variables: * mode:c * indent-tabs-mode:nil * tab-width:4 * c-basic-offset:4 * End: * For VIM: * vim:set softtabstop=4 shiftwidth=4 tabstop=4 expandtab: */