/* * Copyright (C) 2015 ScyllaDB */ /* * This file is part of Scylla. * * Scylla is free software: you can redistribute it and/or modify * it under the terms of the GNU Affero General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * Scylla 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 Scylla. If not, see . */ #pragma once #include #include #include extern std::atomic clocks_offset; static inline std::chrono::seconds get_clocks_offset() { auto off = clocks_offset.load(std::memory_order_relaxed); return std::chrono::seconds(off); } // FIXME: wraps around in 2038 class gc_clock { public: using base = std::chrono::system_clock; using rep = int32_t; using period = std::ratio<1, 1>; // seconds using duration = std::chrono::duration; using time_point = std::chrono::time_point; static time_point now() { return time_point(std::chrono::duration_cast(base::now().time_since_epoch())) + get_clocks_offset(); } }; // Returns a time point which is earlier from t by d, or minimum time point if it cannot be represented. template inline auto saturating_subtract(std::chrono::time_point t, std::chrono::duration d) -> decltype(t) { return std::max(t, decltype(t)::min() + d) - d; } using expiry_opt = std::experimental::optional; using ttl_opt = std::experimental::optional; // 20 years in seconds static constexpr gc_clock::duration max_ttl = gc_clock::duration{20 * 365 * 24 * 60 * 60}; template static inline void forward_jump_clocks(Duration delta) { auto d = std::chrono::duration_cast(delta).count(); clocks_offset.fetch_add(d, std::memory_order_relaxed); }