X-Git-Url: http://git.onelab.eu/?a=blobdiff_plain;f=lib%2Fuuid.c;h=df1206e1c80a9af45745c5ef3edabe4f5c6ef003;hb=28c5588e8e1a8d091c5d2275232c35f2968a97fa;hp=a1100f2fdc81fa9b6b10b79229706630cf807009;hpb=e251c8d0050acf8d3f35c91b6d2708fab5314ce7;p=sliver-openvswitch.git diff --git a/lib/uuid.c b/lib/uuid.c index a1100f2fd..df1206e1c 100644 --- a/lib/uuid.c +++ b/lib/uuid.c @@ -1,4 +1,4 @@ -/* Copyright (c) 2008, 2009, 2010 Nicira Networks +/* Copyright (c) 2008, 2009, 2010, 2011, 2013 Nicira, Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -22,10 +22,13 @@ #include #include #include +#include #include "aes128.h" #include "entropy.h" +#include "ovs-thread.h" #include "sha1.h" +#include "timeval.h" #include "util.h" static struct aes128 key; @@ -33,7 +36,6 @@ static uint64_t counter[2]; BUILD_ASSERT_DECL(sizeof counter == 16); static void do_init(void); -static void read_urandom(void *buffer, size_t n); /* * Initialize the UUID module. Aborts the program with an error message if @@ -48,11 +50,8 @@ static void read_urandom(void *buffer, size_t n); void uuid_init(void) { - static bool inited; - if (!inited) { - do_init(); - inited = true; - } + static pthread_once_t once = PTHREAD_ONCE_INIT; + pthread_once(&once, do_init); } /* Generates a new random UUID in 'uuid'. @@ -82,15 +81,22 @@ uuid_init(void) void uuid_generate(struct uuid *uuid) { + static struct ovs_mutex mutex = OVS_MUTEX_INITIALIZER; + uint64_t copy[2]; + uuid_init(); - /* Increment the counter. */ + /* Copy out the counter's current value, then increment it. */ + ovs_mutex_lock(&mutex); + copy[0] = counter[0]; + copy[1] = counter[1]; if (++counter[1] == 0) { counter[0]++; } + ovs_mutex_unlock(&mutex); /* AES output is exactly 16 bytes, so we encrypt directly into 'uuid'. */ - aes128_encrypt(&key, counter, uuid); + aes128_encrypt(&key, copy, uuid); /* Set bits to indicate a random UUID. See RFC 4122 section 4.4. */ uuid->parts[2] &= ~0xc0000000; @@ -156,29 +162,55 @@ uuid_from_string(struct uuid *uuid, const char *s) bool uuid_from_string_prefix(struct uuid *uuid, const char *s) { - static const char template[] = "00000000-1111-1111-2222-222233333333"; - const char *t; + /* 0 1 2 3 */ + /* 012345678901234567890123456789012345 */ + /* ------------------------------------ */ + /* 00000000-1111-1111-2222-222233333333 */ - uuid_zero(uuid); - for (t = template; ; t++, s++) { - if (*t >= '0' && *t <= '3') { - uint32_t *part = &uuid->parts[*t - '0']; - if (!isxdigit(*s)) { - goto error; - } - *part = (*part << 4) + hexit_value(*s); - } else if (*t == 0) { - return true; - } else if (*t != *s) { - goto error; - } + bool ok; + + uuid->parts[0] = hexits_value(s, 8, &ok); + if (!ok || s[8] != '-') { + goto error; + } + + uuid->parts[1] = hexits_value(s + 9, 4, &ok) << 16; + if (!ok || s[13] != '-') { + goto error; + } + + uuid->parts[1] += hexits_value(s + 14, 4, &ok); + if (!ok || s[18] != '-') { + goto error; + } + + uuid->parts[2] = hexits_value(s + 19, 4, &ok) << 16; + if (!ok || s[23] != '-') { + goto error; } + uuid->parts[2] += hexits_value(s + 24, 4, &ok); + if (!ok) { + goto error; + } + + uuid->parts[3] = hexits_value(s + 28, 8, &ok); + if (!ok) { + goto error; + } + return true; + error: uuid_zero(uuid); return false; } +static void +sha1_update_int(struct sha1_ctx *sha1_ctx, uintmax_t x) +{ + sha1_update(sha1_ctx, &x, sizeof x); +} + static void do_init(void) { @@ -186,27 +218,21 @@ do_init(void) struct sha1_ctx sha1_ctx; uint8_t random_seed[16]; struct timeval now; - pid_t pid, ppid; - uid_t uid; - gid_t gid; /* Get seed data. */ get_entropy_or_die(random_seed, sizeof random_seed); - if (gettimeofday(&now, NULL)) { - ovs_fatal(errno, "gettimeofday failed"); - } - pid = getpid(); - ppid = getppid(); - uid = getuid(); - gid = getgid(); + xgettimeofday(&now); /* Convert seed into key. */ sha1_init(&sha1_ctx); sha1_update(&sha1_ctx, random_seed, sizeof random_seed); - sha1_update(&sha1_ctx, &pid, sizeof pid); - sha1_update(&sha1_ctx, &ppid, sizeof ppid); - sha1_update(&sha1_ctx, &uid, sizeof uid); - sha1_update(&sha1_ctx, &gid, sizeof gid); + sha1_update(&sha1_ctx, &now, sizeof now); + sha1_update_int(&sha1_ctx, getpid()); +#ifndef _WIN32 + sha1_update_int(&sha1_ctx, getppid()); + sha1_update_int(&sha1_ctx, getuid()); + sha1_update_int(&sha1_ctx, getgid()); +#endif sha1_final(&sha1_ctx, sha1); /* Generate key. */