0
0
mirror of https://github.com/yse/easy_profiler.git synced 2025-01-01 20:06:16 +08:00
easy_profiler/src/reader.cpp

230 lines
5.9 KiB
C++
Raw Normal View History

2016-06-19 23:46:42 +03:00
#include "profiler/reader.h"
#include <fstream>
#include <iterator>
2016-06-26 00:56:24 +03:00
#include <algorithm>
#include <unordered_map>
//////////////////////////////////////////////////////////////////////////
#ifdef _WIN32
class cstring
{
protected:
const char* str;
size_t str_len;
public:
cstring(const char* _str) : str(_str), str_len(strlen(_str))
{
}
cstring(const cstring& _other) : str(_other.str), str_len(_other.str_len)
{
}
inline bool operator == (const cstring& _other) const
{
return str_len == _other.str_len && !strncmp(str, _other.str, str_len);
}
inline bool operator != (const cstring& _other) const
{
return !operator == (_other);
}
inline bool operator < (const cstring& _other) const
{
if (str_len == _other.str_len)
{
return strncmp(str, _other.str, str_len) < 0;
}
return str_len < _other.str_len;
}
};
class hashed_cstr : public cstring
{
typedef cstring Parent;
public:
size_t str_hash;
hashed_cstr(const char* _str) : Parent(_str), str_hash(0)
{
str_hash = ::std::_Hash_seq((const unsigned char *)str, str_len);
}
hashed_cstr(const hashed_cstr& _other) : Parent(_other), str_hash(_other.str_hash)
{
}
inline bool operator == (const hashed_cstr& _other) const
{
return str_hash == _other.str_hash && Parent::operator == (_other);
}
inline bool operator != (const hashed_cstr& _other) const
{
return !operator == (_other);
}
};
namespace std {
template <>
struct hash<hashed_cstr>
{
inline size_t operator () (const hashed_cstr& _str) const
{
return _str.str_hash;
}
};
}
typedef ::std::unordered_map<hashed_cstr, ::profiler::BlockStatistics*> StatsMap;
#else
// TODO: optimize for Linux too
#include <string>
typedef ::std::unordered_map<::std::string, ::profiler::BlockStatistics*> StatsMap;
#endif
//////////////////////////////////////////////////////////////////////////
2016-06-19 23:46:42 +03:00
2016-06-26 02:15:39 +03:00
void update_statistics(StatsMap& _stats_map, ::profiler::SerilizedBlock* _current, ::profiler::BlockStatistics*& _stats)
{
auto duration = _current->block()->duration();
StatsMap::key_type key(_current->getBlockName());
auto it = _stats_map.find(key);
if (it != _stats_map.end())
{
_stats = it->second;
++_stats->calls_number;
_stats->total_duration += duration;
//if (duration > _stats->max_duration_block->block()->duration())
if (duration > _stats->max_duration)
{
_stats->max_duration_block = _current;
_stats->max_duration = duration;
}
//if (duration < _stats->min_duration_block->block()->duration())
if (duration < _stats->min_duration)
{
_stats->min_duration_block = _current;
_stats->min_duration = duration;
}
}
else
{
_stats = new ::profiler::BlockStatistics(duration, _current);
_stats_map.insert(::std::make_pair(key, _stats));
}
}
//////////////////////////////////////////////////////////////////////////
2016-06-19 23:46:42 +03:00
extern "C"{
unsigned int fillTreesFromFile(const char* filename, thread_blocks_tree_t& threaded_trees, bool gather_statistics)
2016-06-19 23:46:42 +03:00
{
std::ifstream inFile(filename, std::fstream::binary);
if (!inFile.is_open()){
return 0;
2016-06-19 23:46:42 +03:00
}
2016-06-26 02:15:39 +03:00
StatsMap overall_statistics, frame_statistics;
2016-06-26 00:56:24 +03:00
unsigned int blocks_counter = 0;
2016-06-19 23:46:42 +03:00
while (!inFile.eof()){
uint16_t sz = 0;
inFile.read((char*)&sz, sizeof(sz));
if (sz == 0)
{
inFile.read((char*)&sz, sizeof(sz));
continue;
}
2016-06-26 00:56:24 +03:00
// TODO: use salloc::shared_allocator for allocation/deallocation safety
2016-06-19 23:46:42 +03:00
char* data = new char[sz];
inFile.read((char*)&data[0], sz);
profiler::BaseBlockData* baseData = (profiler::BaseBlockData*)data;
BlocksTree& root = threaded_trees[baseData->getThreadId()];
BlocksTree tree;
tree.node = new profiler::SerilizedBlock(sz, data);
++blocks_counter;
2016-06-19 23:46:42 +03:00
if (!root.children.empty())
{
BlocksTree& back = root.children.back();
auto t1 = back.node->block()->getEnd();
auto mt0 = tree.node->block()->getBegin();
if (mt0 < t1)//parent - starts earlier than last ends
{
auto lower = std::lower_bound(root.children.begin(), root.children.end(), tree);
2016-06-19 23:46:42 +03:00
std::move(lower, root.children.end(), std::back_inserter(tree.children));
2016-06-19 23:46:42 +03:00
root.children.erase(lower, root.children.end());
2016-06-19 23:46:42 +03:00
2016-06-26 02:15:39 +03:00
if (gather_statistics)
{
frame_statistics.clear();
//frame_statistics.reserve(tree.children.size()); // this gives slow-down on Windows
//frame_statistics.reserve(tree.children.size() * 2); // this gives no speed-up on Windows
// TODO: check this behavior on Linux
for (auto& child : tree.children)
{
update_statistics(frame_statistics, child.node, child.frame_statistics);
}
}
}
}
2016-06-19 23:46:42 +03:00
root.children.push_back(std::move(tree));
2016-06-19 23:46:42 +03:00
2016-06-25 23:10:05 +03:00
//delete[] data;
2016-06-19 23:46:42 +03:00
2016-06-26 21:04:24 +03:00
2016-06-26 00:56:24 +03:00
2016-06-26 02:15:39 +03:00
if (gather_statistics)
2016-06-26 00:56:24 +03:00
{
2016-06-26 02:15:39 +03:00
BlocksTree& current = root.children.back();
update_statistics(overall_statistics, current.node, current.total_statistics);
2016-06-26 00:56:24 +03:00
}
2016-06-26 02:15:39 +03:00
}
2016-06-26 21:04:24 +03:00
if (gather_statistics)
{
for (auto& threaded_tree : threaded_trees)
{
frame_statistics.clear();
for (auto& root : threaded_tree.second.children)
{
update_statistics(frame_statistics, root.node, root.frame_statistics);
}
}
}
2016-06-26 02:15:39 +03:00
// No need to delete BlockStatistics instances - they will be deleted on BlocksTree destructors
2016-06-26 00:56:24 +03:00
return blocks_counter;
2016-06-19 23:46:42 +03:00
}
}