tracy/client/TracyLock.hpp

333 lines
11 KiB
C++
Raw Normal View History

2017-10-04 13:41:02 +00:00
#ifndef __TRACYLOCK_HPP__
#define __TRACYLOCK_HPP__
#include <atomic>
#include <limits>
2017-10-04 14:45:46 +00:00
#include "../common/TracySystem.hpp"
#include "../common/TracyAlign.hpp"
2017-10-04 13:41:02 +00:00
#include "TracyProfiler.hpp"
namespace tracy
{
extern std::atomic<uint32_t> s_lockCounter;
2017-10-04 13:41:02 +00:00
template<class T>
class Lockable
{
public:
2017-10-06 14:33:11 +00:00
tracy_force_inline Lockable( const SourceLocation* srcloc )
: m_id( s_lockCounter.fetch_add( 1, std::memory_order_relaxed ) )
2017-10-04 13:41:02 +00:00
{
assert( m_id != std::numeric_limits<uint32_t>::max() );
2017-12-10 20:37:39 +00:00
Magic magic;
auto& token = s_token.ptr;
auto& tail = token->get_tail_index();
auto item = token->enqueue_begin<moodycamel::CanAlloc>( magic );
MemWrite( &item->hdr.type, QueueType::LockAnnounce );
MemWrite( &item->lockAnnounce.id, m_id );
MemWrite( &item->lockAnnounce.lckloc, (uint64_t)srcloc );
MemWrite( &item->lockAnnounce.type, LockType::Lockable );
2018-07-11 10:24:58 +00:00
#ifdef TRACY_ON_DEMAND
s_profiler.DeferItem( *item );
#endif
2018-07-11 10:24:58 +00:00
tail.store( magic + 1, std::memory_order_release );
2017-10-04 13:41:02 +00:00
}
Lockable( const Lockable& ) = delete;
Lockable& operator=( const Lockable& ) = delete;
2017-10-06 14:33:11 +00:00
tracy_force_inline void lock()
2017-10-04 13:41:02 +00:00
{
#ifndef TRACY_ON_DEMAND
2017-10-04 14:45:46 +00:00
const auto thread = GetThreadHandle();
{
Magic magic;
2017-10-10 23:44:35 +00:00
auto& token = s_token.ptr;
2017-10-10 23:27:22 +00:00
auto& tail = token->get_tail_index();
2017-10-10 23:04:21 +00:00
auto item = token->enqueue_begin<moodycamel::CanAlloc>( magic );
MemWrite( &item->hdr.type, QueueType::LockWait );
MemWrite( &item->lockWait.id, m_id );
MemWrite( &item->lockWait.thread, thread );
MemWrite( &item->lockWait.time, Profiler::GetTime() );
MemWrite( &item->lockWait.type, LockType::Lockable );
2017-10-10 23:27:22 +00:00
tail.store( magic + 1, std::memory_order_release );
2017-10-04 14:45:46 +00:00
}
#endif
2017-10-04 14:45:46 +00:00
2017-10-04 13:41:02 +00:00
m_lockable.lock();
2017-10-04 14:45:46 +00:00
#ifndef TRACY_ON_DEMAND
2017-10-04 14:45:46 +00:00
{
Magic magic;
2017-10-10 23:44:35 +00:00
auto& token = s_token.ptr;
2017-10-10 23:27:22 +00:00
auto& tail = token->get_tail_index();
2017-10-10 23:04:21 +00:00
auto item = token->enqueue_begin<moodycamel::CanAlloc>( magic );
MemWrite( &item->hdr.type, QueueType::LockObtain );
MemWrite( &item->lockObtain.id, m_id );
MemWrite( &item->lockObtain.thread, thread );
MemWrite( &item->lockObtain.time, Profiler::GetTime() );
2017-10-10 23:27:22 +00:00
tail.store( magic + 1, std::memory_order_release );
2017-10-04 14:45:46 +00:00
}
#endif
2017-10-04 13:41:02 +00:00
}
2017-10-06 14:33:11 +00:00
tracy_force_inline void unlock()
2017-10-04 13:41:02 +00:00
{
m_lockable.unlock();
2017-10-04 14:45:46 +00:00
#ifndef TRACY_ON_DEMAND
2017-10-04 14:45:46 +00:00
Magic magic;
2017-10-10 23:44:35 +00:00
auto& token = s_token.ptr;
2017-10-10 23:27:22 +00:00
auto& tail = token->get_tail_index();
2017-10-10 23:04:21 +00:00
auto item = token->enqueue_begin<moodycamel::CanAlloc>( magic );
MemWrite( &item->hdr.type, QueueType::LockRelease );
MemWrite( &item->lockRelease.id, m_id );
MemWrite( &item->lockRelease.thread, GetThreadHandle() );
MemWrite( &item->lockRelease.time, Profiler::GetTime() );
2017-10-10 23:27:22 +00:00
tail.store( magic + 1, std::memory_order_release );
#endif
2017-10-04 13:41:02 +00:00
}
2017-10-06 14:33:11 +00:00
tracy_force_inline bool try_lock()
2017-10-04 13:41:02 +00:00
{
2017-10-04 14:45:46 +00:00
const auto ret = m_lockable.try_lock();
#ifndef TRACY_ON_DEMAND
2017-10-04 14:45:46 +00:00
if( ret )
{
Magic magic;
2017-10-10 23:44:35 +00:00
auto& token = s_token.ptr;
2017-10-10 23:27:22 +00:00
auto& tail = token->get_tail_index();
2017-10-10 23:04:21 +00:00
auto item = token->enqueue_begin<moodycamel::CanAlloc>( magic );
MemWrite( &item->hdr.type, QueueType::LockObtain );
2018-03-31 12:13:46 +00:00
MemWrite( &item->lockObtain.id, m_id );
MemWrite( &item->lockObtain.thread, GetThreadHandle() );
MemWrite( &item->lockObtain.time, Profiler::GetTime() );
2017-10-10 23:27:22 +00:00
tail.store( magic + 1, std::memory_order_release );
2017-10-04 14:45:46 +00:00
}
#endif
2017-10-04 14:45:46 +00:00
return ret;
2017-10-04 13:41:02 +00:00
}
tracy_force_inline void Mark( const SourceLocation* srcloc ) const
2017-10-06 14:32:32 +00:00
{
#ifndef TRACY_ON_DEMAND
2017-10-06 14:32:32 +00:00
Magic magic;
2017-10-10 23:44:35 +00:00
auto& token = s_token.ptr;
2017-10-10 23:27:22 +00:00
auto& tail = token->get_tail_index();
2017-10-10 23:04:21 +00:00
auto item = token->enqueue_begin<moodycamel::CanAlloc>( magic );
MemWrite( &item->hdr.type, QueueType::LockMark );
MemWrite( &item->lockMark.id, m_id );
MemWrite( &item->lockMark.thread, GetThreadHandle() );
MemWrite( &item->lockMark.srcloc, (uint64_t)srcloc );
2017-10-10 23:27:22 +00:00
tail.store( magic + 1, std::memory_order_release );
#endif
2017-10-06 14:32:32 +00:00
}
2017-10-04 13:41:02 +00:00
private:
T m_lockable;
uint32_t m_id;
2017-10-04 13:41:02 +00:00
};
2017-12-10 20:49:45 +00:00
template<class T>
class SharedLockable
{
public:
tracy_force_inline SharedLockable( const SourceLocation* srcloc )
: m_id( s_lockCounter.fetch_add( 1, std::memory_order_relaxed ) )
{
assert( m_id != std::numeric_limits<uint32_t>::max() );
Magic magic;
auto& token = s_token.ptr;
auto& tail = token->get_tail_index();
auto item = token->enqueue_begin<moodycamel::CanAlloc>( magic );
MemWrite( &item->hdr.type, QueueType::LockAnnounce );
MemWrite( &item->lockAnnounce.id, m_id );
MemWrite( &item->lockAnnounce.lckloc, (uint64_t)srcloc );
MemWrite( &item->lockAnnounce.type, LockType::SharedLockable );
2018-07-11 10:24:58 +00:00
#ifdef TRACY_ON_DEMAND
s_profiler.DeferItem( *item );
#endif
2018-07-11 10:24:58 +00:00
tail.store( magic + 1, std::memory_order_release );
2017-12-10 20:49:45 +00:00
}
SharedLockable( const SharedLockable& ) = delete;
SharedLockable& operator=( const SharedLockable& ) = delete;
tracy_force_inline void lock()
{
#ifndef TRACY_ON_DEMAND
2017-12-10 20:49:45 +00:00
const auto thread = GetThreadHandle();
{
Magic magic;
auto& token = s_token.ptr;
auto& tail = token->get_tail_index();
auto item = token->enqueue_begin<moodycamel::CanAlloc>( magic );
MemWrite( &item->hdr.type, QueueType::LockWait );
MemWrite( &item->lockWait.id, m_id );
MemWrite( &item->lockWait.thread, thread );
MemWrite( &item->lockWait.time, Profiler::GetTime() );
MemWrite( &item->lockWait.type, LockType::SharedLockable );
2017-12-10 20:49:45 +00:00
tail.store( magic + 1, std::memory_order_release );
}
#endif
2017-12-10 20:49:45 +00:00
m_lockable.lock();
#ifndef TRACY_ON_DEMAND
2017-12-10 20:49:45 +00:00
{
Magic magic;
auto& token = s_token.ptr;
auto& tail = token->get_tail_index();
auto item = token->enqueue_begin<moodycamel::CanAlloc>( magic );
MemWrite( &item->hdr.type, QueueType::LockObtain );
MemWrite( &item->lockObtain.id, m_id );
MemWrite( &item->lockObtain.thread, thread );
MemWrite( &item->lockObtain.time, Profiler::GetTime() );
2017-12-10 20:49:45 +00:00
tail.store( magic + 1, std::memory_order_release );
}
#endif
2017-12-10 20:49:45 +00:00
}
tracy_force_inline void unlock()
{
m_lockable.unlock();
#ifndef TRACY_ON_DEMAND
2017-12-10 20:49:45 +00:00
Magic magic;
auto& token = s_token.ptr;
auto& tail = token->get_tail_index();
auto item = token->enqueue_begin<moodycamel::CanAlloc>( magic );
2018-03-31 12:19:45 +00:00
MemWrite( &item->hdr.type, QueueType::LockRelease );
MemWrite( &item->lockRelease.id, m_id );
MemWrite( &item->lockRelease.thread, GetThreadHandle() );
MemWrite( &item->lockRelease.time, Profiler::GetTime() );
2017-12-10 20:49:45 +00:00
tail.store( magic + 1, std::memory_order_release );
#endif
2017-12-10 20:49:45 +00:00
}
tracy_force_inline bool try_lock()
{
const auto ret = m_lockable.try_lock();
#ifndef TRACY_ON_DEMAND
2017-12-10 20:49:45 +00:00
if( ret )
{
Magic magic;
auto& token = s_token.ptr;
auto& tail = token->get_tail_index();
auto item = token->enqueue_begin<moodycamel::CanAlloc>( magic );
MemWrite( &item->hdr.type, QueueType::LockObtain );
2018-03-31 12:13:46 +00:00
MemWrite( &item->lockObtain.id, m_id );
MemWrite( &item->lockObtain.thread, GetThreadHandle() );
MemWrite( &item->lockObtain.time, Profiler::GetTime() );
2017-12-10 20:49:45 +00:00
tail.store( magic + 1, std::memory_order_release );
}
#endif
2017-12-10 20:49:45 +00:00
return ret;
}
tracy_force_inline void lock_shared()
{
#ifndef TRACY_ON_DEMAND
const auto thread = GetThreadHandle();
{
Magic magic;
auto& token = s_token.ptr;
auto& tail = token->get_tail_index();
auto item = token->enqueue_begin<moodycamel::CanAlloc>( magic );
MemWrite( &item->hdr.type, QueueType::LockSharedWait );
MemWrite( &item->lockWait.id, m_id );
MemWrite( &item->lockWait.thread, thread );
MemWrite( &item->lockWait.time, Profiler::GetTime() );
MemWrite( &item->lockWait.type, LockType::SharedLockable );
tail.store( magic + 1, std::memory_order_release );
}
#endif
2017-12-10 20:49:45 +00:00
m_lockable.lock_shared();
#ifndef TRACY_ON_DEMAND
{
Magic magic;
auto& token = s_token.ptr;
auto& tail = token->get_tail_index();
auto item = token->enqueue_begin<moodycamel::CanAlloc>( magic );
MemWrite( &item->hdr.type, QueueType::LockSharedObtain );
MemWrite( &item->lockObtain.id, m_id );
MemWrite( &item->lockObtain.thread, thread );
MemWrite( &item->lockObtain.time, Profiler::GetTime() );
tail.store( magic + 1, std::memory_order_release );
}
#endif
2017-12-10 20:49:45 +00:00
}
tracy_force_inline void unlock_shared()
{
m_lockable.unlock_shared();
#ifndef TRACY_ON_DEMAND
Magic magic;
auto& token = s_token.ptr;
auto& tail = token->get_tail_index();
auto item = token->enqueue_begin<moodycamel::CanAlloc>( magic );
MemWrite( &item->hdr.type, QueueType::LockSharedRelease );
MemWrite( &item->lockRelease.id, m_id );
MemWrite( &item->lockRelease.thread, GetThreadHandle() );
MemWrite( &item->lockRelease.time, Profiler::GetTime() );
tail.store( magic + 1, std::memory_order_release );
#endif
2017-12-10 20:49:45 +00:00
}
tracy_force_inline bool try_lock_shared()
{
const auto ret = m_lockable.try_lock_shared();
#ifndef TRACY_ON_DEMAND
if( ret )
{
Magic magic;
auto& token = s_token.ptr;
auto& tail = token->get_tail_index();
auto item = token->enqueue_begin<moodycamel::CanAlloc>( magic );
MemWrite( &item->hdr.type, QueueType::LockSharedObtain );
2018-03-31 12:13:46 +00:00
MemWrite( &item->lockObtain.id, m_id );
MemWrite( &item->lockObtain.thread, GetThreadHandle() );
MemWrite( &item->lockObtain.time, Profiler::GetTime() );
tail.store( magic + 1, std::memory_order_release );
}
#endif
2017-12-10 20:49:45 +00:00
return ret;
}
tracy_force_inline void Mark( const SourceLocation* srcloc ) const
{
#ifndef TRACY_ON_DEMAND
2017-12-10 20:49:45 +00:00
Magic magic;
auto& token = s_token.ptr;
auto& tail = token->get_tail_index();
auto item = token->enqueue_begin<moodycamel::CanAlloc>( magic );
MemWrite( &item->hdr.type, QueueType::LockMark );
MemWrite( &item->lockMark.id, m_id );
MemWrite( &item->lockMark.thread, GetThreadHandle() );
MemWrite( &item->lockMark.srcloc, (uint64_t)srcloc );
2017-12-10 20:49:45 +00:00
tail.store( magic + 1, std::memory_order_release );
#endif
2017-12-10 20:49:45 +00:00
}
private:
T m_lockable;
uint32_t m_id;
};
2017-10-04 13:41:02 +00:00
};
#endif