2006-01-03 22:13:59 +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>
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* mod_softtimer.c -- Software Timer Module
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
#include <switch.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
|
2007-03-06 01:19:41 +00:00
|
|
|
#ifndef UINT32_MAX
|
|
|
|
#define UINT32_MAX 0xffffffff
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#define MAX_TICK UINT32_MAX - 1024
|
|
|
|
|
|
|
|
|
2006-09-16 20:18:24 +00:00
|
|
|
static switch_memory_pool_t *module_pool = NULL;
|
|
|
|
|
|
|
|
static struct {
|
|
|
|
int32_t RUNNING;
|
|
|
|
switch_mutex_t *mutex;
|
|
|
|
} globals;
|
2006-01-03 22:13:59 +00:00
|
|
|
|
2006-09-16 20:18:24 +00:00
|
|
|
static const char modname[] = "mod_softtimer";
|
|
|
|
#define MAX_ELEMENTS 1000
|
2006-01-03 22:13:59 +00:00
|
|
|
|
|
|
|
struct timer_private {
|
2007-03-05 20:53:54 +00:00
|
|
|
switch_size_t reference;
|
2007-03-06 01:19:41 +00:00
|
|
|
switch_size_t start;
|
|
|
|
uint32_t roll;
|
2007-03-07 23:24:09 +00:00
|
|
|
uint32_t ready;
|
2006-01-03 22:13:59 +00:00
|
|
|
};
|
2006-09-16 20:18:24 +00:00
|
|
|
typedef struct timer_private timer_private_t;
|
2006-01-03 22:13:59 +00:00
|
|
|
|
2006-09-16 20:18:24 +00:00
|
|
|
struct timer_matrix {
|
2007-03-05 20:53:54 +00:00
|
|
|
switch_size_t tick;
|
2006-09-16 20:18:24 +00:00
|
|
|
uint32_t count;
|
2007-03-06 01:19:41 +00:00
|
|
|
uint32_t roll;
|
2006-09-16 20:18:24 +00:00
|
|
|
};
|
|
|
|
typedef struct timer_matrix timer_matrix_t;
|
2006-01-03 22:13:59 +00:00
|
|
|
|
2006-09-16 20:18:24 +00:00
|
|
|
static timer_matrix_t TIMER_MATRIX[MAX_ELEMENTS+1];
|
2006-01-03 22:13:59 +00:00
|
|
|
|
2006-09-16 20:18:24 +00:00
|
|
|
#define IDLE_SPEED 100
|
2006-01-03 22:13:59 +00:00
|
|
|
|
|
|
|
|
2006-09-16 20:18:24 +00:00
|
|
|
static inline switch_status_t timer_init(switch_timer_t *timer)
|
|
|
|
{
|
|
|
|
timer_private_t *private_info;
|
|
|
|
|
2007-02-28 15:14:39 +00:00
|
|
|
if (globals.RUNNING != 1) {
|
|
|
|
return SWITCH_STATUS_FALSE;
|
|
|
|
}
|
|
|
|
|
2006-09-16 20:18:24 +00:00
|
|
|
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;
|
2007-03-06 01:19:41 +00:00
|
|
|
private_info->start = private_info->reference = TIMER_MATRIX[timer->interval].tick;
|
|
|
|
private_info->roll = TIMER_MATRIX[timer->interval].roll;
|
2007-03-07 23:24:09 +00:00
|
|
|
private_info->ready = 1;
|
2006-09-16 20:18:24 +00:00
|
|
|
return SWITCH_STATUS_SUCCESS;
|
2006-01-03 22:13:59 +00:00
|
|
|
}
|
|
|
|
|
2006-09-16 20:18:24 +00:00
|
|
|
return SWITCH_STATUS_MEMERR;
|
|
|
|
}
|
|
|
|
|
2007-03-06 01:19:41 +00:00
|
|
|
|
|
|
|
#define check_roll() if (private_info->roll < TIMER_MATRIX[timer->interval].roll) {\
|
|
|
|
private_info->roll++;\
|
|
|
|
private_info->reference = private_info->start = TIMER_MATRIX[timer->interval].tick;\
|
|
|
|
}\
|
|
|
|
|
|
|
|
|
|
|
|
|
2006-09-16 20:18:24 +00:00
|
|
|
static inline switch_status_t timer_step(switch_timer_t *timer)
|
|
|
|
{
|
|
|
|
timer_private_t *private_info = timer->private_info;
|
2007-03-06 01:19:41 +00:00
|
|
|
uint64_t samples;
|
2006-09-16 20:18:24 +00:00
|
|
|
|
2007-03-07 23:24:09 +00:00
|
|
|
if (globals.RUNNING != 1 || private_info->ready == 0) {
|
2007-02-28 15:14:39 +00:00
|
|
|
return SWITCH_STATUS_FALSE;
|
|
|
|
}
|
|
|
|
|
2007-03-06 01:19:41 +00:00
|
|
|
check_roll();
|
|
|
|
samples = timer->samples * (private_info->reference - private_info->start);
|
|
|
|
|
|
|
|
if (samples > UINT32_MAX) {
|
|
|
|
private_info->start = private_info->reference;
|
|
|
|
samples = timer->samples;
|
|
|
|
}
|
|
|
|
|
|
|
|
timer->samplecount = (uint32_t)samples;
|
|
|
|
private_info->reference++;
|
2007-03-05 20:53:54 +00:00
|
|
|
|
2007-02-07 19:18:15 +00:00
|
|
|
return SWITCH_STATUS_SUCCESS;
|
2006-01-03 22:13:59 +00:00
|
|
|
}
|
|
|
|
|
2006-09-16 20:18:24 +00:00
|
|
|
|
|
|
|
static inline switch_status_t timer_next(switch_timer_t *timer)
|
2006-09-12 22:23:45 +00:00
|
|
|
{
|
2006-09-16 20:18:24 +00:00
|
|
|
timer_private_t *private_info = timer->private_info;
|
2007-03-06 01:19:41 +00:00
|
|
|
|
2006-09-16 20:18:24 +00:00
|
|
|
timer_step(timer);
|
2007-02-28 15:14:39 +00:00
|
|
|
|
2007-03-07 23:24:09 +00:00
|
|
|
while (globals.RUNNING == 1 && private_info->ready && TIMER_MATRIX[timer->interval].tick < private_info->reference) {
|
2007-03-06 01:19:41 +00:00
|
|
|
check_roll();
|
|
|
|
switch_yield(1000);
|
2006-09-16 20:18:24 +00:00
|
|
|
}
|
2007-02-28 15:14:39 +00:00
|
|
|
|
2007-03-05 20:53:54 +00:00
|
|
|
if (globals.RUNNING == 1) {
|
2007-02-28 15:14:39 +00:00
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
return SWITCH_STATUS_FALSE;
|
2006-09-12 22:23:45 +00:00
|
|
|
}
|
|
|
|
|
2007-02-07 19:18:15 +00:00
|
|
|
static inline switch_status_t timer_check(switch_timer_t *timer)
|
2006-09-12 22:23:45 +00:00
|
|
|
|
|
|
|
{
|
2006-09-16 20:18:24 +00:00
|
|
|
timer_private_t *private_info = timer->private_info;
|
2007-02-07 18:44:00 +00:00
|
|
|
switch_status_t status = SWITCH_STATUS_SUCCESS;
|
2007-03-05 20:53:54 +00:00
|
|
|
switch_size_t diff;
|
2006-09-16 20:18:24 +00:00
|
|
|
|
2007-03-07 23:24:09 +00:00
|
|
|
if (globals.RUNNING != 1 || !private_info->ready) {
|
2007-02-28 15:14:39 +00:00
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2007-03-06 01:19:41 +00:00
|
|
|
check_roll();
|
|
|
|
|
2006-09-16 20:18:24 +00:00
|
|
|
if (TIMER_MATRIX[timer->interval].tick < private_info->reference) {
|
2007-02-07 19:18:15 +00:00
|
|
|
diff = private_info->reference - TIMER_MATRIX[timer->interval].tick;
|
2006-09-12 22:23:45 +00:00
|
|
|
} else {
|
2007-02-07 19:18:15 +00:00
|
|
|
diff = 0;
|
2006-09-12 22:23:45 +00:00
|
|
|
}
|
2006-09-16 20:18:24 +00:00
|
|
|
|
2007-02-07 19:18:15 +00:00
|
|
|
if (diff) {
|
2007-02-07 18:44:00 +00:00
|
|
|
status = SWITCH_STATUS_FALSE;
|
|
|
|
} else {
|
|
|
|
timer_step(timer);
|
|
|
|
}
|
|
|
|
|
2006-09-16 20:18:24 +00:00
|
|
|
return status;
|
2006-09-12 22:23:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-09-16 20:18:24 +00:00
|
|
|
static inline switch_status_t timer_destroy(switch_timer_t *timer)
|
2006-01-03 22:13:59 +00:00
|
|
|
{
|
2007-03-07 23:24:09 +00:00
|
|
|
timer_private_t *private_info = timer->private_info;
|
2006-09-16 20:18:24 +00:00
|
|
|
switch_mutex_lock(globals.mutex);
|
|
|
|
TIMER_MATRIX[timer->interval].count--;
|
2007-03-06 01:19:41 +00:00
|
|
|
if (TIMER_MATRIX[timer->interval].count == 0) {
|
|
|
|
TIMER_MATRIX[timer->interval].tick = 0;
|
|
|
|
}
|
2006-09-16 20:18:24 +00:00
|
|
|
switch_mutex_unlock(globals.mutex);
|
2007-03-07 23:24:09 +00:00
|
|
|
private_info->ready = 0;
|
2006-01-03 22:13:59 +00:00
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2006-09-16 20:18:24 +00:00
|
|
|
static const switch_timer_interface_t timer_interface = {
|
2006-09-16 20:21:17 +00:00
|
|
|
/*.interface_name */ "soft",
|
2006-09-16 20:18:24 +00:00
|
|
|
/*.timer_init */ timer_init,
|
|
|
|
/*.timer_next */ timer_next,
|
|
|
|
/*.timer_step */ timer_step,
|
|
|
|
/*.timer_check */ timer_check,
|
|
|
|
/*.timer_destroy */ timer_destroy
|
2006-01-03 22:13:59 +00:00
|
|
|
};
|
|
|
|
|
2006-09-16 20:18:24 +00:00
|
|
|
static const switch_loadable_module_interface_t mod_softtimer_module_interface = {
|
2006-01-20 15:05:05 +00:00
|
|
|
/*.module_name */ modname,
|
|
|
|
/*.endpoint_interface */ NULL,
|
2007-02-26 21:38:10 +00:00
|
|
|
/*.timer_interface */ &timer_interface
|
2006-01-03 22:13:59 +00:00
|
|
|
};
|
|
|
|
|
2006-04-30 18:24:24 +00:00
|
|
|
SWITCH_MOD_DECLARE(switch_status_t) switch_module_load(const switch_loadable_module_interface_t **module_interface, char *filename)
|
2006-01-20 15:05:05 +00:00
|
|
|
{
|
|
|
|
|
2006-09-16 20:18:24 +00:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2006-01-03 22:13:59 +00:00
|
|
|
/* connect my internal structure to the blank pointer passed to me */
|
2006-09-16 20:18:24 +00:00
|
|
|
*module_interface = &mod_softtimer_module_interface;
|
2006-01-03 22:13:59 +00:00
|
|
|
|
|
|
|
/* indicate that the module should continue to be loaded */
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
2006-09-16 20:18:24 +00:00
|
|
|
|
2006-09-22 02:10:27 +00:00
|
|
|
/* 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
|
|
|
|
*/
|
2006-09-16 20:18:24 +00:00
|
|
|
|
2007-02-07 18:44:00 +00:00
|
|
|
#define STEP_MS 1
|
|
|
|
#define STEP_MIC 1000
|
2006-09-16 20:18:24 +00:00
|
|
|
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) {
|
2006-09-22 02:10:27 +00:00
|
|
|
reference += STEP_MIC;
|
2006-09-16 20:18:24 +00:00
|
|
|
|
|
|
|
while (switch_time_now() < reference) {
|
2006-09-22 02:10:27 +00:00
|
|
|
switch_yield(STEP_MIC);
|
2006-09-16 20:18:24 +00:00
|
|
|
}
|
|
|
|
|
2006-09-22 02:10:27 +00:00
|
|
|
current_ms += STEP_MS;
|
2006-09-16 20:18:24 +00:00
|
|
|
|
|
|
|
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) {
|
2007-03-05 20:53:54 +00:00
|
|
|
TIMER_MATRIX[index].tick++;
|
2007-03-06 01:19:41 +00:00
|
|
|
if (TIMER_MATRIX[index].tick == MAX_TICK) {
|
|
|
|
TIMER_MATRIX[index].tick = 0;
|
|
|
|
TIMER_MATRIX[index].roll++;
|
|
|
|
}
|
2006-09-16 20:18:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
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:
|
|
|
|
*/
|