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.
204 lines
5.2 KiB
204 lines
5.2 KiB
/*
|
|
* Header file for hash table functions
|
|
*
|
|
* Copyright (C) 2008-2009 Juha Heinanen
|
|
*
|
|
* This file is part of SIP Router, a free SIP server.
|
|
*
|
|
* SIP Router 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
|
|
*
|
|
* SIP Router 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
|
|
*/
|
|
|
|
/*!
|
|
* \file
|
|
* \brief SIP Router LCR :: Header file for hash table functions
|
|
* \ingroup lcr
|
|
* Module: \ref lcr
|
|
*/
|
|
|
|
#include "../../mem/shm_mem.h"
|
|
#include "../../lib/kcore/hash_func.h"
|
|
#include "lcr_mod.h"
|
|
|
|
#define rule_hash(_s) core_hash(_s, 0, lcr_rule_hash_size_param)
|
|
|
|
/* Add lcr entry into hash table */
|
|
int rule_hash_table_insert(struct rule_info **hash_table,
|
|
unsigned int lcr_id, unsigned int rule_id,
|
|
unsigned short prefix_len, char *prefix,
|
|
unsigned short from_uri_len, char *from_uri,
|
|
pcre *from_uri_re, unsigned short request_uri_len,
|
|
char *request_uri, pcre *request_uri_re,
|
|
unsigned short stopper)
|
|
{
|
|
struct rule_info *rule;
|
|
str prefix_str;
|
|
unsigned int hash_val;
|
|
|
|
rule = (struct rule_info *)shm_malloc(sizeof(struct rule_info));
|
|
if (rule == NULL) {
|
|
LM_ERR("Cannot allocate memory for rule hash table entry\n");
|
|
if (from_uri_re) shm_free(from_uri_re);
|
|
if (request_uri_re) shm_free(request_uri_re);
|
|
return 0;
|
|
}
|
|
memset(rule, 0, sizeof(struct rule_info));
|
|
|
|
rule->rule_id = rule_id;
|
|
rule->prefix_len = prefix_len;
|
|
if (prefix_len) {
|
|
memcpy(rule->prefix, prefix, prefix_len);
|
|
}
|
|
rule->from_uri_len = from_uri_len;
|
|
if (from_uri_len) {
|
|
memcpy(rule->from_uri, from_uri, from_uri_len);
|
|
(rule->from_uri)[from_uri_len] = '\0';
|
|
rule->from_uri_re = from_uri_re;
|
|
}
|
|
rule->request_uri_len = request_uri_len;
|
|
if (request_uri_len) {
|
|
memcpy(rule->request_uri, request_uri, request_uri_len);
|
|
(rule->request_uri)[request_uri_len] = '\0';
|
|
rule->request_uri_re = request_uri_re;
|
|
}
|
|
rule->stopper = stopper;
|
|
rule->targets = (struct target *)NULL;
|
|
|
|
prefix_str.len = rule->prefix_len;
|
|
prefix_str.s = rule->prefix;
|
|
|
|
hash_val = rule_hash(&prefix_str);
|
|
rule->next = hash_table[hash_val];
|
|
hash_table[hash_val] = rule;
|
|
|
|
LM_DBG("inserted rule <%u>, prefix <%.*s>, from_uri <%.*s>, request_uri <%.*s>, stopper <%u>, "
|
|
"into index <%u>\n",
|
|
rule_id, prefix_len, prefix, from_uri_len, from_uri, request_uri_len, request_uri, stopper,
|
|
hash_val);
|
|
|
|
return 1;
|
|
}
|
|
|
|
|
|
/* Find gw table index with gw_id */
|
|
int get_gw_index(struct gw_info *gws, unsigned int gw_id,
|
|
unsigned short *gw_index)
|
|
{
|
|
unsigned short gw_count, i;
|
|
|
|
gw_count = gws[0].ip_addr.u.addr32[0];
|
|
|
|
for (i = 1; i <= gw_count; i++) {
|
|
if (gws[i].gw_id == gw_id) {
|
|
*gw_index = i;
|
|
return 1;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* Insert target into hash table rule */
|
|
int rule_hash_table_insert_target(struct rule_info **hash_table,
|
|
struct gw_info *gws,
|
|
unsigned int rule_id, unsigned int gw_id,
|
|
unsigned int priority, unsigned int weight)
|
|
{
|
|
unsigned int i;
|
|
unsigned short gw_index;
|
|
struct rule_info *r;
|
|
struct target *target;
|
|
|
|
target = (struct target *)shm_malloc(sizeof(struct target));
|
|
if (target == NULL) {
|
|
LM_ERR("cannot allocate memory for rule target\n");
|
|
return 0;
|
|
}
|
|
|
|
if (get_gw_index(gws, gw_id, &gw_index) == 0) {
|
|
LM_DBG("could not find (disabled) gw with id <%u>\n", gw_id);
|
|
shm_free(target);
|
|
return 2;
|
|
}
|
|
|
|
target->gw_index = gw_index;
|
|
target->priority = priority;
|
|
target->weight = weight;
|
|
|
|
for (i = 0; i < lcr_rule_hash_size_param; i++) {
|
|
r = hash_table[i];
|
|
while (r) {
|
|
if (r->rule_id == rule_id) {
|
|
target->next = r->targets;
|
|
r->targets = target;
|
|
return 1;
|
|
}
|
|
r = r->next;
|
|
}
|
|
}
|
|
|
|
LM_DBG("could not find (disabled) rule with id <%u>\n", rule_id);
|
|
shm_free(target);
|
|
return 2;
|
|
}
|
|
|
|
|
|
/*
|
|
* Return pointer to lcr hash table entry to which given prefix hashes to.
|
|
*/
|
|
struct rule_info *rule_hash_table_lookup(struct rule_info **hash_table,
|
|
unsigned short prefix_len,
|
|
char *prefix)
|
|
{
|
|
str prefix_str;
|
|
|
|
prefix_str.len = prefix_len;
|
|
prefix_str.s = prefix;
|
|
|
|
return (hash_table)[rule_hash(&prefix_str)];
|
|
}
|
|
|
|
|
|
/* Free contents of lcr hash table */
|
|
void rule_hash_table_contents_free(struct rule_info **hash_table)
|
|
{
|
|
int i;
|
|
struct rule_info *r, *next_r;
|
|
struct target *t, *next_t;
|
|
|
|
if (hash_table == 0)
|
|
return;
|
|
|
|
for (i = 0; i <= lcr_rule_hash_size_param; i++) {
|
|
r = hash_table[i];
|
|
while (r) {
|
|
if (r->from_uri_re) {
|
|
shm_free(r->from_uri_re);
|
|
}
|
|
if (r->request_uri_re)
|
|
shm_free(r->request_uri_re);
|
|
t = r->targets;
|
|
while (t) {
|
|
next_t = t->next;
|
|
shm_free(t);
|
|
t = next_t;
|
|
}
|
|
next_r = r->next;
|
|
shm_free(r);
|
|
r = next_r;
|
|
}
|
|
hash_table[i] = NULL;
|
|
}
|
|
}
|