// -*- mode: C++; c-file-style: "cc-mode" -*- //************************************************************************* // // Code available from: https://verilator.org // // Copyright 2003-2023 by Wilson Snyder. This program is free software; you can // redistribute it and/or modify it under the terms of either the GNU // Lesser General Public License Version 3 or the Perl Artistic License // Version 2.0. // SPDX-License-Identifier: LGPL-3.0-only OR Artistic-2.0 // //************************************************************************* /// /// \file /// \brief Verilated/Verilator common header for OS portability /// /// This header is included by user wrappers and defines the Verilated /// public-facing API. /// /// User wrapper code does not generally need to include this, instead /// include verilated.h. /// /// This header is used by both the Verilator source code (run on the build /// and host system), and the Verilated output (run on the target system). /// /// Configuration code needed by only the host system is in /// config_build.h.in, code needed by Verilated code only is in /// verilated.h, and code needed by both is here (verilatedos.h). /// //************************************************************************* #ifndef VERILATOR_VERILATEDOS_H_ #define VERILATOR_VERILATEDOS_H_ // Current clang-format versions botch #ifdef inclusion, so // clang-format off //========================================================================= // Compiler pragma abstraction #if defined(__clang__) # define VL_CLANG_ATTR(attr) __attribute__(( attr )) #else # define VL_CLANG_ATTR(attr) #endif #ifdef __GNUC__ # define VL_ATTR_ALWINLINE __attribute__((always_inline)) inline # define VL_ATTR_NOINLINE __attribute__((noinline)) # define VL_ATTR_COLD __attribute__((cold)) # define VL_ATTR_HOT __attribute__((hot)) # define VL_ATTR_NORETURN __attribute__((noreturn)) // clang and gcc-8.0+ support no_sanitize("string") style attribute # if defined(__clang__) || (__GNUC__ >= 8) # define VL_ATTR_NO_SANITIZE_ALIGN __attribute__((no_sanitize("alignment"))) #else // The entire undefined sanitizer has to be disabled for older gcc # define VL_ATTR_NO_SANITIZE_ALIGN __attribute__((no_sanitize_undefined)) #endif # define VL_ATTR_PRINTF(fmtArgNum) __attribute__((format(printf, (fmtArgNum), (fmtArgNum) + 1))) # define VL_ATTR_PURE __attribute__((pure)) # define VL_ATTR_UNUSED __attribute__((unused)) # if !defined(_WIN32) && !defined(__MINGW32__) // All VL_ATTR_WEAK symbols must be marked with the macOS -U linker flag in verilated.mk.in # define VL_ATTR_WEAK __attribute__((weak)) # endif # define VL_LIKELY(x) __builtin_expect(!!(x), 1) // Prefer over C++20 [[likely]] # define VL_UNLIKELY(x) __builtin_expect(!!(x), 0) // Prefer over C++20 [[unlikely]] # define VL_UNREACHABLE __builtin_unreachable() // C++23 std::unreachable() # define VL_PREFETCH_RD(p) __builtin_prefetch((p), 0) # define VL_PREFETCH_RW(p) __builtin_prefetch((p), 1) #endif // Function acquires a capability/lock (-fthread-safety) #define VL_ACQUIRE(...) \ VL_CLANG_ATTR(annotate("ACQUIRE")) \ VL_CLANG_ATTR(acquire_capability(__VA_ARGS__)) // Function acquires a shared capability/lock (-fthread-safety) #define VL_ACQUIRE_SHARED(...) \ VL_CLANG_ATTR(annotate("ACQUIRE_SHARED")) \ VL_CLANG_ATTR(acquire_shared_capability(__VA_ARGS__)) // Function releases a capability/lock (-fthread-safety) #define VL_RELEASE(...) \ VL_CLANG_ATTR(annotate("RELEASE")) \ VL_CLANG_ATTR(release_capability(__VA_ARGS__)) // Function releases a shared capability/lock (-fthread-safety) #define VL_RELEASE_SHARED(...) \ VL_CLANG_ATTR(annotate("RELEASE_SHARED")) \ VL_CLANG_ATTR(release_shared_capability(__VA_ARGS__)) // Function returns bool if acquired a capability (-fthread-safety) #define VL_TRY_ACQUIRE(...) \ VL_CLANG_ATTR(try_acquire_capability(__VA_ARGS__)) // Function returns bool if acquired shared (-fthread-safety) #define VL_TRY_ACQUIRE_SHARED(...) \ VL_CLANG_ATTR(try_acquire_shared_capability(__VA_ARGS__)) // Function requires a capability inbound (-fthread-safety) #define VL_CAPABILITY(x) \ VL_CLANG_ATTR(capability(x)) // Function requires not having a capability inbound (-fthread-safety) #define VL_REQUIRES(x) \ VL_CLANG_ATTR(annotate("REQUIRES")) \ VL_CLANG_ATTR(requires_capability(x)) // Name of capability/lock (-fthread-safety) #define VL_GUARDED_BY(x) \ VL_CLANG_ATTR(annotate("GUARDED_BY")) \ VL_CLANG_ATTR(guarded_by(x)) // The data that the annotated pointer points to is protected by the given capability. // The pointer itself is not protected. // Allowed on: pointer data member. (-fthread-safety) #define VL_PT_GUARDED_BY(x) \ VL_CLANG_ATTR(annotate("PT_GUARDED_BY")) \ VL_CLANG_ATTR(pt_guarded_by(x)) // Name of mutex protecting this variable (-fthread-safety) #define VL_EXCLUDES(x) \ VL_CLANG_ATTR(annotate("EXCLUDES")) \ VL_CLANG_ATTR(locks_excluded(x)) // Scoped threaded capability/lock (-fthread-safety) #define VL_SCOPED_CAPABILITY \ VL_CLANG_ATTR(scoped_lockable) // Annotated function returns reference to the given capability. // Allowed on: function, method. (-fthread-safety) #define VL_RETURN_CAPABILITY(x) \ VL_CLANG_ATTR(lock_returned(x)) // Assert that capability is already held. // Allowed on: function, method. (-fthread-safety) #define VL_ASSERT_CAPABILITY(x) \ VL_CLANG_ATTR(assert_capability(x)) // Defaults for unsupported compiler features #ifndef VL_ATTR_ALWINLINE # define VL_ATTR_ALWINLINE ///< Attribute to inline, even when not optimizing #endif #ifndef VL_ATTR_NOINLINE # define VL_ATTR_NOINLINE ///< Attribute to never inline, even when optimizing #endif #ifndef VL_ATTR_COLD # define VL_ATTR_COLD ///< Attribute that function is rarely executed #endif #ifndef VL_ATTR_HOT # define VL_ATTR_HOT ///< Attribute that function is highly executed #endif #ifndef VL_ATTR_NORETURN # define VL_ATTR_NORETURN ///< Attribute that function does not ever return #endif #ifndef VL_ATTR_NO_SANITIZE_ALIGN # define VL_ATTR_NO_SANITIZE_ALIGN ///< Attribute that function contains intended unaligned access #endif #ifndef VL_ATTR_PRINTF # define VL_ATTR_PRINTF(fmtArgNum) ///< Attribute for function with printf format checking #endif #ifndef VL_ATTR_PURE # define VL_ATTR_PURE ///< Attribute that function is pure (and thus also VL_MT_SAFE) #endif #ifndef VL_ATTR_UNUSED # define VL_ATTR_UNUSED ///< Attribute that function that may be never used #endif #ifndef VL_ATTR_WEAK # define VL_ATTR_WEAK ///< Attribute that function external that is optionally defined #endif #ifndef VL_LIKELY # define VL_LIKELY(x) (!!(x)) ///< Return boolean expression that is more often true # define VL_UNLIKELY(x) (!!(x)) ///< Return boolean expression that is more often false #endif /// Boolean expression never hit by users (branch coverage disabled) # define VL_UNCOVERABLE(x) VL_UNLIKELY(x) #ifndef VL_UNREACHABLE # define VL_UNREACHABLE ///< Statement that may never be reached (for coverage etc) #endif #ifndef VL_PREFETCH_RD # define VL_PREFETCH_RD(p) ///< Prefetch pointer argument with read intent #endif #ifndef VL_PREFETCH_RW # define VL_PREFETCH_RW(p) ///< Prefetch pointer argument with read/write intent #endif #ifndef VL_NO_LEGACY # define VL_ATTR_ALIGNED(alignment) // Deprecated # define VL_FUNC __func__ // Deprecated # define VL_THREAD // Deprecated # define VL_THREAD_LOCAL thread_local // Deprecated # define VL_STATIC_OR_THREAD static // Deprecated #endif // Comment tag that Function is pure (and thus also VL_MT_SAFE) #define VL_PURE VL_CLANG_ATTR(annotate("PURE")) // Comment tag that function is threadsafe #define VL_MT_SAFE VL_CLANG_ATTR(annotate("MT_SAFE")) // Comment tag that function is threadsafe, only if // other threads doesn't change tree topology #define VL_MT_STABLE VL_CLANG_ATTR(annotate("MT_STABLE")) // Comment tag that function is threadsafe, only // during normal operation (post-init) #define VL_MT_SAFE_POSTINIT VL_CLANG_ATTR(annotate("MT_SAFE_POSTINIT")) // Attribute that function is clang threadsafe and uses given mutex #define VL_MT_SAFE_EXCLUDES(mutex) VL_CLANG_ATTR(annotate("MT_SAFE_EXCLUDES")) VL_EXCLUDES(mutex) // Comment tag that function is not threadsafe #define VL_MT_UNSAFE VL_CLANG_ATTR(annotate("MT_UNSAFE")) // Comment tag that function is not threadsafe // protected to make sure single-caller #define VL_MT_UNSAFE_ONE VL_CLANG_ATTR(annotate("MT_UNSAFE_ONE")) // Comment tag that function is entry point of parallelization #define VL_MT_START VL_CLANG_ATTR(annotate("MT_START")) #ifndef VL_NO_LEGACY # define VL_ULL(c) (c##ULL) // Add appropriate suffix to 64-bit constant (deprecated) #endif // Convert argument to IData // This is not necessarily the same as "#UL", depending on what the IData typedef is. #define VL_UL(c) (static_cast(c##UL)) #if defined(VL_CPPCHECK) || defined(__clang_analyzer__) || __cplusplus < 201103L # define VL_DANGLING(var) #else /// After e.g. delete, set variable to nullptr to indicate must not use later # define VL_DANGLING(var) \ do { \ *const_cast(reinterpret_cast(&var)) = nullptr; \ } while (false) #endif /// Perform an e.g. delete, then set variable to nullptr to indicate must not use later. /// Unlike VL_DO_CLEAR the setting of the variable is only for debug reasons. #define VL_DO_DANGLING(stmt, var) \ do { \ do { \ stmt; \ } while (false); \ VL_DANGLING(var); \ } while (false) /// Perform an e.g. delete, then set variable to nullptr as a requirement #define VL_DO_CLEAR(stmt, stmt2) \ do { \ do { \ stmt; \ } while (false); \ do { \ stmt2; \ } while (false); \ } while (false) #ifdef _MSC_VER # if _MSC_VER < 1929 # error "Verilator requires at least Visual Studio 2019 version 16.11.2" # endif #endif //========================================================================= // C++-2011 #if __cplusplus >= 201103L || defined(__GXX_EXPERIMENTAL_CXX0X__) || defined(VL_CPPCHECK) || defined(_MSC_VER) #else # error "Verilator requires a C++11 or newer compiler" #endif #ifndef VL_NO_LEGACY // These are deprecated historical defines. We leave them in case users referenced them. # define VL_EQ_DELETE = delete # define vl_unique_ptr std::unique_ptr # define vl_unordered_map std::unordered_map # define vl_unordered_set std::unordered_set # define VL_INCLUDE_UNORDERED_MAP # define VL_INCLUDE_UNORDERED_SET # define VL_FINAL final # define VL_MUTABLE mutable # define VL_OVERRIDE override #endif //========================================================================= // C++-2017 #if __cplusplus >= 201703L # define VL_CONSTEXPR_CXX17 constexpr #else # define VL_CONSTEXPR_CXX17 #endif //========================================================================= // Optimization #ifndef VL_INLINE_OPT # define VL_INLINE_OPT ///< "inline" if compiling all objects in single compiler run #endif //========================================================================= // Internal coverage #ifdef VL_GCOV extern "C" void __gcov_dump(); // Dump internal code coverage data before e.g. std::abort() # define VL_GCOV_DUMP() __gcov_dump() #else # define VL_GCOV_DUMP() #endif //========================================================================= // Warning disabled #ifndef VL_WARNINGS # ifdef _MSC_VER # pragma warning(disable:4099) // C4099: type name first seen using 'class' now seen using 'struct' (V3AstNode) # pragma warning(disable:4100) // C4100: unreferenced formal parameter (L4) # pragma warning(disable:4127) // C4127: conditional expression is constant (L4) # pragma warning(disable:4146) // C4146: unary minus operator applied to unsigned type, result still unsigned # pragma warning(disable:4189) // C4189: local variable is initialized but not referenced (L4) # pragma warning(disable:4244) // C4244: conversion from 'uint64_t' to 'uint_32_t', possible loss of data # pragma warning(disable:4245) // C4245: conversion from 'int' to 'unsigned', signed/unsigned mismatch # pragma warning(disable:4996) // C4996: sscanf/fopen/etc may be unsafe # endif #endif //========================================================================= // Basic integer types #ifdef __MINGW32__ # define __USE_MINGW_ANSI_STDIO 1 // Force old MinGW (GCC 5 and older) to use C99 formats #endif // The inttypes supplied with some GCC & MINGW32 versions requires STDC_FORMAT_MACROS // to be declared in order to get the PRIxx macros used by fstapi.c #define __STDC_FORMAT_MACROS // Now that C++ requires these standard types the vl types are deprecated #include #include #include #ifndef VL_NO_LEGACY using vluint8_t = uint8_t; ///< 8-bit unsigned type (backward compatibility) using vluint16_t = uint16_t; ///< 16-bit unsigned type (backward compatibility) using vluint32_t = uint32_t; ///< 32-bit unsigned type (backward compatibility) using vluint64_t = uint64_t; ///< 64-bit unsigned type (backward compatibility) using vlsint8_t = int8_t; ///< 8-bit signed type (backward compatibility) using vlsint16_t = int16_t; ///< 16-bit signed type (backward compatibility) using vlsint32_t = int32_t; ///< 32-bit signed type (backward compatibility) using vlsint64_t = int64_t; ///< 64-bit signed type (backward compatibility) #endif #if defined(__CYGWIN__) # include // __WORDSIZE # include // ssize_t #elif defined(_WIN32) && defined(_MSC_VER) # ifndef _SSIZE_T_DEFINED # ifdef _WIN64 using ssize_t = uint64_t; ///< signed size_t; returned from read() # else using ssize_t = uint32_t; ///< signed size_t; returned from read() # endif # endif #else // Linux or compliant Unix flavors, -m64 # include // Solaris # include // __WORDSIZE # include // ssize_t #endif //========================================================================= // Printing printf/scanf formats // Use Microsoft-specific format specifiers for Microsoft Visual C++ only // Deprecated, favor C++11's PRIx64, etc, instead #ifndef VL_NO_LEGACY # ifdef _MSC_VER # define VL_PRI64 "I64" ///< print a uint64_t (backward compatibility) # else // use standard C99 format specifiers # if defined(__WORDSIZE) && (__WORDSIZE == 64) # define VL_PRI64 "l" ///< print a uint64_t (backward compatibility) # else # define VL_PRI64 "ll" ///< print a uint64_t (backward compatibility) # endif # endif #endif #if defined(_WIN32) && defined(_MSC_VER) # if (_MSC_VER < 1900) # define VL_SNPRINTF _snprintf # else # define VL_SNPRINTF snprintf # endif # define VL_VSNPRINTF vsnprintf #else # define VL_SNPRINTF snprintf # define VL_VSNPRINTF vsnprintf #endif //========================================================================= // File system functions #ifdef _WIN32 # define VL_DEV_NULL "nul" #else // Linux or compliant Unix flavors # define VL_DEV_NULL "/dev/null" #endif //========================================================================= // Integer size macros #define VL_BYTESIZE 8 ///< Bits in a CData / byte #define VL_SHORTSIZE 16 ///< Bits in a SData / short #define VL_IDATASIZE 32 ///< Bits in an IData / word #define VL_QUADSIZE 64 ///< Bits in a QData / quadword #define VL_EDATASIZE 32 ///< Bits in an EData (WData entry) #define VL_EDATASIZE_LOG2 5 ///< log2(VL_EDATASIZE) #define VL_CACHE_LINE_BYTES 64 ///< Bytes in a cache line (for alignment) #ifndef VL_NO_LEGACY # define VL_WORDSIZE VL_IDATASIZE // Legacy define #endif /// Return number of bytes argument-number of bits needs (1 bit=1 byte) #define VL_BYTES_I(nbits) (((nbits) + (VL_BYTESIZE - 1)) / VL_BYTESIZE) /// Return Words/EDatas in argument-number of bits needs (1 bit=1 word) #define VL_WORDS_I(nbits) (((nbits) + (VL_EDATASIZE - 1)) / VL_EDATASIZE) // Number of Words/EDatas a quad requires #define VL_WQ_WORDS_E VL_WORDS_I(VL_QUADSIZE) //========================================================================= // Class definition helpers // Comment tag to indicate a base class, e.g. cannot label "class final" #define VL_NOT_FINAL // Declare a class as uncopyable; put after a private: #define VL_UNCOPYABLE(Type) \ Type(const Type& other) = delete; \ Type& operator=(const Type&) = delete //========================================================================= // Verilated function size macros #define VL_MULS_MAX_WORDS 16 ///< Max size in words of MULS operation #ifndef VL_VALUE_STRING_MAX_WORDS #define VL_VALUE_STRING_MAX_WORDS 64 ///< Max size in words of String conversion operation #endif #define VL_VALUE_STRING_MAX_CHARS (VL_VALUE_STRING_MAX_WORDS * VL_EDATASIZE / VL_BYTESIZE) //========================================================================= // Base macros #define VL_SIZEBITS_I (VL_IDATASIZE - 1) ///< Bit mask for bits in a word #define VL_SIZEBITS_Q (VL_QUADSIZE - 1) ///< Bit mask for bits in a quad #define VL_SIZEBITS_E (VL_EDATASIZE - 1) ///< Bit mask for bits in a quad /// Return mask for words with 1's where relevant bits are (0=all bits) /// Arguments must not have side effects #define VL_MASK_I(nbits) (((nbits) & VL_SIZEBITS_I) ? ((1U << ((nbits) & VL_SIZEBITS_I)) - 1) : ~0) /// Return mask for quads with 1's where relevant bits are (0=all bits) /// Arguments must not have side effects #define VL_MASK_Q(nbits) \ (((nbits) & VL_SIZEBITS_Q) ? ((1ULL << ((nbits) & VL_SIZEBITS_Q)) - 1ULL) : ~0ULL) /// Return mask for EData with 1's where relevant bits are (0=all bits) /// Arguments must not have side effects #define VL_MASK_E(nbits) VL_MASK_I(nbits) #define VL_EUL(n) VL_UL(n) // Make constant number EData sized #define VL_BITWORD_I(bit) ((bit) / VL_IDATASIZE) ///< Word number for sv DPI vectors #define VL_BITWORD_E(bit) ((bit) >> VL_EDATASIZE_LOG2) ///< Word number for a wide quantity #define VL_BITBIT_I(bit) ((bit) & VL_SIZEBITS_I) ///< Bit number for a bit in a long #define VL_BITBIT_Q(bit) ((bit) & VL_SIZEBITS_Q) ///< Bit number for a bit in a quad #define VL_BITBIT_E(bit) ((bit) & VL_SIZEBITS_E) ///< Bit number for a bit in an EData // Return true if data[bit] set; not 0/1 return, but 0/non-zero return. #define VL_BITISSET_I(data, bit) ((data) & (VL_UL(1) << VL_BITBIT_I(bit))) #define VL_BITISSET_Q(data, bit) ((data) & (1ULL << VL_BITBIT_Q(bit))) #define VL_BITISSET_E(data, bit) ((data) & (VL_EUL(1) << VL_BITBIT_E(bit))) #define VL_BITISSET_W(data, bit) ((data)[VL_BITWORD_E(bit)] & (VL_EUL(1) << VL_BITBIT_E(bit))) //========================================================================= // Floating point // #defines, to avoid requiring math.h on all compile runs #ifdef _MSC_VER static inline double VL_TRUNC(double n) { return (n < 0) ? std::ceil(n) : std::floor(n); } static inline double VL_ROUND(double n) { return (n < 0) ? std::ceil(n-0.5) : std::floor(n + 0.5); } #else # define VL_TRUNC(n) std::trunc(n) # define VL_ROUND(n) std::round(n) #endif //========================================================================= // Performance counters #if defined(__i386__) || defined(__x86_64__) // The uint64_t argument is loaded with a high-performance counter for profiling // or 0x0 if not implemented on this platform #define VL_GET_CPU_TICK(val) \ { \ uint32_t hi; \ uint32_t lo; \ asm volatile("rdtsc" : "=a"(lo), "=d"(hi)); \ (val) = ((uint64_t)lo) | (((uint64_t)hi) << 32); \ } #elif defined(__aarch64__) // 1 GHz virtual system timer on SBSA level 5 compliant systems, else often 100 MHz # define VL_GET_CPU_TICK(val) \ { \ asm volatile("isb" : : : "memory"); \ asm volatile("mrs %[rt],CNTVCT_EL0" : [rt] "=r"(val)); \ } #else // We just silently ignore unknown OSes, as only leads to missing statistics # define VL_GET_CPU_TICK(val) (val) = 0; #endif //========================================================================= // Threading related OS-specific functions #ifdef _WIN32 # define WIN32_LEAN_AND_MEAN # ifndef NOMINMAX # define NOMINMAX # endif # include "windows.h" # define VL_CPU_RELAX() YieldProcessor() #elif defined(__i386__) || defined(__x86_64__) || defined(VL_CPPCHECK) // For more efficient busy waiting on SMT CPUs, let the processor know // we're just waiting so it can let another thread run # define VL_CPU_RELAX() asm volatile("rep; nop" ::: "memory") #elif defined(__ia64__) # define VL_CPU_RELAX() asm volatile("hint @pause" ::: "memory") #elif defined(__armel__) || defined(__ARMEL__) // Arm, but broken, must be before __arm__ # define VL_CPU_RELAX() asm volatile("nop" ::: "memory"); #elif defined(__aarch64__) || defined(__arm__) # define VL_CPU_RELAX() asm volatile("yield" ::: "memory") #elif defined(__hppa__) // HPPA does not currently have yield/pause # define VL_CPU_RELAX() asm volatile("nop" ::: "memory") #elif defined(__loongarch__) // LoongArch does not currently have yield/pause # define VL_CPU_RELAX() asm volatile("nop" ::: "memory") #elif defined(__mips64el__) || defined(__mips__) || defined(__mips64__) || defined(__mips64) # define VL_CPU_RELAX() asm volatile("pause" ::: "memory") #elif defined(__powerpc64__) # define VL_CPU_RELAX() asm volatile("or 1, 1, 1; or 2, 2, 2;" ::: "memory") #elif defined(__riscv) // RiscV does not currently have yield/pause, but one is proposed # define VL_CPU_RELAX() asm volatile("nop" ::: "memory") #elif defined(__s390x__) # define VL_CPU_RELAX() asm volatile("lr 0,0" ::: "memory") #elif defined(__sparc__) # define VL_CPU_RELAX() asm volatile("rd %%ccr, %%g0" ::: "memory") #elif defined(VL_IGNORE_UNKNOWN_ARCH) # define VL_CPU_RELAX() #else # error "Missing VL_CPU_RELAX() definition." #endif //========================================================================= // String/time related OS-specific functions #ifdef _MSC_VER # define VL_STRCASECMP _stricmp #else # define VL_STRCASECMP strcasecmp #endif //========================================================================= // Macros controlling target specific optimizations // Define VL_PORTABLE_ONLY to disable all target specific optimizations #ifndef VL_PORTABLE_ONLY # ifdef __x86_64__ # define VL_X86_64 1 # endif #endif // VL_PORTABLE_ONLY // clang-format on //========================================================================= // Stringify macros #define VL_STRINGIFY(...) VL_STRINGIFY2(__VA_ARGS__) #define VL_STRINGIFY2(...) #__VA_ARGS__ //========================================================================= // Offset of field in type // Address zero can cause compiler problems #define VL_OFFSETOF(type, field) \ (reinterpret_cast(&(reinterpret_cast(0x10000000)->field)) - 0x10000000) //========================================================================= // Conversions #include namespace vlstd { template struct reverse_wrapper { const T& m_v; explicit reverse_wrapper(const T& a_v) : m_v(a_v) {} auto begin() -> decltype(m_v.rbegin()) { return m_v.rbegin(); } auto end() -> decltype(m_v.rend()) { return m_v.rend(); } }; // C++20's std::ranges::reverse_view template reverse_wrapper reverse_view(const T& v) { return reverse_wrapper(v); } // C++17's std::as_const // `VL_MT_SAFE` annotation only applies to this function. // Object that is returned by this function is not considered // as MT_SAFE and any function call on this object still // needs to be `VL_MT_SAFE`. template T const& as_const(T& v) VL_MT_SAFE { return v; } // C++14's std::exchange template T exchange(T& obj, U&& new_value) { T old_value = std::move(obj); obj = std::forward(new_value); return old_value; } }; // namespace vlstd //========================================================================= #endif // Guard