2012-04-29 20:52:25 +02:00
|
|
|
//
|
|
|
|
// UUID.cpp
|
|
|
|
//
|
2013-09-03 14:42:50 +02:00
|
|
|
// $Id: //poco/1.4/Foundation/src/UUID.cpp#2 $
|
2012-04-29 20:52:25 +02:00
|
|
|
//
|
|
|
|
// Library: Foundation
|
|
|
|
// Package: UUID
|
|
|
|
// Module: UUID
|
|
|
|
//
|
|
|
|
// Copyright (c) 2004-2006, Applied Informatics Software Engineering GmbH.
|
|
|
|
// and Contributors.
|
|
|
|
//
|
|
|
|
// Permission is hereby granted, free of charge, to any person or organization
|
|
|
|
// obtaining a copy of the software and accompanying documentation covered by
|
|
|
|
// this license (the "Software") to use, reproduce, display, distribute,
|
|
|
|
// execute, and transmit the Software, and to prepare derivative works of the
|
|
|
|
// Software, and to permit third-parties to whom the Software is furnished to
|
|
|
|
// do so, all subject to the following:
|
|
|
|
//
|
|
|
|
// The copyright notices in the Software and this entire statement, including
|
|
|
|
// the above license grant, this restriction and the following disclaimer,
|
|
|
|
// must be included in all copies of the Software, in whole or in part, and
|
|
|
|
// all derivative works of the Software, unless such copies or derivative
|
|
|
|
// works are solely in the form of machine-executable object code generated by
|
|
|
|
// a source language processor.
|
|
|
|
//
|
|
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
// FITNESS FOR A PARTICULAR PURPOSE, TITLE AND NON-INFRINGEMENT. IN NO EVENT
|
|
|
|
// SHALL THE COPYRIGHT HOLDERS OR ANYONE DISTRIBUTING THE SOFTWARE BE LIABLE
|
|
|
|
// FOR ANY DAMAGES OR OTHER LIABILITY, WHETHER IN CONTRACT, TORT OR OTHERWISE,
|
|
|
|
// ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
|
|
|
// DEALINGS IN THE SOFTWARE.
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
|
|
#include "Poco/UUID.h"
|
|
|
|
#include "Poco/ByteOrder.h"
|
|
|
|
#include "Poco/Exception.h"
|
|
|
|
#include <algorithm>
|
|
|
|
#include <cstring>
|
|
|
|
|
|
|
|
|
|
|
|
namespace Poco {
|
|
|
|
|
|
|
|
|
|
|
|
UUID::UUID():
|
|
|
|
_timeLow(0),
|
|
|
|
_timeMid(0),
|
|
|
|
_timeHiAndVersion(0),
|
|
|
|
_clockSeq(0)
|
|
|
|
{
|
|
|
|
std::memset(_node, 0, sizeof(_node));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
UUID::UUID(const UUID& uuid):
|
|
|
|
_timeLow(uuid._timeLow),
|
|
|
|
_timeMid(uuid._timeMid),
|
|
|
|
_timeHiAndVersion(uuid._timeHiAndVersion),
|
|
|
|
_clockSeq(uuid._clockSeq)
|
|
|
|
{
|
|
|
|
std::memcpy(_node, uuid._node, sizeof(_node));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
UUID::UUID(const std::string& uuid)
|
|
|
|
{
|
|
|
|
parse(uuid);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
UUID::UUID(const char* uuid)
|
|
|
|
{
|
|
|
|
poco_check_ptr (uuid);
|
|
|
|
parse(std::string(uuid));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
UUID::UUID(UInt32 timeLow, UInt32 timeMid, UInt32 timeHiAndVersion, UInt16 clockSeq, UInt8 node[]):
|
|
|
|
_timeLow(timeLow),
|
|
|
|
_timeMid(timeMid),
|
|
|
|
_timeHiAndVersion(timeHiAndVersion),
|
|
|
|
_clockSeq(clockSeq)
|
|
|
|
{
|
|
|
|
std::memcpy(_node, node, sizeof(_node));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
UUID::UUID(const char* bytes, Version version)
|
|
|
|
{
|
|
|
|
UInt32 i32;
|
|
|
|
UInt16 i16;
|
|
|
|
std::memcpy(&i32, bytes, sizeof(i32));
|
|
|
|
_timeLow = ByteOrder::fromNetwork(i32);
|
|
|
|
bytes += sizeof(i32);
|
|
|
|
std::memcpy(&i16, bytes, sizeof(i16));
|
|
|
|
_timeMid = ByteOrder::fromNetwork(i16);
|
|
|
|
bytes += sizeof(i16);
|
|
|
|
std::memcpy(&i16, bytes, sizeof(i16));
|
|
|
|
_timeHiAndVersion = ByteOrder::fromNetwork(i16);
|
|
|
|
bytes += sizeof(i16);
|
|
|
|
std::memcpy(&i16, bytes, sizeof(i16));
|
|
|
|
_clockSeq = ByteOrder::fromNetwork(i16);
|
|
|
|
bytes += sizeof(i16);
|
|
|
|
std::memcpy(_node, bytes, sizeof(_node));
|
|
|
|
|
|
|
|
_timeHiAndVersion &= 0x0FFF;
|
|
|
|
_timeHiAndVersion |= (version << 12);
|
|
|
|
_clockSeq &= 0x3FFF;
|
|
|
|
_clockSeq |= 0x8000;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
UUID::~UUID()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
UUID& UUID::operator = (const UUID& uuid)
|
|
|
|
{
|
|
|
|
if (&uuid != this)
|
|
|
|
{
|
|
|
|
_timeLow = uuid._timeLow;
|
|
|
|
_timeMid = uuid._timeMid;
|
|
|
|
_timeHiAndVersion = uuid._timeHiAndVersion;
|
|
|
|
_clockSeq = uuid._clockSeq;
|
|
|
|
std::memcpy(_node, uuid._node, sizeof(_node));
|
|
|
|
}
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void UUID::swap(UUID& uuid)
|
|
|
|
{
|
|
|
|
std::swap(_timeLow, uuid._timeLow);
|
|
|
|
std::swap(_timeMid, uuid._timeMid);
|
|
|
|
std::swap(_timeHiAndVersion, uuid._timeHiAndVersion);
|
|
|
|
std::swap(_clockSeq, uuid._clockSeq);
|
|
|
|
std::swap_ranges(_node, _node + 6, &uuid._node[0]);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void UUID::parse(const std::string& uuid)
|
|
|
|
{
|
|
|
|
if (!tryParse(uuid))
|
|
|
|
throw SyntaxException(uuid);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool UUID::tryParse(const std::string& uuid)
|
|
|
|
{
|
2013-09-03 14:42:50 +02:00
|
|
|
if (uuid.size() < 32)
|
2012-04-29 20:52:25 +02:00
|
|
|
return false;
|
|
|
|
|
2013-09-03 14:42:50 +02:00
|
|
|
bool haveHyphens = false;
|
|
|
|
if (uuid[8] == '-' && uuid[13] == '-' && uuid[18] == '-' && uuid[23] == '-')
|
|
|
|
{
|
|
|
|
if (uuid.size() >= 36)
|
|
|
|
haveHyphens = true;
|
|
|
|
else
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2012-04-29 20:52:25 +02:00
|
|
|
std::string::const_iterator it = uuid.begin();
|
|
|
|
_timeLow = 0;
|
|
|
|
for (int i = 0; i < 8; ++i)
|
|
|
|
{
|
|
|
|
_timeLow = (_timeLow << 4) | nibble(*it++);
|
|
|
|
}
|
2013-09-03 14:42:50 +02:00
|
|
|
if (haveHyphens) ++it;
|
2012-04-29 20:52:25 +02:00
|
|
|
_timeMid = 0;
|
|
|
|
for (int i = 0; i < 4; ++i)
|
|
|
|
{
|
|
|
|
_timeMid = (_timeMid << 4) | nibble(*it++);
|
|
|
|
}
|
2013-09-03 14:42:50 +02:00
|
|
|
if (haveHyphens) ++it;
|
2012-04-29 20:52:25 +02:00
|
|
|
_timeHiAndVersion = 0;
|
|
|
|
for (int i = 0; i < 4; ++i)
|
|
|
|
{
|
|
|
|
_timeHiAndVersion = (_timeHiAndVersion << 4) | nibble(*it++);
|
|
|
|
}
|
2013-09-03 14:42:50 +02:00
|
|
|
if (haveHyphens) ++it;
|
2012-04-29 20:52:25 +02:00
|
|
|
_clockSeq = 0;
|
|
|
|
for (int i = 0; i < 4; ++i)
|
|
|
|
{
|
|
|
|
_clockSeq = (_clockSeq << 4) | nibble(*it++);
|
|
|
|
}
|
2013-09-03 14:42:50 +02:00
|
|
|
if (haveHyphens) ++it;
|
2012-04-29 20:52:25 +02:00
|
|
|
for (int i = 0; i < 6; ++i)
|
|
|
|
{
|
|
|
|
_node[i] = (nibble(*it++) << 4) | nibble(*it++) ;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
std::string UUID::toString() const
|
|
|
|
{
|
|
|
|
std::string result;
|
|
|
|
result.reserve(36);
|
|
|
|
appendHex(result, _timeLow);
|
|
|
|
result += '-';
|
|
|
|
appendHex(result, _timeMid);
|
|
|
|
result += '-';
|
|
|
|
appendHex(result, _timeHiAndVersion);
|
|
|
|
result += '-';
|
|
|
|
appendHex(result, _clockSeq);
|
|
|
|
result += '-';
|
|
|
|
for (int i = 0; i < sizeof(_node); ++i)
|
|
|
|
appendHex(result, _node[i]);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void UUID::copyFrom(const char* buffer)
|
|
|
|
{
|
|
|
|
UInt32 i32;
|
|
|
|
UInt16 i16;
|
|
|
|
std::memcpy(&i32, buffer, sizeof(i32));
|
|
|
|
_timeLow = ByteOrder::fromNetwork(i32);
|
|
|
|
buffer += sizeof(i32);
|
|
|
|
std::memcpy(&i16, buffer, sizeof(i16));
|
|
|
|
_timeMid = ByteOrder::fromNetwork(i16);
|
|
|
|
buffer += sizeof(i16);
|
|
|
|
std::memcpy(&i16, buffer, sizeof(i16));
|
|
|
|
_timeHiAndVersion = ByteOrder::fromNetwork(i16);
|
|
|
|
buffer += sizeof(i16);
|
|
|
|
std::memcpy(&i16, buffer, sizeof(i16));
|
|
|
|
_clockSeq = ByteOrder::fromNetwork(i16);
|
|
|
|
buffer += sizeof(i16);
|
|
|
|
std::memcpy(_node, buffer, sizeof(_node));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void UUID::copyTo(char* buffer) const
|
|
|
|
{
|
|
|
|
UInt32 i32 = ByteOrder::toNetwork(_timeLow);
|
|
|
|
std::memcpy(buffer, &i32, sizeof(i32));
|
|
|
|
buffer += sizeof(i32);
|
|
|
|
UInt16 i16 = ByteOrder::toNetwork(_timeMid);
|
|
|
|
std::memcpy(buffer, &i16, sizeof(i16));
|
|
|
|
buffer += sizeof(i16);
|
|
|
|
i16 = ByteOrder::toNetwork(_timeHiAndVersion);
|
|
|
|
std::memcpy(buffer, &i16, sizeof(i16));
|
|
|
|
buffer += sizeof(i16);
|
|
|
|
i16 = ByteOrder::toNetwork(_clockSeq);
|
|
|
|
std::memcpy(buffer, &i16, sizeof(i16));
|
|
|
|
buffer += sizeof(i16);
|
|
|
|
std::memcpy(buffer, _node, sizeof(_node));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int UUID::variant() const
|
|
|
|
{
|
|
|
|
int v = _clockSeq >> 13;
|
|
|
|
if ((v & 6) == 6)
|
|
|
|
return v;
|
|
|
|
else if (v & 4)
|
|
|
|
return 2;
|
|
|
|
else
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int UUID::compare(const UUID& uuid) const
|
|
|
|
{
|
|
|
|
if (_timeLow != uuid._timeLow) return _timeLow < uuid._timeLow ? -1 : 1;
|
|
|
|
if (_timeMid != uuid._timeMid) return _timeMid < uuid._timeMid ? -1 : 1;
|
|
|
|
if (_timeHiAndVersion != uuid._timeHiAndVersion) return _timeHiAndVersion < uuid._timeHiAndVersion ? -1 : 1;
|
|
|
|
if (_clockSeq != uuid._clockSeq) return _clockSeq < uuid._clockSeq ? -1 : 1;
|
|
|
|
for (int i = 0; i < sizeof(_node); ++i)
|
|
|
|
{
|
|
|
|
if (_node[i] < uuid._node[i])
|
|
|
|
return -1;
|
|
|
|
else if (_node[i] > uuid._node[i])
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void UUID::appendHex(std::string& str, UInt8 n)
|
|
|
|
{
|
|
|
|
static const char* digits = "0123456789abcdef";
|
|
|
|
str += digits[(n >> 4) & 0xF];
|
|
|
|
str += digits[n & 0xF];
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void UUID::appendHex(std::string& str, UInt16 n)
|
|
|
|
{
|
|
|
|
appendHex(str, UInt8(n >> 8));
|
|
|
|
appendHex(str, UInt8(n & 0xFF));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void UUID::appendHex(std::string& str, UInt32 n)
|
|
|
|
{
|
|
|
|
appendHex(str, UInt16(n >> 16));
|
|
|
|
appendHex(str, UInt16(n & 0xFFFF));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
UInt8 UUID::nibble(char hex)
|
|
|
|
{
|
|
|
|
if (hex >= 'a' && hex <= 'f')
|
|
|
|
return UInt8(hex - 'a' + 10);
|
|
|
|
else if (hex >= 'A' && hex <= 'F')
|
|
|
|
return UInt8(hex - 'A' + 10);
|
|
|
|
else if (hex >= '0' && hex <= '9')
|
|
|
|
return UInt8(hex - '0');
|
|
|
|
else
|
|
|
|
return UInt8(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void UUID::fromNetwork()
|
|
|
|
{
|
|
|
|
_timeLow = ByteOrder::fromNetwork(_timeLow);
|
|
|
|
_timeMid = ByteOrder::fromNetwork(_timeMid);
|
|
|
|
_timeHiAndVersion = ByteOrder::fromNetwork(_timeHiAndVersion);
|
|
|
|
_clockSeq = ByteOrder::fromNetwork(_clockSeq);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void UUID::toNetwork()
|
|
|
|
{
|
|
|
|
_timeLow = ByteOrder::toNetwork(_timeLow);
|
|
|
|
_timeMid = ByteOrder::toNetwork(_timeMid);
|
|
|
|
_timeHiAndVersion = ByteOrder::toNetwork(_timeHiAndVersion);
|
|
|
|
_clockSeq = ByteOrder::toNetwork(_clockSeq);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
namespace
|
|
|
|
{
|
|
|
|
static UUID uuidNull;
|
|
|
|
static UUID uuidDNS("6ba7b810-9dad-11d1-80b4-00c04fd430c8");
|
|
|
|
static UUID uuidURI("6ba7b811-9dad-11d1-80b4-00c04fd430c8");
|
|
|
|
static UUID uuidOID("6ba7b812-9dad-11d1-80b4-00c04fd430c8");
|
|
|
|
static UUID uuidX500("6ba7b814-9dad-11d1-80b4-00c04fd430c8");
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const UUID& UUID::null()
|
|
|
|
{
|
|
|
|
return uuidNull;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const UUID& UUID::dns()
|
|
|
|
{
|
|
|
|
return uuidDNS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const UUID& UUID::uri()
|
|
|
|
{
|
|
|
|
return uuidURI;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const UUID& UUID::oid()
|
|
|
|
{
|
|
|
|
return uuidOID;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const UUID& UUID::x500()
|
|
|
|
{
|
|
|
|
return uuidX500;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
} // namespace Poco
|