1
0
mirror of https://github.com/samba-team/samba.git synced 2025-01-12 09:18:10 +03:00
samba-mirror/source3/lib/util_uuid.c
Andrew Bartlett 9d0783bf21 r1492: Rework our random number generation system.
On systems with /dev/urandom, this avoids a change to secrets.tdb for every fork().

For other systems, we now only re-seed after a fork, and on startup.
No need to do it per-operation.  This removes the 'need_reseed'
parameter from generate_random_buffer().

Andrew Bartlett
(This used to be commit 36741d3cf5)
2007-10-10 10:52:13 -05:00

175 lines
4.1 KiB
C

/*
* Unix SMB/CIFS implementation.
* UUID server routines
* Copyright (C) Theodore Ts'o 1996, 1997,
* Copyright (C) Jim McDonough <jmcd@us.ibm.com> 2002, 2003
*
* This program 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.
*
* This program 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., 675 Mass Ave, Cambridge, MA 02139, USA.
*/
#include "includes.h"
/*
* Offset between 15-Oct-1582 and 1-Jan-70
*/
#define TIME_OFFSET_HIGH 0x01B21DD2
#define TIME_OFFSET_LOW 0x13814000
void smb_uuid_pack(const struct uuid uu, UUID_FLAT *ptr)
{
SIVAL(ptr, 0, uu.time_low);
SSVAL(ptr, 4, uu.time_mid);
SSVAL(ptr, 6, uu.time_hi_and_version);
memcpy(ptr+8, uu.clock_seq, 2);
memcpy(ptr+10, uu.node, 6);
}
void smb_uuid_unpack(const UUID_FLAT in, struct uuid *uu)
{
uu->time_low = IVAL(in.info, 0);
uu->time_mid = SVAL(in.info, 4);
uu->time_hi_and_version = SVAL(in.info, 6);
memcpy(uu->clock_seq, in.info+8, 2);
memcpy(uu->node, in.info+10, 6);
}
struct uuid smb_uuid_unpack_static(const UUID_FLAT in)
{
static struct uuid uu;
smb_uuid_unpack(in, &uu);
return uu;
}
void smb_uuid_generate_random(struct uuid *uu)
{
UUID_FLAT tmp;
generate_random_buffer(tmp.info, sizeof(tmp.info));
smb_uuid_unpack(tmp, uu);
uu->clock_seq[0] = (uu->clock_seq[0] & 0x3F) | 0x80;
uu->time_hi_and_version = (uu->time_hi_and_version & 0x0FFF) | 0x4000;
}
char *smb_uuid_to_string(const struct uuid uu)
{
char *out;
asprintf(&out, "%08x-%04x-%04x-%02x%02x-%02x%02x%02x%02x%02x%02x",
uu.time_low, uu.time_mid, uu.time_hi_and_version,
uu.clock_seq[0], uu.clock_seq[1],
uu.node[0], uu.node[1], uu.node[2],
uu.node[3], uu.node[4], uu.node[5]);
return out;
}
const char *smb_uuid_string_static(const struct uuid uu)
{
static char out[37];
slprintf(out, sizeof(out),
"%08x-%04x-%04x-%02x%02x-%02x%02x%02x%02x%02x%02x",
uu.time_low, uu.time_mid, uu.time_hi_and_version,
uu.clock_seq[0], uu.clock_seq[1],
uu.node[0], uu.node[1], uu.node[2],
uu.node[3], uu.node[4], uu.node[5]);
return out;
}
BOOL smb_string_to_uuid(const char *in, struct uuid* uu)
{
BOOL ret = False;
const char *ptr = in;
char *end = (char *)in;
int i;
if (!in || !uu) goto out;
uu->time_low = strtoul(ptr, &end, 16);
if ((end - ptr) != 8 || *end != '-') goto out;
ptr = (end + 1);
uu->time_mid = strtoul(ptr, &end, 16);
if ((end - ptr) != 4 || *end != '-') goto out;
ptr = (end + 1);
uu->time_hi_and_version = strtoul(ptr, &end, 16);
if ((end - ptr) != 4 || *end != '-') goto out;
ptr = (end + 1);
for (i = 0; i < 2; i++) {
int adj = 0;
if (*ptr >= '0' && *ptr <= '9') {
adj = '0';
} else if (*ptr >= 'a' && *ptr <= 'f') {
adj = 'a';
} else if (*ptr >= 'A' && *ptr <= 'F') {
adj = 'A';
} else {
goto out;
}
uu->clock_seq[i] = (*ptr - adj) << 4;
ptr++;
if (*ptr >= '0' && *ptr <= '9') {
adj = '0';
} else if (*ptr >= 'a' && *ptr <= 'f') {
adj = 'a';
} else if (*ptr >= 'A' && *ptr <= 'F') {
adj = 'A';
} else {
goto out;
}
uu->clock_seq[i] |= (*ptr - adj);
ptr++;
}
if (*ptr != '-') goto out;
ptr++;
for (i = 0; i < 6; i++) {
int adj = 0;
if (*ptr >= '0' && *ptr <= '9') {
adj = '0';
} else if (*ptr >= 'a' && *ptr <= 'f') {
adj = 'a';
} else if (*ptr >= 'A' && *ptr <= 'F') {
adj = 'A';
} else {
goto out;
}
uu->node[i] = (*ptr - adj) << 4;
ptr++;
if (*ptr >= '0' && *ptr <= '9') {
adj = '0';
} else if (*ptr >= 'a' && *ptr <= 'f') {
adj = 'a';
} else if (*ptr >= 'A' && *ptr <= 'F') {
adj = 'A';
} else {
goto out;
}
uu->node[i] |= (*ptr - adj);
ptr++;
}
ret = True;
out:
return ret;
}