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.
180 lines
4.0 KiB
180 lines
4.0 KiB
/*
|
|
* $Id$
|
|
*
|
|
* sruid - unique id generator
|
|
*
|
|
* Copyright (C) 2012 Daniel-Constantin Mierla (asipto.com)
|
|
*
|
|
* 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 <stdio.h>
|
|
#include <unistd.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <time.h>
|
|
|
|
#include "../../dprint.h"
|
|
#include "../../globals.h"
|
|
#include "../../pt.h"
|
|
|
|
#include "sruid.h"
|
|
|
|
/* starting polynomials */
|
|
#define MASK_32 0xb4bcd35c
|
|
#define MASK_31 0x7a5bc2e3
|
|
|
|
unsigned int lfsr32, lfsr31;
|
|
|
|
int shift_lfsr(unsigned int *lfsr, unsigned int mask){
|
|
int feedback;
|
|
feedback = *lfsr & 0x1;
|
|
*lfsr >>=1;
|
|
if (feedback == 1)
|
|
*lfsr ^= mask;
|
|
return *lfsr;
|
|
}
|
|
|
|
void init_lfsr(void){
|
|
lfsr32 = (unsigned int)time(NULL);
|
|
lfsr31 = (unsigned int)getpid();
|
|
}
|
|
|
|
/*
|
|
* returns a 32 bit random integer
|
|
*
|
|
*/
|
|
int get_random(){
|
|
return (shift_lfsr(&lfsr32, MASK_32) ^ shift_lfsr(&lfsr31, MASK_31)) & 0xffffffff;
|
|
}
|
|
|
|
/**
|
|
*
|
|
*/
|
|
int sruid_init(sruid_t *sid, char sep, char *cid, int mode)
|
|
{
|
|
int i;
|
|
|
|
if(sid==NULL)
|
|
return -1;
|
|
memset(sid, 0, sizeof(sruid_t));
|
|
memcpy(sid->buf, "srid", 4);
|
|
if(cid!=NULL)
|
|
{
|
|
for(i=0; i<4 && cid[i]!='\0'; i++)
|
|
sid->buf[i] = cid[i];
|
|
}
|
|
sid->buf[4] = sep;
|
|
|
|
if(server_id!=0)
|
|
i = snprintf(sid->buf+5, SRUID_SIZE - 5 /*so far*/ - 8 /* extra int */,
|
|
"%x%c%x%c%x%c", (unsigned int)server_id, sep,
|
|
(unsigned int)time(NULL), sep, (unsigned int)my_pid(), sep);
|
|
else
|
|
i = snprintf(sid->buf+5, SRUID_SIZE - 5 /*so far*/ - 8 /* extra int */,
|
|
"%x%c%x%c",
|
|
(unsigned int)time(NULL), sep, (unsigned int)my_pid(), sep);
|
|
if(i<=0 || i>SRUID_SIZE-13)
|
|
{
|
|
LM_ERR("could not initialize sruid struct - output len: %d\n", i);
|
|
return -1;
|
|
}
|
|
sid->out = sid->buf + i + 5;
|
|
sid->uid.s = sid->buf;
|
|
sid->mode = (sruid_mode_t)mode;
|
|
LM_DBG("root for sruid is [%.*s] (%u / %d)\n", i+5, sid->uid.s,
|
|
sid->counter, i+5);
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
*
|
|
*/
|
|
int sruid_reinit(sruid_t *sid, int mode)
|
|
{
|
|
int i;
|
|
char sep;
|
|
|
|
if(sid==NULL)
|
|
return -1;
|
|
|
|
sep = sid->buf[4];
|
|
sid->buf[5] = '\0';
|
|
|
|
if(server_id!=0)
|
|
i = snprintf(sid->buf+5, SRUID_SIZE - 5 /*so far*/ - 8 /* extra int */,
|
|
"%x%c%x%c%x%c", (unsigned int)server_id, sep,
|
|
(unsigned int)time(NULL), sep, (unsigned int)my_pid(), sep);
|
|
else
|
|
i = snprintf(sid->buf+5, SRUID_SIZE - 5 /*so far*/ - 8 /* extra int */,
|
|
"%x%c%x%c",
|
|
(unsigned int)time(NULL), sep, (unsigned int)my_pid(), sep);
|
|
if(i<=0 || i>SRUID_SIZE-13)
|
|
{
|
|
LM_ERR("could not re-initialize sruid struct - output len: %d\n", i);
|
|
return -1;
|
|
}
|
|
sid->out = sid->buf + i + 5;
|
|
sid->uid.s = sid->buf;
|
|
sid->mode = (sruid_mode_t)mode;
|
|
LM_DBG("re-init root for sruid is [%.*s] (%u / %d)\n", i+5, sid->uid.s,
|
|
sid->counter, i+5);
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
*
|
|
*/
|
|
int sruid_next(sruid_t *sid)
|
|
{
|
|
unsigned short digit;
|
|
int i;
|
|
unsigned int val;
|
|
|
|
if(sid==NULL)
|
|
return -1;
|
|
|
|
sid->counter++;
|
|
if(sid->counter==0) {
|
|
if(sid->mode == SRUID_INC)
|
|
{
|
|
/* counter overflow - re-init to have new timestamp */
|
|
if(sruid_reinit(sid, SRUID_INC)<0)
|
|
return -1;
|
|
}
|
|
sid->counter=1;
|
|
}
|
|
|
|
if(sid->mode == SRUID_LFSR)
|
|
val = get_random();
|
|
else
|
|
val = sid->counter;
|
|
i = 0;
|
|
while(val!=0)
|
|
{
|
|
digit = val & 0x0f;
|
|
sid->out[i++] = (digit >= 10) ? digit + 'a' - 10 : digit + '0';
|
|
val >>= 4;
|
|
}
|
|
sid->out[i] = '\0';
|
|
sid->uid.len = sid->out + i - sid->buf;
|
|
LM_DBG("new sruid is [%.*s] (%u / %d)\n", sid->uid.len, sid->uid.s,
|
|
sid->counter, sid->uid.len);
|
|
return 0;
|
|
}
|
|
|