2021-09-17 11:53:00 +02:00
|
|
|
/*
|
|
|
|
|
* 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
/** \file
|
|
|
|
|
* \ingroup bli
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include "BLI_uuid.h"
|
|
|
|
|
|
2021-09-17 14:51:34 +02:00
|
|
|
#include <cstdio>
|
|
|
|
|
#include <cstring>
|
2021-09-17 11:53:00 +02:00
|
|
|
#include <random>
|
|
|
|
|
|
|
|
|
|
/* Ensure the UUID struct doesn't have any padding, to be compatible with memcmp(). */
|
|
|
|
|
static_assert(sizeof(UUID) == 16, "expect UUIDs to be 128 bit exactly");
|
|
|
|
|
|
|
|
|
|
UUID BLI_uuid_generate_random()
|
|
|
|
|
{
|
|
|
|
|
static std::mt19937_64 rng = []() {
|
|
|
|
|
std::mt19937_64 rng;
|
|
|
|
|
|
|
|
|
|
/* Ensure the RNG really can output 64-bit values. */
|
2021-09-17 14:51:34 +02:00
|
|
|
static_assert(std::mt19937_64::min() == 0LL);
|
|
|
|
|
static_assert(std::mt19937_64::max() == 0xffffffffffffffffLL);
|
2021-09-17 11:53:00 +02:00
|
|
|
|
|
|
|
|
struct timespec ts;
|
|
|
|
|
timespec_get(&ts, TIME_UTC);
|
|
|
|
|
rng.seed(ts.tv_nsec);
|
|
|
|
|
|
|
|
|
|
return rng;
|
|
|
|
|
}();
|
|
|
|
|
|
|
|
|
|
UUID uuid;
|
|
|
|
|
|
|
|
|
|
/* RFC4122 suggests setting certain bits to a fixed value, and then randomizing the remaining
|
|
|
|
|
* bits. The opposite is easier to implement, though, so that's what's done here. */
|
|
|
|
|
|
|
|
|
|
/* Read two 64-bit numbers to randomize all 128 bits of the UUID. */
|
|
|
|
|
uint64_t *uuid_as_int64 = reinterpret_cast<uint64_t *>(&uuid);
|
|
|
|
|
uuid_as_int64[0] = rng();
|
|
|
|
|
uuid_as_int64[1] = rng();
|
|
|
|
|
|
|
|
|
|
/* Set the most significant four bits to 0b0100 to indicate version 4 (random UUID). */
|
|
|
|
|
uuid.time_hi_and_version &= ~0xF000;
|
|
|
|
|
uuid.time_hi_and_version |= 0x4000;
|
|
|
|
|
|
|
|
|
|
/* Set the most significant two bits to 0b10 to indicate compatibility with RFC4122. */
|
|
|
|
|
uuid.clock_seq_hi_and_reserved &= ~0x40;
|
|
|
|
|
uuid.clock_seq_hi_and_reserved |= 0x80;
|
|
|
|
|
|
|
|
|
|
return uuid;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool BLI_uuid_equal(const UUID uuid1, const UUID uuid2)
|
|
|
|
|
{
|
2021-09-17 14:51:34 +02:00
|
|
|
return std::memcmp(&uuid1, &uuid2, sizeof(uuid1)) == 0;
|
2021-09-17 11:53:00 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void BLI_uuid_format(char *buffer, const UUID uuid)
|
|
|
|
|
{
|
2021-09-17 14:51:34 +02:00
|
|
|
std::sprintf(buffer,
|
|
|
|
|
"%08x-%04x-%04x-%02x%02x-%02x%02x%02x%02x%02x%02x",
|
|
|
|
|
uuid.time_low,
|
|
|
|
|
uuid.time_mid,
|
|
|
|
|
uuid.time_hi_and_version,
|
|
|
|
|
uuid.clock_seq_hi_and_reserved,
|
|
|
|
|
uuid.clock_seq_low,
|
|
|
|
|
uuid.node[0],
|
|
|
|
|
uuid.node[1],
|
|
|
|
|
uuid.node[2],
|
|
|
|
|
uuid.node[3],
|
|
|
|
|
uuid.node[4],
|
|
|
|
|
uuid.node[5]);
|
2021-09-17 11:53:00 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool BLI_uuid_parse_string(UUID *uuid, const char *buffer)
|
|
|
|
|
{
|
2021-09-17 14:51:34 +02:00
|
|
|
const int num_fields_parsed = std::sscanf(
|
|
|
|
|
buffer,
|
|
|
|
|
"%8x-%4hx-%4hx-%2hhx%2hhx-%2hhx%2hhx%2hhx%2hhx%2hhx%2hhx",
|
|
|
|
|
&uuid->time_low,
|
|
|
|
|
&uuid->time_mid,
|
|
|
|
|
&uuid->time_hi_and_version,
|
|
|
|
|
&uuid->clock_seq_hi_and_reserved,
|
|
|
|
|
&uuid->clock_seq_low,
|
|
|
|
|
&uuid->node[0],
|
|
|
|
|
&uuid->node[1],
|
|
|
|
|
&uuid->node[2],
|
|
|
|
|
&uuid->node[3],
|
|
|
|
|
&uuid->node[4],
|
|
|
|
|
&uuid->node[5]);
|
2021-09-17 11:53:00 +02:00
|
|
|
return num_fields_parsed == 11;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
std::ostream &operator<<(std::ostream &stream, UUID uuid)
|
|
|
|
|
{
|
|
|
|
|
std::string buffer(36, '\0');
|
|
|
|
|
BLI_uuid_format(buffer.data(), uuid);
|
|
|
|
|
stream << buffer;
|
|
|
|
|
return stream;
|
|
|
|
|
}
|