tracy/server/TracyVector.hpp

358 lines
9.7 KiB
C++
Raw Normal View History

#ifndef __TRACYVECTOR_HPP__
#define __TRACYVECTOR_HPP__
2019-03-26 22:02:32 +00:00
#include <algorithm>
#include <assert.h>
#include <limits>
#include <stdint.h>
#include <stdlib.h>
#include <type_traits>
#include "../public/common/TracyForceInline.hpp"
2017-09-16 23:22:46 +00:00
#include "TracyMemory.hpp"
#include "TracyPopcnt.hpp"
2019-11-02 14:39:52 +00:00
#include "TracyShortPtr.hpp"
#include "TracySlab.hpp"
2017-09-16 23:22:46 +00:00
//#define TRACY_VECTOR_DEBUG
namespace tracy
{
#pragma pack( push, 1 )
template<typename T>
class Vector
{
2019-11-09 22:06:51 +00:00
constexpr uint8_t MaxCapacity() { return 0x7F; }
public:
using iterator = T*;
using const_iterator = const T*;
2019-10-29 00:32:09 +00:00
tracy_force_inline Vector()
{
2020-08-15 00:14:29 +00:00
memset( (char*)this, 0, sizeof( Vector<T> ) );
}
2017-09-22 23:10:48 +00:00
Vector( const Vector& ) = delete;
2019-10-29 00:32:09 +00:00
tracy_force_inline Vector( Vector&& src ) noexcept
2017-09-22 23:10:48 +00:00
{
2020-08-15 00:14:29 +00:00
memcpy( (char*)this, &src, sizeof( Vector<T> ) );
memset( (char*)&src, 0, sizeof( Vector<T> ) );
2017-09-22 23:10:48 +00:00
}
2019-10-29 00:32:09 +00:00
tracy_force_inline Vector( const T& value )
: m_ptr( (T*)malloc( sizeof( T ) ) )
2018-07-22 14:01:58 +00:00
, m_size( 1 )
, m_capacity( 0 )
2019-11-09 22:06:51 +00:00
, m_magic( 0 )
2018-07-22 14:01:58 +00:00
{
memUsage += sizeof( T );
new(m_ptr) T( value );
2018-07-22 14:01:58 +00:00
}
2019-10-29 00:32:09 +00:00
tracy_force_inline ~Vector()
{
2019-11-10 16:54:50 +00:00
if( m_capacity != MaxCapacity() && m_ptr )
{
memUsage -= Capacity() * sizeof( T );
free( m_ptr );
}
}
2017-09-22 23:10:48 +00:00
Vector& operator=( const Vector& ) = delete;
2019-10-29 00:32:09 +00:00
tracy_force_inline Vector& operator=( Vector&& src ) noexcept
2017-09-22 23:10:48 +00:00
{
2019-11-10 16:54:50 +00:00
if( m_capacity != MaxCapacity() && m_ptr )
{
memUsage -= Capacity() * sizeof( T );
free( m_ptr );
}
2020-08-15 00:14:29 +00:00
memcpy( (char*)this, &src, sizeof( Vector<T> ) );
memset( (char*)&src, 0, sizeof( Vector<T> ) );
2017-09-22 23:10:48 +00:00
return *this;
}
2019-10-29 00:32:09 +00:00
tracy_force_inline void swap( Vector& other )
2019-03-26 22:02:32 +00:00
{
uint8_t tmp[sizeof( Vector<T> )];
2020-08-15 00:14:29 +00:00
memcpy( (char*)tmp, &other, sizeof( Vector<T> ) );
memcpy( (char*)&other, this, sizeof( Vector<T> ) );
memcpy( (char*)this, tmp, sizeof( Vector<T> ) );
2019-03-26 22:02:32 +00:00
}
2018-05-02 16:27:37 +00:00
tracy_force_inline bool empty() const { return m_size == 0; }
tracy_force_inline size_t size() const { return m_size; }
tracy_force_inline void set_size( size_t sz ) { assert( m_capacity != MaxCapacity() ); m_size = sz; }
2018-04-30 11:56:58 +00:00
2018-05-02 16:27:37 +00:00
tracy_force_inline T* data() { return m_ptr; }
tracy_force_inline const T* data() const { return m_ptr; };
2018-05-02 16:27:37 +00:00
tracy_force_inline T* begin() { return m_ptr; }
tracy_force_inline const T* begin() const { return m_ptr; }
tracy_force_inline T* end() { return m_ptr + m_size; }
tracy_force_inline const T* end() const { return m_ptr + m_size; }
2018-05-02 16:27:37 +00:00
tracy_force_inline T& front() { assert( m_size > 0 ); return m_ptr[0]; }
tracy_force_inline const T& front() const { assert( m_size > 0 ); return m_ptr[0]; }
2017-10-19 18:34:07 +00:00
2018-05-02 16:27:37 +00:00
tracy_force_inline T& back() { assert( m_size > 0 ); return m_ptr[m_size - 1]; }
tracy_force_inline const T& back() const { assert( m_size > 0 ); return m_ptr[m_size - 1]; }
2018-05-02 16:27:37 +00:00
tracy_force_inline T& operator[]( size_t idx ) { return m_ptr[idx]; }
tracy_force_inline const T& operator[]( size_t idx ) const { return m_ptr[idx]; }
2017-09-16 23:05:47 +00:00
2018-05-02 16:27:37 +00:00
tracy_force_inline void push_back( const T& v )
{
assert( m_capacity != MaxCapacity() );
if( m_size == Capacity() ) AllocMore();
new(m_ptr+m_size) T( v );
m_size++;
}
2018-05-02 16:27:37 +00:00
tracy_force_inline void push_back_non_empty( const T& v )
{
assert( m_capacity != MaxCapacity() );
assert( m_ptr );
if( m_size == CapacityNoNullptrCheck() ) AllocMore();
new(m_ptr+m_size) T( v );
m_size++;
}
2018-05-02 16:27:37 +00:00
tracy_force_inline void push_back_no_space_check( const T& v )
2018-03-15 20:32:06 +00:00
{
assert( m_capacity != MaxCapacity() );
assert( m_size < Capacity() );
new(m_ptr+m_size) T( v );
m_size++;
2018-03-15 20:32:06 +00:00
}
2018-05-02 16:27:37 +00:00
tracy_force_inline void push_back( T&& v )
{
assert( m_capacity != MaxCapacity() );
if( m_size == Capacity() ) AllocMore();
new(m_ptr+m_size) T( std::move( v ) );
m_size++;
}
2018-05-02 16:27:37 +00:00
tracy_force_inline T& push_next()
{
assert( m_capacity != MaxCapacity() );
if( m_size == Capacity() ) AllocMore();
new(m_ptr+m_size) T();
return m_ptr[m_size++];
}
tracy_force_inline T& push_next_non_empty()
{
assert( m_capacity != MaxCapacity() );
assert( m_ptr );
if( m_size == CapacityNoNullptrCheck() ) AllocMore();
new(m_ptr+m_size) T();
return m_ptr[m_size++];
}
2018-05-02 16:27:37 +00:00
tracy_force_inline T& push_next_no_space_check()
2018-04-29 11:39:06 +00:00
{
assert( m_capacity != MaxCapacity() );
2018-04-29 11:39:06 +00:00
assert( m_size < Capacity() );
new(m_ptr+m_size) T();
2018-04-29 11:39:06 +00:00
return m_ptr[m_size++];
}
2017-09-25 21:01:22 +00:00
T* insert( T* it, const T& v )
2017-09-16 23:05:47 +00:00
{
assert( m_capacity != MaxCapacity() );
2017-09-16 23:05:47 +00:00
assert( it >= m_ptr && it <= m_ptr + m_size );
2017-09-25 21:01:22 +00:00
const auto dist = it - m_ptr;
if( m_size == Capacity() ) AllocMore();
2017-09-25 21:01:22 +00:00
if( dist != m_size ) memmove( m_ptr + dist + 1, m_ptr + dist, ( m_size - dist ) * sizeof( T ) );
2017-09-16 23:05:47 +00:00
m_size++;
new(m_ptr+dist) T( v );
2017-09-25 21:01:22 +00:00
m_ptr[dist] = v;
return m_ptr + dist;
2017-09-16 23:05:47 +00:00
}
2017-09-25 21:01:22 +00:00
T* insert( T* it, T&& v )
2017-09-16 23:05:47 +00:00
{
assert( m_capacity != MaxCapacity() );
2017-09-16 23:05:47 +00:00
assert( it >= m_ptr && it <= m_ptr + m_size );
2017-09-25 21:01:22 +00:00
const auto dist = it - m_ptr;
if( m_size == Capacity() ) AllocMore();
2017-09-25 21:01:22 +00:00
if( dist != m_size ) memmove( m_ptr + dist + 1, m_ptr + dist, ( m_size - dist ) * sizeof( T ) );
2017-09-16 23:05:47 +00:00
m_size++;
new(m_ptr+dist) T( std::move( v ) );
2017-09-25 21:01:22 +00:00
return m_ptr + dist;
2017-09-16 23:05:47 +00:00
}
void insert( T* it, T* begin, T* end )
{
assert( m_capacity != MaxCapacity() );
assert( it >= m_ptr && it <= m_ptr + m_size );
const auto sz = end - begin;
const auto dist = it - m_ptr;
while( m_size + sz > Capacity() ) AllocMore();
if( dist != m_size ) memmove( m_ptr + dist + sz, m_ptr + dist, ( m_size - dist ) * sizeof( T ) );
m_size += sz;
memcpy( m_ptr + dist, begin, sz * sizeof( T ) );
}
T* erase( T* it )
{
assert( m_capacity != MaxCapacity() );
assert( it >= m_ptr && it <= m_ptr + m_size );
m_size--;
2019-07-11 18:20:50 +00:00
memmove( it, it+1, ( m_size - ( it - m_ptr ) ) * sizeof( T ) );
return it;
}
2017-09-25 21:11:56 +00:00
T* erase( T* begin, T* end )
{
assert( m_capacity != MaxCapacity() );
2017-09-25 21:11:56 +00:00
assert( begin >= m_ptr && begin <= m_ptr + m_size );
assert( end >= m_ptr && end <= m_ptr + m_size );
assert( begin <= end );
const auto dist = end - begin;
if( dist > 0 )
{
memmove( begin, end, ( m_size - ( end - m_ptr ) ) * sizeof( T ) );
m_size -= dist;
}
return begin;
2017-11-11 19:51:03 +00:00
}
2018-05-02 16:27:37 +00:00
tracy_force_inline void pop_back()
2017-11-11 19:51:03 +00:00
{
assert( m_capacity != MaxCapacity() );
2017-11-11 19:51:03 +00:00
assert( m_size > 0 );
m_size--;
2017-09-25 21:11:56 +00:00
}
2018-05-02 16:27:37 +00:00
tracy_force_inline T& back_and_pop()
2017-11-19 18:17:54 +00:00
{
assert( m_capacity != MaxCapacity() );
2017-11-19 18:17:54 +00:00
assert( m_size > 0 );
m_size--;
return m_ptr[m_size];
}
2018-05-02 16:27:37 +00:00
tracy_force_inline void reserve( size_t cap )
2017-09-30 23:51:29 +00:00
{
2017-11-15 20:26:58 +00:00
if( cap == 0 || cap <= Capacity() ) return;
reserve_non_zero( cap );
}
void reserve_non_zero( size_t cap )
{
assert( m_capacity != MaxCapacity() );
cap--;
cap |= cap >> 1;
cap |= cap >> 2;
cap |= cap >> 4;
cap |= cap >> 8;
cap |= cap >> 16;
cap = TracyCountBits( cap );
memUsage += ( ( 1 << cap ) - Capacity() ) * sizeof( T );
2017-09-30 23:51:29 +00:00
m_capacity = cap;
Realloc();
}
2018-05-02 16:27:37 +00:00
tracy_force_inline void reserve_and_use( size_t sz )
{
assert( m_capacity != MaxCapacity() );
reserve( sz );
m_size = sz;
}
template<size_t U>
tracy_force_inline void reserve_exact( uint32_t sz, Slab<U>& slab )
{
assert( !m_ptr );
m_capacity = MaxCapacity();
m_size = sz;
m_ptr = (T*)slab.AllocBig( sizeof( T ) * sz );
}
2018-05-02 16:27:37 +00:00
tracy_force_inline void clear()
2017-10-19 21:25:49 +00:00
{
assert( m_capacity != MaxCapacity() );
2017-10-19 21:25:49 +00:00
m_size = 0;
}
2019-11-09 22:06:51 +00:00
tracy_force_inline bool is_magic() const { return m_magic; }
tracy_force_inline void set_magic() { assert( !m_magic ); m_magic = 1; }
private:
tracy_no_inline void AllocMore()
{
assert( m_capacity != MaxCapacity() );
2017-11-15 20:26:58 +00:00
if( m_ptr == nullptr )
{
memUsage += sizeof( T );
m_ptr = (T*)malloc( sizeof( T ) );
m_capacity = 0;
2017-11-15 20:26:58 +00:00
}
else
{
memUsage += Capacity() * sizeof( T );
2017-11-15 20:26:58 +00:00
m_capacity++;
Realloc();
}
2017-09-30 23:51:29 +00:00
}
void Realloc()
{
T* ptr = (T*)malloc( sizeof( T ) * CapacityNoNullptrCheck() );
2017-09-28 18:47:57 +00:00
if( m_size != 0 )
{
if( std::is_trivially_copyable<T>() )
{
2020-08-15 00:14:29 +00:00
memcpy( (char*)ptr, m_ptr, m_size * sizeof( T ) );
}
else
{
for( uint32_t i=0; i<m_size; i++ )
{
new(ptr+i) T( std::move( m_ptr[i] ) );
}
}
free( m_ptr );
2017-09-28 18:47:57 +00:00
}
m_ptr = ptr;
}
2018-05-02 16:27:37 +00:00
tracy_force_inline uint32_t Capacity() const
2017-11-15 20:26:58 +00:00
{
return m_ptr == nullptr ? 0 : 1 << m_capacity;
}
2018-05-02 16:27:37 +00:00
tracy_force_inline uint32_t CapacityNoNullptrCheck() const
{
return 1 << m_capacity;
}
#ifdef TRACY_VECTOR_DEBUG
T* m_ptr;
#else
2019-11-02 14:39:52 +00:00
short_ptr<T> m_ptr;
#endif
uint32_t m_size;
2019-11-09 22:06:51 +00:00
uint8_t m_capacity : 7;
uint8_t m_magic : 1;
};
2019-11-09 22:04:42 +00:00
template<typename T> struct VectorAdapterDirect { const T& operator()( const T& it ) const { return it; } };
template<typename T> struct VectorAdapterPointer { const T& operator()( const short_ptr<T>& it ) const { return *it; } };
#pragma pack( pop )
2019-11-02 14:39:52 +00:00
enum { VectorSize = sizeof( Vector<int> ) };
}
#endif