| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160 | #ifndef BOOST_UUID_TIME_GENERATOR_V1_HPP_INCLUDED#define BOOST_UUID_TIME_GENERATOR_V1_HPP_INCLUDED// Copyright 2024 Peter Dimov// Distributed under the Boost Software License, Version 1.0.// https://www.boost.org/LICENSE_1_0.txt#include <boost/uuid/uuid.hpp>#include <boost/uuid/uuid_clock.hpp>#include <boost/uuid/detail/random_provider.hpp>#include <boost/uuid/detail/endian.hpp>#include <boost/config.hpp>#include <boost/config/workaround.hpp>#include <atomic>#include <cstdint>#include <cstring>namespace boost {namespace uuids {// time_generator_v1class time_generator_v1{public:    struct state_type    {        std::uint64_t timestamp;        std::uint16_t clock_seq;// https://gcc.gnu.org/bugzilla/show_bug.cgi?id=114865#if BOOST_WORKAROUND(BOOST_LIBSTDCXX_VERSION, >= 130000)# if BOOST_CXX_VERSION >= 201402L        std::uint16_t padding[ 3 ] = {};# else        std::uint16_t padding[ 3 ];# endif#endif    };private:    uuid::node_type node_ = {};    std::atomic<state_type>* ps_ = nullptr;    state_type state_ = {};public:    using result_type = uuid;    time_generator_v1();    time_generator_v1( uuid::node_type const& node, state_type const& state ) noexcept;    time_generator_v1( uuid::node_type const& node, std::atomic<state_type>& state ) noexcept;    result_type operator()() noexcept;private:    static state_type get_new_state( state_type const& oldst ) noexcept;};// constructorsinline time_generator_v1::time_generator_v1(){    detail::random_provider prov;    // generate a pseudorandom node identifier    std::uint32_t tmp[ 3 ];    prov.generate( tmp, tmp + 3 );    std::memcpy( node_.data(), tmp, node_.size() );    node_[ 0 ] |= 0x01; // mark as multicast    // generate a pseudorandom 14 bit clock sequence    state_.clock_seq = static_cast<std::uint16_t>( tmp[ 2 ] & 0x3FFF );}inline time_generator_v1::time_generator_v1( uuid::node_type const& node, state_type const& state ) noexcept: node_( node ), state_( state ){}inline time_generator_v1::time_generator_v1( uuid::node_type const& node, std::atomic<state_type>& state ) noexcept: node_( node ), ps_( &state ){}// get_new_stateinline time_generator_v1::state_type time_generator_v1::get_new_state( state_type const& oldst ) noexcept{    state_type newst( oldst );    std::uint64_t timestamp = uuid_clock::now().time_since_epoch().count();    if( timestamp <= newst.timestamp )    {        newst.clock_seq = ( newst.clock_seq + 1 ) & 0x3FFF;    }    newst.timestamp = timestamp;    return newst;}// operator()inline time_generator_v1::result_type time_generator_v1::operator()() noexcept{    if( ps_ )    {        auto oldst = ps_->load( std::memory_order_relaxed );        for( ;; )        {            auto newst = get_new_state( oldst );            if( ps_->compare_exchange_strong( oldst, newst, std::memory_order_relaxed, std::memory_order_relaxed ) )            {                state_ = newst;                break;            }        }    }    else    {        state_ = get_new_state( state_ );    }    uuid result;    std::uint32_t time_low = static_cast< std::uint32_t >( state_.timestamp );    detail::store_big_u32( result.data + 0, time_low );    std::uint16_t time_mid = static_cast< std::uint16_t >( state_.timestamp >> 32 );    detail::store_big_u16( result.data + 4, time_mid );    std::uint16_t time_hi_and_version = static_cast< std::uint16_t >( state_.timestamp >> 48 ) | 0x1000;    detail::store_big_u16( result.data + 6, time_hi_and_version );    detail::store_big_u16( result.data + 8, state_.clock_seq | 0x8000 );    std::memcpy( result.data + 10, node_.data(), 6 );    return result;}}} // namespace boost::uuids#endif // BOOST_UUID_TIME_GENERATOR_V1_HPP_INCLUDED
 |