1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768 |
-
- #ifndef TWOBLUECUBES_CATCH_OPTIMIZER_HPP_INCLUDED
- #define TWOBLUECUBES_CATCH_OPTIMIZER_HPP_INCLUDED
- #if defined(_MSC_VER)
- # include <atomic>
- #endif
- namespace Catch {
- namespace Benchmark {
- #if defined(__GNUC__) || defined(__clang__)
- template <typename T>
- inline void keep_memory(T* p) {
- asm volatile("" : : "g"(p) : "memory");
- }
- inline void keep_memory() {
- asm volatile("" : : : "memory");
- }
- namespace Detail {
- inline void optimizer_barrier() { keep_memory(); }
- }
- #elif defined(_MSC_VER)
- #pragma optimize("", off)
- template <typename T>
- inline void keep_memory(T* p) {
-
- *reinterpret_cast<char volatile*>(p) = *reinterpret_cast<char const volatile*>(p);
- }
-
- #pragma optimize("", on)
- namespace Detail {
- inline void optimizer_barrier() {
- std::atomic_thread_fence(std::memory_order_seq_cst);
- }
- }
- #endif
- template <typename T>
- inline void deoptimize_value(T&& x) {
- keep_memory(&x);
- }
- template <typename Fn, typename... Args>
- inline auto invoke_deoptimized(Fn&& fn, Args&&... args) -> typename std::enable_if<!std::is_same<void, decltype(fn(args...))>::value>::type {
- deoptimize_value(std::forward<Fn>(fn) (std::forward<Args...>(args...)));
- }
- template <typename Fn, typename... Args>
- inline auto invoke_deoptimized(Fn&& fn, Args&&... args) -> typename std::enable_if<std::is_same<void, decltype(fn(args...))>::value>::type {
- std::forward<Fn>(fn) (std::forward<Args...>(args...));
- }
- }
- }
- #endif
|