blob: 100243578b4b30f329bf64c9b02d5ef38237a06b [file] [log] [blame]
/*
* Copyright (C) 2010 Google Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met:
*
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above
* copyright notice, this list of conditions and the following disclaimer
* in the documentation and/or other materials provided with the
* distribution.
* * Neither the name of Google Inc. nor the names of its
* contributors may be used to endorse or promote products derived from
* this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "config.h"
#include <wtf/UUID.h>
#include <mutex>
#include <wtf/ASCIICType.h>
#include <wtf/CryptographicallyRandomNumber.h>
#include <wtf/HexNumber.h>
#include <wtf/Lock.h>
#include <wtf/WeakRandom.h>
#include <wtf/text/StringToIntegerConversion.h>
#if OS(DARWIN)
#include <sys/sysctl.h>
#endif
namespace WTF {
static ALWAYS_INLINE UInt128 convertRandomUInt128ToUUIDVersion4(UInt128 buffer)
{
// By default, we generate a v4 UUID value, as per https://datatracker.ietf.org/doc/html/rfc4122#section-4.4.
auto high = static_cast<uint64_t>((buffer >> 64) & 0xffffffffffff0fff) | 0x4000;
auto low = static_cast<uint64_t>(buffer & 0x3fffffffffffffff) | 0x8000000000000000;
return (static_cast<UInt128>(high) << 64) | low;
}
static UInt128 generateCryptographicallyRandomUUIDVersion4()
{
UInt128 buffer { };
static_assert(sizeof(buffer) == 16);
cryptographicallyRandomValues(reinterpret_cast<unsigned char*>(&buffer), 16);
return convertRandomUInt128ToUUIDVersion4(buffer);
}
UInt128 UUID::generateWeakRandomUUIDVersion4()
{
static Lock lock;
UInt128 buffer { 0 };
{
Locker locker { lock };
static std::optional<WeakRandom> weakRandom;
if (!weakRandom)
weakRandom.emplace();
buffer = static_cast<UInt128>(weakRandom->getUint64()) << 64 | weakRandom->getUint64();
}
return convertRandomUInt128ToUUIDVersion4(buffer);
}
UUID::UUID()
: m_data(generateCryptographicallyRandomUUIDVersion4())
{
}
String UUID::toString() const
{
return makeString(*this);
}
std::optional<UUID> UUID::parse(StringView value)
{
// UUIDs have the form xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx with hexadecimal digits for x.
if (value.length() != 36)
return { };
if (value[8] != '-' || value[13] != '-' || value[18] != '-' || value[23] != '-')
return { };
// parseInteger may accept integers starting with +, let's check this beforehand.
if (value[0] == '+' || value[9] == '+' || value[19] == '+' || value[24] == '+')
return { };
auto firstValue = parseInteger<uint64_t>(value.left(8), 16);
if (!firstValue)
return { };
auto secondValue = parseInteger<uint64_t>(value.substring(9, 4), 16);
if (!secondValue)
return { };
auto thirdValue = parseInteger<uint64_t>(value.substring(14, 4), 16);
if (!thirdValue)
return { };
auto fourthValue = parseInteger<uint64_t>(value.substring(19, 4), 16);
if (!fourthValue)
return { };
auto fifthValue = parseInteger<uint64_t>(value.substring(24, 12), 16);
if (!fifthValue)
return { };
uint64_t high = (*firstValue << 32) | (*secondValue << 16) | *thirdValue;
uint64_t low = (*fourthValue << 48) | *fifthValue;
auto result = (static_cast<UInt128>(high) << 64) | low;
if (result == deletedValue)
return { };
return UUID(result);
}
std::optional<UUID> UUID::parseVersion4(StringView value)
{
auto uuid = parse(value);
if (!uuid)
return { };
// Version 4 UUIDs have the form xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx with hexadecimal digits for x and one of 8, 9, A, or B for y.
auto high = static_cast<uint64_t>(uuid->m_data >> 64);
if ((high & 0xf000) != 0x4000)
return { };
auto low = static_cast<uint64_t>(uuid->m_data & 0xffffffffffffffff);
if ((low >> 62) != 2)
return { };
return uuid;
}
String createVersion4UUIDString()
{
return makeString(UUID::createVersion4());
}
String bootSessionUUIDString()
{
#if OS(DARWIN)
static LazyNeverDestroyed<String> bootSessionUUID;
static std::once_flag onceKey;
std::call_once(onceKey, [] {
constexpr size_t maxUUIDLength = 37;
char uuid[maxUUIDLength];
size_t uuidLength = maxUUIDLength;
if (sysctlbyname("kern.bootsessionuuid", uuid, &uuidLength, nullptr, 0))
return;
bootSessionUUID.construct(static_cast<const char*>(uuid), uuidLength - 1);
});
return bootSessionUUID;
#else
return String();
#endif
}
bool isVersion4UUID(StringView value)
{
return !!UUID::parseVersion4(value);
}
} // namespace WTF