123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152 |
- #ifndef BOOST_UUID_TIME_GENERATOR_V7_HPP_INCLUDED
- #define BOOST_UUID_TIME_GENERATOR_V7_HPP_INCLUDED
- #include <boost/uuid/uuid.hpp>
- #include <boost/uuid/detail/chacha20.hpp>
- #include <boost/uuid/detail/random_provider.hpp>
- #include <boost/uuid/detail/endian.hpp>
- #include <random>
- #include <cstdint>
- #include <cstring>
- namespace boost {
- namespace uuids {
- class time_generator_v7
- {
- private:
-
-
-
-
- using state_type = std::uint64_t;
- state_type state_ = {};
- detail::chacha20_12 rng_;
- public:
- using result_type = uuid;
- time_generator_v7();
- time_generator_v7( time_generator_v7 const& rhs );
- time_generator_v7( time_generator_v7&& rhs ) noexcept;
- time_generator_v7& operator=( time_generator_v7 const& rhs ) noexcept;
- time_generator_v7& operator=( time_generator_v7&& rhs ) noexcept;
- result_type operator()() noexcept;
- private:
- static state_type get_new_state( state_type const& oldst ) noexcept;
- };
- inline time_generator_v7::time_generator_v7()
- {
- detail::random_provider seeder;
- rng_.seed( seeder );
- }
- inline time_generator_v7::time_generator_v7( time_generator_v7 const& rhs ): state_( rhs.state_ )
- {
- detail::random_provider seeder;
- rng_.seed( seeder );
- }
- inline time_generator_v7::time_generator_v7( time_generator_v7&& rhs ) noexcept: state_( std::move( rhs.state_ ) ), rng_( std::move( rhs.rng_ ) )
- {
- rhs.rng_.perturb();
- }
- inline time_generator_v7& time_generator_v7::operator=( time_generator_v7 const& rhs ) noexcept
- {
- state_ = rhs.state_;
- return *this;
- }
- inline time_generator_v7& time_generator_v7::operator=( time_generator_v7&& rhs ) noexcept
- {
- state_ = std::move( rhs.state_ );
- rng_ = std::move( rhs.rng_ );
- rhs.rng_.perturb();
- return *this;
- }
- inline time_generator_v7::state_type time_generator_v7::get_new_state( state_type const& oldst ) noexcept
- {
-
- std::uint64_t now_in_us = std::chrono::time_point_cast< std::chrono::microseconds >( std::chrono::system_clock::now() ).time_since_epoch().count();
- std::uint64_t time_ms = now_in_us / 1000;
- std::uint64_t time_us = now_in_us % 1000;
- std::uint64_t newst = ( time_ms << 16 ) | ( time_us << 6 );
- // if the time has advanced, reset counter to zero
- if( newst > oldst )
- {
- return newst;
- }
-
- if( time_ms < ( oldst >> 16 ) )
- {
- return newst;
- }
-
- return oldst + 1;
- }
- inline time_generator_v7::result_type time_generator_v7::operator()() noexcept
- {
- uuid result;
-
- std::uniform_int_distribution<std::uint32_t> dist;
- detail::store_native_u32( result.data + 8, dist( rng_ ) );
- detail::store_native_u32( result.data + 12, dist( rng_ ) );
-
- state_ = get_new_state( state_ );
- std::uint64_t time_ms = state_ >> 16;
- std::uint64_t time_us = ( state_ & 0xFFFF ) >> 6;
- std::uint64_t timestamp = ( time_ms << 16 ) | 0x7000 | time_us;
- detail::store_big_u64( result.data + 0, timestamp );
- // set variant and counter
- result.data[ 8 ] = static_cast< std::uint8_t >( 0x80 | ( state_ & 0x3F ) );
- return result;
- }
- }}
- #endif
|