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.
kamailio/lib/srutils/sruid.c

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;
}