tracy/server/TracyFileWrite.hpp
2019-09-29 21:02:01 +02:00

147 lines
3.3 KiB
C++

#ifndef __TRACYFILEWRITE_HPP__
#define __TRACYFILEWRITE_HPP__
#ifdef _MSC_VER
# pragma warning( disable: 4267 ) // conversion from don't care to whatever, possible loss of data
#endif
#include <algorithm>
#include <assert.h>
#include <stdio.h>
#include <string.h>
#include "TracyFileHeader.hpp"
#include "../common/tracy_lz4.hpp"
#include "../common/tracy_lz4hc.hpp"
#include "../common/TracyForceInline.hpp"
namespace tracy
{
class FileWrite
{
public:
enum class Compression
{
Fast,
Slow,
Extreme
};
static FileWrite* Open( const char* fn, Compression comp = Compression::Fast )
{
auto f = fopen( fn, "wb" );
return f ? new FileWrite( f, comp ) : nullptr;
}
~FileWrite()
{
if( m_offset > 0 )
{
WriteLz4Block();
}
fclose( m_file );
if( m_stream ) LZ4_freeStream( m_stream );
if( m_streamHC ) LZ4_freeStreamHC( m_streamHC );
}
tracy_force_inline void Write( const void* ptr, size_t size )
{
if( m_offset + size <= BufSize )
{
WriteSmall( ptr, size );
}
else
{
WriteBig( ptr, size );
}
}
private:
FileWrite( FILE* f, Compression comp )
: m_stream( nullptr )
, m_streamHC( nullptr )
, m_file( f )
, m_buf( m_bufData[0] )
, m_second( m_bufData[1] )
, m_offset( 0 )
{
switch( comp )
{
case Compression::Fast:
m_stream = LZ4_createStream();
break;
case Compression::Slow:
m_streamHC = LZ4_createStreamHC();
break;
case Compression::Extreme:
m_streamHC = LZ4_createStreamHC();
LZ4_resetStreamHC( m_streamHC, LZ4HC_CLEVEL_MAX );
break;
default:
assert( false );
break;
}
fwrite( Lz4Header, 1, sizeof( Lz4Header ), m_file );
}
tracy_force_inline void WriteSmall( const void* ptr, size_t size )
{
memcpy( m_buf + m_offset, ptr, size );
m_offset += size;
}
void WriteBig( const void* ptr, size_t size )
{
auto src = (const char*)ptr;
while( size > 0 )
{
const auto sz = std::min( size, BufSize - m_offset );
memcpy( m_buf + m_offset, src, sz );
m_offset += sz;
src += sz;
size -= sz;
if( m_offset == BufSize )
{
WriteLz4Block();
}
}
}
void WriteLz4Block()
{
char lz4[LZ4Size];
uint32_t sz;
if( m_stream )
{
sz = LZ4_compress_fast_continue( m_stream, m_buf, lz4, m_offset, LZ4Size, 1 );
}
else
{
sz = LZ4_compress_HC_continue( m_streamHC, m_buf, lz4, m_offset, LZ4Size );
}
fwrite( &sz, 1, sizeof( sz ), m_file );
fwrite( lz4, 1, sz, m_file );
m_offset = 0;
std::swap( m_buf, m_second );
}
enum { BufSize = 64 * 1024 };
enum { LZ4Size = LZ4_COMPRESSBOUND( BufSize ) };
LZ4_stream_t* m_stream;
LZ4_streamHC_t* m_streamHC;
FILE* m_file;
char m_bufData[2][BufSize];
char* m_buf;
char* m_second;
size_t m_offset;
};
}
#endif