| |
| |
| |
| |
| |
| |
| |
|
|
| |
|
|
| #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 |
|
|