mirror of https://github.com/sipwise/kamailio.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
696 lines
13 KiB
696 lines
13 KiB
/*
|
|
* $Id$
|
|
*
|
|
* Copyright (C) 2007 Elena-Ramona Modroiu
|
|
*
|
|
* This file is part of Kamailio, a free SIP server.
|
|
*
|
|
* Kamailio is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version
|
|
*
|
|
* Kamailio is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*
|
|
*/
|
|
|
|
#include <assert.h>
|
|
#include <ctype.h>
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
|
|
#include "../../dprint.h"
|
|
#include "../../mem/mem.h"
|
|
#include "../../mem/shm_mem.h"
|
|
#include "../../shm_init.h"
|
|
#include "../../ut.h"
|
|
#include "../../pvar.h"
|
|
|
|
#include "pv_shv.h"
|
|
|
|
int shvar_locks_no=16;
|
|
gen_lock_set_t* shvar_locks=0;
|
|
|
|
static sh_var_t *sh_vars = 0;
|
|
static str shv_cpy = {0, 0};
|
|
|
|
/*
|
|
* Initialize locks
|
|
*/
|
|
int shvar_init_locks(void)
|
|
{
|
|
int i;
|
|
|
|
/* already initialized */
|
|
if(shvar_locks!=0)
|
|
return 0;
|
|
|
|
i = shvar_locks_no;
|
|
do {
|
|
if ((( shvar_locks=lock_set_alloc(i))!=0)&&
|
|
(lock_set_init(shvar_locks)!=0))
|
|
{
|
|
shvar_locks_no = i;
|
|
LM_INFO("locks array size %d\n", shvar_locks_no);
|
|
return 0;
|
|
|
|
}
|
|
if (shvar_locks){
|
|
lock_set_dealloc(shvar_locks);
|
|
shvar_locks=0;
|
|
}
|
|
i--;
|
|
if(i==0)
|
|
{
|
|
LM_ERR("failed to allocate locks\n");
|
|
return -1;
|
|
}
|
|
} while (1);
|
|
}
|
|
|
|
|
|
void shvar_unlock_locks(void)
|
|
{
|
|
unsigned int i;
|
|
|
|
if (shvar_locks==0)
|
|
return;
|
|
|
|
for (i=0;i<shvar_locks_no;i++) {
|
|
#ifdef GEN_LOCK_T_PREFERED
|
|
lock_release(&shvar_locks->locks[i]);
|
|
#else
|
|
shvar_release_idx(i);
|
|
#endif
|
|
};
|
|
}
|
|
|
|
|
|
void shvar_destroy_locks(void)
|
|
{
|
|
if (shvar_locks !=0){
|
|
lock_set_destroy(shvar_locks);
|
|
lock_set_dealloc(shvar_locks);
|
|
}
|
|
}
|
|
|
|
#ifndef GEN_LOCK_T_PREFERED
|
|
void shvar_lock_idx(int idx)
|
|
{
|
|
lock_set_get(shvar_locks, idx);
|
|
}
|
|
|
|
void shvar_release_idx(int idx)
|
|
{
|
|
lock_set_release(shvar_locks, idx);
|
|
}
|
|
#endif
|
|
|
|
/*
|
|
* Get lock
|
|
*/
|
|
void lock_shvar(sh_var_t *shv)
|
|
{
|
|
if(shv==NULL)
|
|
return;
|
|
#ifdef GEN_LOCK_T_PREFERED
|
|
lock_get(shv->lock);
|
|
#else
|
|
ul_lock_idx(shv->lockidx);
|
|
#endif
|
|
}
|
|
|
|
|
|
/*
|
|
* Release lock
|
|
*/
|
|
void unlock_shvar(sh_var_t *shv)
|
|
{
|
|
if(shv==NULL)
|
|
return;
|
|
#ifdef GEN_LOCK_T_PREFERED
|
|
lock_release(shv->lock);
|
|
#else
|
|
ul_release_idx(shv->lockidx);
|
|
#endif
|
|
}
|
|
|
|
|
|
sh_var_t* add_shvar(str *name)
|
|
{
|
|
sh_var_t *sit;
|
|
|
|
if(name==0 || name->s==0 || name->len<=0)
|
|
return 0;
|
|
|
|
if(!shm_initialized())
|
|
{
|
|
LM_ERR("shm not intialized - cannot define shm now\n");
|
|
return 0;
|
|
}
|
|
|
|
if(shvar_init_locks()!=0)
|
|
{
|
|
LM_ERR("cannot init shv locks\n");
|
|
return 0;
|
|
}
|
|
|
|
for(sit=sh_vars; sit; sit=sit->next)
|
|
{
|
|
if(sit->name.len==name->len
|
|
&& strncmp(name->s, sit->name.s, name->len)==0)
|
|
return sit;
|
|
}
|
|
sit = (sh_var_t*)shm_malloc(sizeof(sh_var_t));
|
|
if(sit==0)
|
|
{
|
|
LM_ERR("out of shm\n");
|
|
return 0;
|
|
}
|
|
memset(sit, 0, sizeof(sh_var_t));
|
|
sit->name.s = (char*)shm_malloc((name->len+1)*sizeof(char));
|
|
|
|
if(sit->name.s==0)
|
|
{
|
|
LM_ERR("out of shm!\n");
|
|
shm_free(sit);
|
|
return 0;
|
|
}
|
|
sit->name.len = name->len;
|
|
strncpy(sit->name.s, name->s, name->len);
|
|
sit->name.s[sit->name.len] = '\0';
|
|
|
|
if(sh_vars!=0)
|
|
sit->n = sh_vars->n + 1;
|
|
else
|
|
sit->n = 1;
|
|
|
|
#ifdef GEN_LOCK_T_PREFERED
|
|
sit->lock = &shvar_locks->locks[sit->n%shvar_locks_no];
|
|
#else
|
|
sit->lockidx = sit->n%shvar_locks_no;
|
|
#endif
|
|
|
|
sit->next = sh_vars;
|
|
|
|
sh_vars = sit;
|
|
|
|
return sit;
|
|
}
|
|
|
|
/* call it with lock set */
|
|
sh_var_t* set_shvar_value(sh_var_t* shv, int_str *value, int flags)
|
|
{
|
|
if(shv==NULL)
|
|
return NULL;
|
|
if(value==NULL)
|
|
{
|
|
if(shv->v.flags&VAR_VAL_STR)
|
|
{
|
|
shm_free(shv->v.value.s.s);
|
|
shv->v.flags &= ~VAR_VAL_STR;
|
|
}
|
|
memset(&shv->v.value, 0, sizeof(int_str));
|
|
|
|
return shv;
|
|
}
|
|
|
|
if(flags&VAR_VAL_STR)
|
|
{
|
|
if(shv->v.flags&VAR_VAL_STR)
|
|
{ /* old and new value is str */
|
|
if(value->s.len>shv->v.value.s.len)
|
|
{ /* not enough space to copy */
|
|
shm_free(shv->v.value.s.s);
|
|
memset(&shv->v.value, 0, sizeof(int_str));
|
|
shv->v.value.s.s =
|
|
(char*)shm_malloc((value->s.len+1)*sizeof(char));
|
|
if(shv->v.value.s.s==0)
|
|
{
|
|
LM_ERR("out of shm\n");
|
|
goto error;
|
|
}
|
|
}
|
|
} else {
|
|
memset(&shv->v.value, 0, sizeof(int_str));
|
|
shv->v.value.s.s =
|
|
(char*)shm_malloc((value->s.len+1)*sizeof(char));
|
|
if(shv->v.value.s.s==0)
|
|
{
|
|
LM_ERR("out of shm!\n");
|
|
goto error;
|
|
}
|
|
shv->v.flags |= VAR_VAL_STR;
|
|
}
|
|
strncpy(shv->v.value.s.s, value->s.s, value->s.len);
|
|
shv->v.value.s.len = value->s.len;
|
|
shv->v.value.s.s[value->s.len] = '\0';
|
|
} else {
|
|
if(shv->v.flags&VAR_VAL_STR)
|
|
{
|
|
shm_free(shv->v.value.s.s);
|
|
shv->v.flags &= ~VAR_VAL_STR;
|
|
memset(&shv->v.value, 0, sizeof(int_str));
|
|
}
|
|
shv->v.value.n = value->n;
|
|
}
|
|
|
|
return shv;
|
|
error:
|
|
/* set the var to init value */
|
|
memset(&shv->v.value, 0, sizeof(int_str));
|
|
shv->v.flags &= ~VAR_VAL_STR;
|
|
return NULL;
|
|
}
|
|
|
|
sh_var_t* get_shvar_by_name(str *name)
|
|
{
|
|
sh_var_t *it;
|
|
|
|
if(name==0 || name->s==0 || name->len<=0)
|
|
return 0;
|
|
|
|
for(it=sh_vars; it; it=it->next)
|
|
{
|
|
if(it->name.len==name->len
|
|
&& strncmp(name->s, it->name.s, name->len)==0)
|
|
return it;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
void reset_shvars(void)
|
|
{
|
|
sh_var_t *it;
|
|
for(it=sh_vars; it; it=it->next)
|
|
{
|
|
if(it->v.flags&VAR_VAL_STR)
|
|
{
|
|
shm_free(it->v.value.s.s);
|
|
it->v.flags &= ~VAR_VAL_STR;
|
|
}
|
|
memset(&it->v.value, 0, sizeof(int_str));
|
|
}
|
|
}
|
|
|
|
void destroy_shvars(void)
|
|
{
|
|
sh_var_t *it;
|
|
sh_var_t *it0;
|
|
|
|
it = sh_vars;
|
|
while(it)
|
|
{
|
|
it0 = it;
|
|
it = it->next;
|
|
shm_free(it0->name.s);
|
|
if(it0->v.flags&VAR_VAL_STR)
|
|
shm_free(it0->v.value.s.s);
|
|
shm_free(it0);
|
|
}
|
|
|
|
sh_vars = 0;
|
|
}
|
|
|
|
|
|
/********* PV functions *********/
|
|
int pv_parse_shvar_name(pv_spec_p sp, str *in)
|
|
{
|
|
if(in==NULL || in->s==NULL || sp==NULL)
|
|
return -1;
|
|
|
|
sp->pvp.pvn.type = PV_NAME_PVAR;
|
|
sp->pvp.pvn.u.dname = (void*)add_shvar(in);
|
|
|
|
if(sp->pvp.pvn.u.dname==NULL)
|
|
{
|
|
LM_ERR("cannot register shvar [%.*s]\n", in->len, in->s);
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int pv_get_shvar(struct sip_msg *msg, pv_param_t *param,
|
|
pv_value_t *res)
|
|
{
|
|
int len = 0;
|
|
char *sval = NULL;
|
|
sh_var_t *shv=NULL;
|
|
|
|
if(msg==NULL || res==NULL)
|
|
return -1;
|
|
|
|
if(param==NULL || param->pvn.u.dname==0)
|
|
return pv_get_null(msg, param, res);
|
|
|
|
shv= (sh_var_t*)param->pvn.u.dname;
|
|
|
|
lock_shvar(shv);
|
|
if(shv->v.flags&VAR_VAL_STR)
|
|
{
|
|
if(shv_cpy.s==NULL || shv_cpy.len < shv->v.value.s.len)
|
|
{
|
|
if(shv_cpy.s!=NULL)
|
|
pkg_free(shv_cpy.s);
|
|
shv_cpy.s = (char*)pkg_malloc(shv->v.value.s.len*sizeof(char));
|
|
if(shv_cpy.s==NULL)
|
|
{
|
|
unlock_shvar(shv);
|
|
LM_ERR("no more pkg mem\n");
|
|
return pv_get_null(msg, param, res);
|
|
}
|
|
}
|
|
strncpy(shv_cpy.s, shv->v.value.s.s, shv->v.value.s.len);
|
|
shv_cpy.len = shv->v.value.s.len;
|
|
|
|
unlock_shvar(shv);
|
|
|
|
res->rs = shv_cpy;
|
|
res->flags = PV_VAL_STR;
|
|
} else {
|
|
res->ri = shv->v.value.n;
|
|
|
|
unlock_shvar(shv);
|
|
|
|
sval = sint2str(res->ri, &len);
|
|
res->rs.s = sval;
|
|
res->rs.len = len;
|
|
res->flags = PV_VAL_STR|PV_VAL_INT|PV_TYPE_INT;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int pv_set_shvar(struct sip_msg* msg, pv_param_t *param,
|
|
int op, pv_value_t *val)
|
|
{
|
|
int_str isv;
|
|
int flags;
|
|
|
|
if(param==NULL)
|
|
{
|
|
LM_ERR("bad parameters\n");
|
|
return -1;
|
|
}
|
|
|
|
if(param->pvn.u.dname==0)
|
|
{
|
|
LM_ERR("error - cannot find shvar\n");
|
|
goto error;
|
|
}
|
|
lock_shvar((sh_var_t*)param->pvn.u.dname);
|
|
if(val == NULL)
|
|
{
|
|
isv.n = 0;
|
|
set_shvar_value((sh_var_t*)param->pvn.u.dname, &isv, 0);
|
|
goto done;
|
|
}
|
|
flags = 0;
|
|
if(val->flags&PV_TYPE_INT)
|
|
{
|
|
isv.n = val->ri;
|
|
} else {
|
|
isv.s = val->rs;
|
|
flags |= VAR_VAL_STR;
|
|
}
|
|
if(set_shvar_value((sh_var_t*)param->pvn.u.dname, &isv, flags)==NULL)
|
|
{
|
|
LM_ERR("error - cannot set shvar [%.*s] \n",
|
|
((sh_var_t*)param->pvn.u.dname)->name.len,
|
|
((sh_var_t*)param->pvn.u.dname)->name.s);
|
|
goto error;
|
|
}
|
|
done:
|
|
unlock_shvar((sh_var_t*)param->pvn.u.dname);
|
|
return 0;
|
|
error:
|
|
unlock_shvar((sh_var_t*)param->pvn.u.dname);
|
|
return -1;
|
|
}
|
|
|
|
struct mi_root* mi_shvar_set(struct mi_root* cmd_tree, void* param)
|
|
{
|
|
str sp;
|
|
str name;
|
|
int ival;
|
|
int_str isv;
|
|
int flags;
|
|
struct mi_node* node;
|
|
sh_var_t *shv = NULL;
|
|
|
|
node = cmd_tree->node.kids;
|
|
if(node == NULL)
|
|
return init_mi_tree( 400, MI_SSTR(MI_MISSING_PARM_S));
|
|
name = node->value;
|
|
if(name.len<=0 || name.s==NULL)
|
|
{
|
|
LM_ERR("bad shv name\n");
|
|
return init_mi_tree( 500, MI_SSTR("bad shv name"));
|
|
}
|
|
shv = get_shvar_by_name(&name);
|
|
if(shv==NULL)
|
|
return init_mi_tree(404, MI_SSTR("Not found"));
|
|
|
|
node = node->next;
|
|
if(node == NULL)
|
|
return init_mi_tree(400, MI_SSTR(MI_MISSING_PARM_S));
|
|
sp = node->value;
|
|
if(sp.s == NULL)
|
|
return init_mi_tree(500, MI_SSTR("type not found"));
|
|
flags = 0;
|
|
if(sp.s[0]=='s' || sp.s[0]=='S')
|
|
flags = VAR_VAL_STR;
|
|
|
|
node= node->next;
|
|
if(node == NULL)
|
|
return init_mi_tree(400, MI_SSTR(MI_MISSING_PARM_S));
|
|
|
|
sp = node->value;
|
|
if(sp.s == NULL)
|
|
{
|
|
return init_mi_tree(500, MI_SSTR("value not found"));
|
|
}
|
|
if(flags == 0)
|
|
{
|
|
if(str2sint(&sp, &ival))
|
|
{
|
|
LM_ERR("bad integer value\n");
|
|
return init_mi_tree( 500, MI_SSTR("bad integer value"));
|
|
}
|
|
isv.n = ival;
|
|
} else {
|
|
isv.s = sp;
|
|
}
|
|
|
|
lock_shvar(shv);
|
|
if(set_shvar_value(shv, &isv, flags)==NULL)
|
|
{
|
|
unlock_shvar(shv);
|
|
LM_ERR("cannot set shv value\n");
|
|
return init_mi_tree( 500, MI_SSTR("cannot set shv value"));
|
|
}
|
|
|
|
unlock_shvar(shv);
|
|
LM_DBG("$shv(%.*s) updated\n", name.len, name.s);
|
|
return init_mi_tree( 200, MI_OK_S, MI_OK_LEN);
|
|
}
|
|
|
|
struct mi_root* mi_shvar_get(struct mi_root* cmd_tree, void* param)
|
|
{
|
|
struct mi_root* rpl_tree = NULL;
|
|
struct mi_node* node;
|
|
struct mi_attr* attr = NULL;
|
|
str name;
|
|
int ival;
|
|
sh_var_t *shv = NULL;
|
|
|
|
node = cmd_tree->node.kids;
|
|
if(node != NULL)
|
|
{
|
|
name = node->value;
|
|
if(name.len<=0 || name.s==NULL)
|
|
{
|
|
LM_ERR("bad shv name\n");
|
|
return init_mi_tree( 500, MI_SSTR("bad shv name"));
|
|
}
|
|
shv = get_shvar_by_name(&name);
|
|
if(shv==NULL)
|
|
return init_mi_tree(404, MI_SSTR("Not found"));
|
|
|
|
rpl_tree = init_mi_tree(200, MI_OK_S, MI_OK_LEN);
|
|
if (rpl_tree==NULL)
|
|
return NULL;
|
|
|
|
node = add_mi_node_child(&rpl_tree->node, MI_DUP_VALUE,
|
|
"VAR",3, name.s, name.len);
|
|
if(node == NULL)
|
|
goto error;
|
|
lock_shvar(shv);
|
|
if(shv->v.flags&VAR_VAL_STR)
|
|
{
|
|
attr = add_mi_attr (node, MI_DUP_VALUE, "type", 4, "string", 6);
|
|
if(attr == 0)
|
|
{
|
|
unlock_shvar(shv);
|
|
goto error;
|
|
}
|
|
attr = add_mi_attr (node, MI_DUP_VALUE, "value", 5,
|
|
shv->v.value.s.s, shv->v.value.s.len);
|
|
if(attr == 0)
|
|
{
|
|
unlock_shvar(shv);
|
|
goto error;
|
|
}
|
|
unlock_shvar(shv);
|
|
} else {
|
|
ival = shv->v.value.n;
|
|
unlock_shvar(shv);
|
|
attr = add_mi_attr (node, MI_DUP_VALUE, "type",4, "integer", 7);
|
|
if(attr == 0)
|
|
goto error;
|
|
name.s = sint2str(ival, &name.len);
|
|
attr = add_mi_attr (node, MI_DUP_VALUE, "value",5,
|
|
name.s, name.len);
|
|
if(attr == 0)
|
|
goto error;
|
|
}
|
|
|
|
goto done;
|
|
}
|
|
|
|
rpl_tree = init_mi_tree(200, MI_OK_S, MI_OK_LEN);
|
|
if (rpl_tree==NULL)
|
|
return NULL;
|
|
|
|
for(shv=sh_vars; shv; shv=shv->next)
|
|
{
|
|
node = add_mi_node_child(&rpl_tree->node, MI_DUP_VALUE,
|
|
"VAR", 3, shv->name.s, shv->name.len);
|
|
if(node == NULL)
|
|
goto error;
|
|
|
|
lock_shvar(shv);
|
|
if(shv->v.flags&VAR_VAL_STR)
|
|
{
|
|
attr = add_mi_attr (node, MI_DUP_VALUE, "type", 4, "string", 6);
|
|
if(attr == 0)
|
|
{
|
|
unlock_shvar(shv);
|
|
goto error;
|
|
}
|
|
attr = add_mi_attr (node, MI_DUP_VALUE, "value", 5,
|
|
shv->v.value.s.s, shv->v.value.s.len);
|
|
if(attr == 0)
|
|
{
|
|
unlock_shvar(shv);
|
|
goto error;
|
|
}
|
|
unlock_shvar(shv);
|
|
} else {
|
|
ival = shv->v.value.n;
|
|
unlock_shvar(shv);
|
|
attr = add_mi_attr (node, MI_DUP_VALUE, "type",4, "integer", 7);
|
|
if(attr == 0)
|
|
goto error;
|
|
name.s = sint2str(ival, &name.len);
|
|
attr = add_mi_attr (node, MI_DUP_VALUE, "value",5,
|
|
name.s, name.len);
|
|
if(attr == 0)
|
|
goto error;
|
|
}
|
|
}
|
|
|
|
done:
|
|
return rpl_tree;
|
|
error:
|
|
if(rpl_tree!=NULL)
|
|
free_mi_tree(rpl_tree);
|
|
return NULL;
|
|
}
|
|
|
|
int param_set_xvar( modparam_t type, void* val, int mode)
|
|
{
|
|
str s;
|
|
char *p;
|
|
int_str isv;
|
|
int flags;
|
|
int ival;
|
|
script_var_t *pkv;
|
|
sh_var_t *shv;
|
|
|
|
if(!shm_initialized()!=0)
|
|
{
|
|
LM_ERR("shm not initialized - cannot set value for PVs\n");
|
|
goto error;
|
|
}
|
|
|
|
s.s = (char*)val;
|
|
if(s.s == NULL || s.s[0] == '\0')
|
|
goto error;
|
|
|
|
p = s.s;
|
|
while(*p && *p!='=') p++;
|
|
|
|
if(*p!='=')
|
|
goto error;
|
|
|
|
s.len = p - s.s;
|
|
if(s.len == 0)
|
|
goto error;
|
|
p++;
|
|
flags = 0;
|
|
if(*p!='s' && *p!='S' && *p!='i' && *p!='I')
|
|
goto error;
|
|
|
|
if(*p=='s' || *p=='S')
|
|
flags = VAR_VAL_STR;
|
|
p++;
|
|
if(*p!=':')
|
|
goto error;
|
|
p++;
|
|
isv.s.s = p;
|
|
isv.s.len = strlen(p);
|
|
if(flags != VAR_VAL_STR) {
|
|
if(str2sint(&isv.s, &ival)<0)
|
|
goto error;
|
|
isv.n = ival;
|
|
}
|
|
if(mode==0) {
|
|
pkv = add_var(&s);
|
|
if(pkv==NULL)
|
|
goto error;
|
|
if(set_var_value(pkv, &isv, flags)==NULL)
|
|
goto error;
|
|
} else {
|
|
shv = add_shvar(&s);
|
|
if(shv==NULL)
|
|
goto error;
|
|
if(set_shvar_value(shv, &isv, flags)==NULL)
|
|
goto error;
|
|
}
|
|
|
|
return 0;
|
|
error:
|
|
LM_ERR("unable to set shv parame [%s]\n", s.s);
|
|
return -1;
|
|
}
|
|
|
|
int param_set_var( modparam_t type, void* val)
|
|
{
|
|
return param_set_xvar(type, val, 0);
|
|
}
|
|
|
|
int param_set_shvar( modparam_t type, void* val)
|
|
{
|
|
return param_set_xvar(type, val, 1);
|
|
}
|
|
|
|
/* vi: set ts=4 sw=4 tw=79:ai:cindent: */
|