tracy/server/TracyEvent.hpp

516 lines
12 KiB
C++
Raw Normal View History

2017-09-14 00:00:13 +00:00
#ifndef __TRACYEVENT_HPP__
#define __TRACYEVENT_HPP__
#include <assert.h>
2017-10-08 21:03:38 +00:00
#include <limits>
#include <stdint.h>
2019-10-13 13:28:52 +00:00
#include <string>
2017-11-15 20:26:58 +00:00
#include <string.h>
2017-10-08 21:03:38 +00:00
2018-02-08 16:08:31 +00:00
#include "TracyCharUtil.hpp"
2017-09-15 18:17:39 +00:00
#include "TracyVector.hpp"
#include "tracy_flat_hash_map.hpp"
2017-09-14 00:00:13 +00:00
namespace tracy
{
#pragma pack( 1 )
struct StringRef
{
enum Type { Ptr, Idx };
2018-03-04 16:52:51 +00:00
StringRef() : str( 0 ), __data( 0 ) {}
StringRef( Type t, uint64_t data )
: str( data )
, __data( 0 )
{
isidx = t == Idx;
active = 1;
}
2018-03-04 16:52:51 +00:00
uint64_t str;
2017-11-11 16:56:41 +00:00
union
{
struct
{
uint8_t isidx : 1;
uint8_t active : 1;
};
uint8_t __data;
};
};
class StringIdx
{
public:
StringIdx() { memset( m_idx, 0, sizeof( m_idx ) ); }
StringIdx( uint32_t idx )
{
SetIdx( idx );
}
void SetIdx( uint32_t idx )
{
idx++;
memcpy( m_idx, &idx, 3 );
}
uint32_t Idx() const
{
uint32_t idx = 0;
memcpy( &idx, m_idx, 3 );
assert( idx != 0 );
return idx - 1;
}
bool Active() const
{
uint32_t zero = 0;
return memcmp( m_idx, &zero, 3 ) != 0;
}
private:
uint8_t m_idx[3];
};
struct __StringIdxOld
{
uint32_t idx : 31;
uint32_t active : 1;
};
2019-10-01 19:48:34 +00:00
class Int24
{
public:
Int24() { memset( m_val, 0, sizeof( m_val ) ); }
Int24( uint32_t val )
{
SetVal( val );
}
void SetVal( uint32_t val )
{
memcpy( m_val, &val, 3 );
}
uint32_t Val() const
{
uint32_t val = 0;
memcpy( &val, m_val, 3 );
return val;
}
private:
uint8_t m_val[3];
};
struct SourceLocation
{
StringRef name;
StringRef function;
StringRef file;
uint32_t line;
uint32_t color;
};
enum { SourceLocationSize = sizeof( SourceLocation ) };
2017-10-22 13:37:24 +00:00
struct ZoneEvent
2017-09-14 00:00:13 +00:00
{
2019-08-15 19:38:00 +00:00
int64_t Start() const { return int64_t( _start_srcloc ) >> 16; }
2019-10-24 21:03:13 +00:00
void SetStart( int64_t start ) { assert( start < (int64_t)( 1ull << 47 ) ); memcpy( ((char*)&_start_srcloc)+2, &start, 6 ); }
int64_t End() const { return int64_t( _end_child1 ) >> 16; }
2019-10-24 21:03:13 +00:00
void SetEnd( int64_t end ) { assert( end < (int64_t)( 1ull << 47 ) ); memcpy( ((char*)&_end_child1)+2, &end, 6 ); }
2019-08-15 19:38:00 +00:00
int16_t SrcLoc() const { return int16_t( _start_srcloc & 0xFFFF ); }
2019-10-24 21:03:13 +00:00
void SetSrcLoc( int16_t srcloc ) { memcpy( &_start_srcloc, &srcloc, 2 ); }
int32_t Child() const { return int32_t( uint32_t( _end_child1 & 0xFFFF ) | ( uint32_t( _child2 ) << 16 ) ); }
2019-10-24 21:03:13 +00:00
void SetChild( int32_t child ) { memcpy( &_end_child1, &child, 2 ); _child2 = uint32_t( child ) >> 16; }
uint64_t _start_srcloc;
uint64_t _end_child1;
StringIdx text;
Int24 callstack;
2018-06-29 14:12:17 +00:00
StringIdx name;
uint16_t _child2;
2017-09-14 00:00:13 +00:00
};
2017-10-22 13:37:24 +00:00
enum { ZoneEventSize = sizeof( ZoneEvent ) };
static_assert( std::is_standard_layout<ZoneEvent>::value, "ZoneEvent is not standard layout" );
2017-10-04 14:16:40 +00:00
2019-08-15 12:15:40 +00:00
2017-10-04 14:16:40 +00:00
struct LockEvent
{
2017-10-04 16:17:31 +00:00
enum class Type : uint8_t
{
Wait,
Obtain,
2017-12-10 21:42:39 +00:00
Release,
WaitShared,
ObtainShared,
ReleaseShared
2017-10-04 16:17:31 +00:00
};
2019-08-15 19:38:00 +00:00
int64_t Time() const { return int64_t( _time_srcloc ) >> 16; }
2019-10-24 21:03:13 +00:00
void SetTime( int64_t time ) { assert( time < (int64_t)( 1ull << 47 ) ); memcpy( ((char*)&_time_srcloc)+2, &time, 6 ); }
2019-08-15 19:38:00 +00:00
int16_t SrcLoc() const { return int16_t( _time_srcloc & 0xFFFF ); }
2019-10-24 21:03:13 +00:00
void SetSrcLoc( int16_t srcloc ) { memcpy( &_time_srcloc, &srcloc, 2 ); }
2019-08-15 19:34:19 +00:00
uint64_t _time_srcloc;
2017-12-05 21:34:48 +00:00
uint8_t thread;
2017-12-10 21:37:56 +00:00
Type type;
};
struct LockEventShared : public LockEvent
{
uint64_t waitShared;
2017-12-10 21:42:39 +00:00
uint64_t sharedList;
2017-10-04 14:16:40 +00:00
};
struct LockEventPtr
{
LockEvent* ptr;
uint8_t lockingThread;
uint8_t lockCount;
uint64_t waitList;
};
2017-10-04 14:16:40 +00:00
enum { LockEventSize = sizeof( LockEvent ) };
enum { LockEventSharedSize = sizeof( LockEventShared ) };
enum { LockEventPtrSize = sizeof( LockEventPtr ) };
2017-10-04 14:16:40 +00:00
enum { MaxLockThreads = sizeof( LockEventPtr::waitList ) * 8 };
static_assert( std::numeric_limits<decltype(LockEventPtr::lockCount)>::max() >= MaxLockThreads, "Not enough space for lock count." );
2017-10-08 21:03:38 +00:00
2017-11-11 20:09:48 +00:00
struct GpuEvent
{
int64_t CpuStart() const { return int64_t( _cpuStart_srcloc ) >> 16; }
2019-10-24 21:03:13 +00:00
void SetCpuStart( int64_t cpuStart ) { assert( cpuStart < (int64_t)( 1ull << 47 ) ); memcpy( ((char*)&_cpuStart_srcloc)+2, &cpuStart, 6 ); }
int64_t CpuEnd() const { return int64_t( _cpuEnd_thread ) >> 16; }
2019-10-24 21:03:13 +00:00
void SetCpuEnd( int64_t cpuEnd ) { assert( cpuEnd < (int64_t)( 1ull << 47 ) ); memcpy( ((char*)&_cpuEnd_thread)+2, &cpuEnd, 6 ); }
int16_t SrcLoc() const { return int16_t( _cpuStart_srcloc & 0xFFFF ); }
2019-10-24 21:03:13 +00:00
void SetSrcLoc( int16_t srcloc ) { memcpy( &_cpuStart_srcloc, &srcloc, 2 ); }
uint16_t Thread() const { return uint16_t( _cpuEnd_thread & 0xFFFF ); }
2019-10-24 21:03:13 +00:00
void SetThread( uint16_t thread ) { memcpy( &_cpuEnd_thread, &thread, 2 ); }
uint64_t _cpuStart_srcloc;
uint64_t _cpuEnd_thread;
2017-11-11 20:09:48 +00:00
int64_t gpuStart;
int64_t gpuEnd;
Int24 callstack;
int32_t child;
2017-11-11 20:09:48 +00:00
};
enum { GpuEventSize = sizeof( GpuEvent ) };
static_assert( std::is_standard_layout<GpuEvent>::value, "GpuEvent is not standard layout" );
2017-11-11 20:09:48 +00:00
2018-04-01 00:03:34 +00:00
struct MemEvent
{
int64_t TimeAlloc() const { return int64_t( _time_thread_alloc ) >> 16; }
2019-10-24 21:03:13 +00:00
void SetTimeAlloc( int64_t time ) { assert( time < (int64_t)( 1ull << 47 ) ); memcpy( ((char*)&_time_thread_alloc)+2, &time, 6 ); }
int64_t TimeFree() const { return int64_t( _time_thread_free ) >> 16; }
2019-10-24 21:03:13 +00:00
void SetTimeFree( int64_t time ) { assert( time < (int64_t)( 1ull << 47 ) ); memcpy( ((char*)&_time_thread_free)+2, &time, 6 ); }
uint16_t ThreadAlloc() const { return uint16_t( _time_thread_alloc ); }
2019-10-24 21:03:13 +00:00
void SetThreadAlloc( uint16_t thread ) { memcpy( &_time_thread_alloc, &thread, 2 ); }
uint16_t ThreadFree() const { return uint16_t( _time_thread_free ); }
2019-10-24 21:03:13 +00:00
void SetThreadFree( uint16_t thread ) { memcpy( &_time_thread_free, &thread, 2 ); }
2018-04-01 00:03:34 +00:00
uint64_t ptr;
uint64_t size;
Int24 csAlloc;
Int24 csFree;
uint64_t _time_thread_alloc;
uint64_t _time_thread_free;
2018-04-01 00:03:34 +00:00
};
enum { MemEventSize = sizeof( MemEvent ) };
static_assert( std::is_standard_layout<MemEvent>::value, "MemEvent is not standard layout" );
2018-06-19 22:25:26 +00:00
struct CallstackFrame
{
StringIdx name;
StringIdx file;
2018-06-19 22:25:26 +00:00
uint32_t line;
};
enum { CallstackFrameSize = sizeof( CallstackFrame ) };
struct CallstackFrameData
{
CallstackFrame* data;
uint8_t size;
};
enum { CallstackFrameDataSize = sizeof( CallstackFrameData ) };
// This union exploits the fact that the current implementations of x64 and arm64 do not provide
// full 64 bit address space. The high bits must be bit-extended, so 0x80... is an invalid pointer.
// This allows using the highest bit as a selector between a native pointer and a table index here.
union CallstackFrameId
{
struct
{
uint64_t idx : 63;
uint64_t sel : 1;
};
uint64_t data;
};
enum { CallstackFrameIdSize = sizeof( CallstackFrameId ) };
struct CallstackFrameTree
{
CallstackFrameId frame;
uint64_t alloc;
uint32_t count;
flat_hash_map<uint64_t, CallstackFrameTree, nohash<uint64_t>> children;
flat_hash_set<uint32_t, nohash<uint32_t>> callstacks;
};
enum { CallstackFrameTreeSize = sizeof( CallstackFrameTree ) };
2018-08-20 00:07:31 +00:00
struct CrashEvent
{
uint64_t thread = 0;
int64_t time = 0;
uint64_t message = 0;
uint32_t callstack = 0;
};
enum { CrashEventSize = sizeof( CrashEvent ) };
2019-08-12 22:13:50 +00:00
struct ContextSwitchData
{
2019-08-17 14:26:59 +00:00
enum : int8_t { NoState = 100 };
2019-08-17 15:05:29 +00:00
enum : int8_t { Wakeup = -2 };
2019-08-17 14:26:59 +00:00
2019-10-24 21:03:13 +00:00
int64_t Start() const { return int64_t( _start_cpu ) >> 16; }
void SetStart( int64_t start ) { assert( start < (int64_t)( 1ull << 47 ) ); memcpy( ((char*)&_start_cpu)+2, &start, 6 ); }
int64_t End() const { return int64_t( _end_reason_state ) >> 16; }
2019-10-24 21:03:13 +00:00
void SetEnd( int64_t end ) { assert( end < (int64_t)( 1ull << 47 ) ); memcpy( ((char*)&_end_reason_state)+2, &end, 6 ); }
uint8_t Cpu() const { return uint8_t( _start_cpu & 0xFF ); }
2019-10-24 21:03:13 +00:00
void SetCpu( uint8_t cpu ) { memcpy( &_start_cpu, &cpu, 1 ); }
int8_t Reason() const { return int8_t( (_end_reason_state >> 8) & 0xFF ); }
2019-10-24 21:03:13 +00:00
void SetReason( int8_t reason ) { memcpy( ((char*)&_end_reason_state)+1, &reason, 1 ); }
int8_t State() const { return int8_t( _end_reason_state & 0xFF ); }
2019-10-24 21:03:13 +00:00
void SetState( int8_t state ) { memcpy( &_end_reason_state, &state, 1 ); }
uint64_t _start_cpu;
uint64_t _end_reason_state;
2019-08-17 15:05:29 +00:00
int64_t wakeup;
2019-08-12 22:13:50 +00:00
};
enum { ContextSwitchDataSize = sizeof( ContextSwitchData ) };
2017-11-11 18:21:07 +00:00
2019-08-16 14:28:58 +00:00
struct ContextSwitchCpu
{
int64_t Start() const { return int64_t( _start_thread ) >> 16; }
2019-10-24 21:03:13 +00:00
void SetStart( int64_t start ) { assert( start < (int64_t)( 1ull << 47 ) ); memcpy( ((char*)&_start_thread)+2, &start, 6 ); }
int64_t End() const { return _end; }
2019-08-29 22:56:11 +00:00
void SetEnd( int64_t end ) { assert( end < (int64_t)( 1ull << 47 ) ); _end = end; }
uint16_t Thread() const { return uint16_t( _start_thread ); }
2019-10-24 21:03:13 +00:00
void SetThread( uint16_t thread ) { memcpy( &_start_thread, &thread, 2 ); }
uint64_t _start_thread;
uint64_t _end;
2019-08-16 14:28:58 +00:00
};
enum { ContextSwitchCpuSize = sizeof( ContextSwitchCpu ) };
2017-11-11 18:21:07 +00:00
struct MessageData
{
int64_t time;
StringRef ref;
2019-08-28 19:03:01 +00:00
uint16_t thread;
2019-05-10 18:21:35 +00:00
uint32_t color;
2017-11-11 18:21:07 +00:00
};
2019-08-15 12:15:40 +00:00
enum { MessageDataSize = sizeof( MessageData ) };
2019-08-15 19:42:24 +00:00
#pragma pack()
2017-11-11 18:21:07 +00:00
struct ThreadData
{
uint64_t id;
2017-11-18 00:14:16 +00:00
uint64_t count;
2017-11-11 18:21:07 +00:00
Vector<ZoneEvent*> timeline;
Vector<ZoneEvent*> stack;
2017-11-11 18:21:07 +00:00
Vector<MessageData*> messages;
2019-01-14 21:56:10 +00:00
uint32_t nextZoneId;
Vector<uint32_t> zoneIdStack;
2017-11-11 18:21:07 +00:00
};
2019-09-23 15:27:49 +00:00
struct GpuCtxThreadData
{
Vector<GpuEvent*> timeline;
Vector<GpuEvent*> stack;
};
2017-11-11 18:44:09 +00:00
struct GpuCtxData
{
int64_t timeDiff;
2017-11-13 23:48:26 +00:00
uint64_t thread;
2017-11-18 00:07:28 +00:00
uint64_t count;
2017-11-17 23:32:15 +00:00
uint8_t accuracyBits;
float period;
2019-09-23 15:27:49 +00:00
flat_hash_map<uint64_t, GpuCtxThreadData, nohash<uint64_t>> threadData;
GpuEvent* query[64*1024];
2017-11-11 18:44:09 +00:00
};
2017-11-11 18:21:07 +00:00
struct LockMap
{
struct TimeRange
{
int64_t start = std::numeric_limits<int64_t>::max();
int64_t end = std::numeric_limits<int64_t>::min();
};
int16_t srcloc;
Vector<LockEventPtr> timeline;
flat_hash_map<uint64_t, uint8_t, nohash<uint64_t>> threadMap;
2017-11-11 18:21:07 +00:00
std::vector<uint64_t> threadList;
2017-12-10 20:37:39 +00:00
LockType type;
int64_t timeAnnounce;
int64_t timeTerminate;
2017-12-10 20:37:39 +00:00
bool valid;
2019-05-12 14:17:17 +00:00
bool isContended;
TimeRange range[64];
2017-11-11 18:21:07 +00:00
};
struct LockHighlight
{
int64_t id;
int64_t begin;
int64_t end;
uint8_t thread;
bool blocked;
};
struct PlotItem
{
int64_t time;
double val;
};
2019-02-21 21:53:26 +00:00
enum class PlotType : uint8_t
{
User,
2019-02-21 21:45:39 +00:00
Memory,
SysTime
};
2017-11-11 18:21:07 +00:00
struct PlotData
{
uint64_t name;
double min;
double max;
2017-12-05 20:24:09 +00:00
Vector<PlotItem> data;
Vector<PlotItem> postpone;
2017-11-11 18:21:07 +00:00
uint64_t postponeTime;
PlotType type;
2017-11-11 18:21:07 +00:00
};
2018-04-01 00:03:34 +00:00
struct MemData
{
Vector<MemEvent> data;
2018-05-02 15:59:50 +00:00
Vector<uint64_t> frees;
flat_hash_map<uint64_t, size_t, nohash<uint64_t>> active;
2018-04-01 00:03:34 +00:00
uint64_t high = std::numeric_limits<uint64_t>::min();
uint64_t low = std::numeric_limits<uint64_t>::max();
2018-04-01 22:00:49 +00:00
uint64_t usage = 0;
PlotData* plot = nullptr;
2018-04-01 00:03:34 +00:00
};
2018-08-05 00:09:59 +00:00
struct FrameEvent
{
int64_t start;
int64_t end;
2019-06-06 19:44:48 +00:00
int32_t frameImage;
2018-08-05 00:09:59 +00:00
};
2019-08-15 12:15:40 +00:00
enum { FrameEventSize = sizeof( FrameEvent ) };
2018-08-04 17:47:09 +00:00
struct FrameData
{
uint64_t name;
2018-08-05 00:09:59 +00:00
Vector<FrameEvent> frames;
uint8_t continuous;
2019-09-16 19:31:43 +00:00
int64_t min = std::numeric_limits<int64_t>::max();
int64_t max = std::numeric_limits<int64_t>::min();
int64_t total = 0;
double sumSq = 0;
2018-08-04 17:47:09 +00:00
};
2017-11-11 18:21:07 +00:00
struct StringLocation
{
const char* ptr;
uint32_t idx;
};
struct SourceLocationHasher
{
size_t operator()( const SourceLocation* ptr ) const
{
return charutil::hash( (const char*)ptr, sizeof( SourceLocation ) );
}
typedef tracy::power_of_two_hash_policy hash_policy;
2017-11-11 18:21:07 +00:00
};
struct SourceLocationComparator
{
bool operator()( const SourceLocation* lhs, const SourceLocation* rhs ) const
{
return memcmp( lhs, rhs, sizeof( SourceLocation ) ) == 0;
}
};
2019-06-06 19:39:54 +00:00
struct FrameImage
{
const char* ptr;
2019-06-08 10:17:18 +00:00
uint32_t csz;
2019-06-06 19:39:54 +00:00
uint16_t w, h;
2019-06-11 22:55:02 +00:00
uint32_t frameRef;
2019-06-12 13:28:32 +00:00
uint8_t flip;
2019-06-06 19:39:54 +00:00
};
2019-08-15 12:15:40 +00:00
enum { FrameImageSize = sizeof( FrameImage ) };
2019-08-12 22:13:50 +00:00
struct ContextSwitch
{
Vector<ContextSwitchData> v;
int64_t runningTime = 0;
2019-08-12 22:13:50 +00:00
};
2019-08-16 14:28:58 +00:00
struct CpuData
{
Vector<ContextSwitchCpu> cs;
};
2019-08-17 23:50:49 +00:00
struct CpuThreadData
{
int64_t runningTime = 0;
uint32_t runningRegions = 0;
uint32_t migrations = 0;
};
2019-08-17 23:53:38 +00:00
enum { CpuThreadDataSize = sizeof( CpuThreadData ) };
2017-09-14 00:00:13 +00:00
}
#endif