123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189 |
- /*
- * $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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 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;
- sid->pid = my_pid();
- 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;
- sid->pid = my_pid();
- 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;
- }
- /**
- *
- */
- int sruid_next_safe(sruid_t *sid)
- {
- if(unlikely(sid->pid!=my_pid())) sruid_reinit(sid, sid->mode);
- return sruid_next(sid);
- }
|