2016-02-18 19:27:17 +03:00
|
|
|
/**
|
|
|
|
Lightweight profiler library for c++
|
|
|
|
Copyright(C) 2016 Sergey Yagovtsev
|
|
|
|
|
|
|
|
This program is free software : you can redistribute it and / or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program 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 this program.If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
**/
|
|
|
|
|
2016-09-06 23:03:05 +03:00
|
|
|
#ifndef EASY_PROFILER__SPIN_LOCK__________H______
|
|
|
|
#define EASY_PROFILER__SPIN_LOCK__________H______
|
2016-02-18 19:27:17 +03:00
|
|
|
|
2016-09-06 23:03:05 +03:00
|
|
|
#define EASY_USE_CRITICAL_SECTION // Use CRITICAL_SECTION instead of std::atomic_flag
|
|
|
|
|
|
|
|
#if defined(_WIN32) && defined(EASY_USE_CRITICAL_SECTION)
|
|
|
|
#include <Windows.h>
|
|
|
|
#else
|
2016-02-18 19:27:17 +03:00
|
|
|
#include <atomic>
|
2016-09-06 23:03:05 +03:00
|
|
|
#endif
|
|
|
|
|
|
|
|
namespace profiler {
|
|
|
|
|
|
|
|
#if defined(_WIN32) && defined(EASY_USE_CRITICAL_SECTION)
|
|
|
|
// std::atomic_flag on Windows works slower than critical section, so we will use it instead of std::atomic_flag...
|
|
|
|
// By the way, Windows critical sections are slower than std::atomic_flag on Unix.
|
2016-09-22 23:06:43 +03:00
|
|
|
class spin_lock { CRITICAL_SECTION m_lock; public:
|
2016-09-06 23:03:05 +03:00
|
|
|
|
|
|
|
void lock() {
|
|
|
|
EnterCriticalSection(&m_lock);
|
|
|
|
}
|
|
|
|
|
|
|
|
void unlock() {
|
|
|
|
LeaveCriticalSection(&m_lock);
|
|
|
|
}
|
|
|
|
|
|
|
|
spin_lock() {
|
|
|
|
InitializeCriticalSection(&m_lock);
|
|
|
|
}
|
|
|
|
|
|
|
|
~spin_lock() {
|
|
|
|
DeleteCriticalSection(&m_lock);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
#else
|
|
|
|
// std::atomic_flag on Unix works fine and very fast (almost instant!)
|
2016-09-22 23:06:43 +03:00
|
|
|
class spin_lock { ::std::atomic_flag m_lock; public:
|
2016-09-06 23:03:05 +03:00
|
|
|
|
|
|
|
void lock() {
|
|
|
|
while (m_lock.test_and_set(::std::memory_order_acquire));
|
|
|
|
}
|
2016-02-18 19:27:17 +03:00
|
|
|
|
2016-09-06 23:03:05 +03:00
|
|
|
void unlock() {
|
|
|
|
m_lock.clear(::std::memory_order_release);
|
|
|
|
}
|
2016-02-18 19:27:17 +03:00
|
|
|
|
2016-09-06 23:03:05 +03:00
|
|
|
spin_lock() {
|
|
|
|
m_lock.clear();
|
|
|
|
}
|
|
|
|
};
|
2016-02-18 19:27:17 +03:00
|
|
|
#endif
|
2016-09-06 23:03:05 +03:00
|
|
|
|
|
|
|
template <class T>
|
2016-09-22 23:06:43 +03:00
|
|
|
class guard_lock
|
2016-09-06 23:03:05 +03:00
|
|
|
{
|
|
|
|
T& m_lock;
|
|
|
|
bool m_isLocked = false;
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
explicit guard_lock(T& m) : m_lock(m) {
|
|
|
|
m_lock.lock();
|
|
|
|
m_isLocked = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
~guard_lock() {
|
|
|
|
unlock();
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void unlock() {
|
|
|
|
if (m_isLocked) {
|
|
|
|
m_lock.unlock();
|
|
|
|
m_isLocked = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
} // END of namespace profiler.
|
|
|
|
|
|
|
|
#ifdef EASY_USE_CRITICAL_SECTION
|
|
|
|
# undef EASY_USE_CRITICAL_SECTION
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#endif // EASY_PROFILER__SPIN_LOCK__________H______
|