2016-07-10 01:21:11 +03:00
|
|
|
/************************************************************************
|
|
|
|
* file name : reader.cpp
|
|
|
|
* ----------------- :
|
|
|
|
* creation time : 2016/06/19
|
|
|
|
* copyright : (c) 2016 Sergey Yagovtsev, Victor Zarubkin
|
|
|
|
* authors : Sergey Yagovtsev, Victor Zarubkin
|
|
|
|
* emails : yse.sey@gmail.com, v.s.zarubkin@gmail.com
|
|
|
|
* ----------------- :
|
|
|
|
* description : The file contains implementation of fillTreesFromFile function
|
|
|
|
* : which reads profiler file and fill profiler blocks tree.
|
|
|
|
* ----------------- :
|
|
|
|
* change log : * 2016/06/19 Sergey Yagovtsev: First fillTreesFromFile implementation.
|
|
|
|
* :
|
|
|
|
* : * 2016/06/25 Victor Zarubkin: Removed unnecessary memory allocation and copy
|
|
|
|
* : when creating and inserting blocks into the tree.
|
|
|
|
* :
|
|
|
|
* : * 2016/06/26 Victor Zarubkin: Added statistics gathering (min, max, average duration,
|
|
|
|
* : number of block calls).
|
|
|
|
* : * 2016/06/26 Victor Zarubkin, Sergey Yagovtsev: Added statistics gathering for root
|
|
|
|
* : blocks in the tree.
|
|
|
|
* :
|
|
|
|
* : * 2016/06/29 Victor Zarubkin: Added calculaton of total children number for blocks.
|
|
|
|
* :
|
|
|
|
* : * 2016/06/30 Victor Zarubkin: Added this header.
|
|
|
|
* : Added tree depth calculation.
|
|
|
|
* :
|
|
|
|
* : *
|
|
|
|
* ----------------- :
|
|
|
|
* license : TODO: add license text
|
|
|
|
************************************************************************/
|
|
|
|
|
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
|
|
|
|
|
2016-07-10 01:21:11 +03:00
|
|
|
/** \brief Simple C-string pointer with length.
|
|
|
|
|
|
|
|
It is used as base class for a key in std::unordered_map.
|
|
|
|
It is used to get better performance than std::string.
|
|
|
|
It simply stores a pointer and a length, there is no
|
|
|
|
any memory allocation and copy.
|
|
|
|
|
|
|
|
\note It is absolutely safe to store pointer because std::unordered_map,
|
|
|
|
which uses it as a key, exists only inside fillTreesFromFile function.
|
|
|
|
|
|
|
|
*/
|
2016-06-26 00:56:24 +03:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2016-07-10 01:21:11 +03:00
|
|
|
/** \brief cstring with precalculated hash.
|
|
|
|
|
|
|
|
This is used to calculate hash for C-string and to cache it
|
|
|
|
to be used in the future without recurring hash calculatoin.
|
|
|
|
|
|
|
|
\note This class is used as a key in std::unordered_map.
|
|
|
|
|
|
|
|
*/
|
2016-06-26 00:56:24 +03:00
|
|
|
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 {
|
|
|
|
|
2016-07-10 01:21:11 +03:00
|
|
|
/** \brief Simply returns precalculated hash of a C-string. */
|
2016-06-26 00:56:24 +03:00
|
|
|
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-07-10 01:21:11 +03:00
|
|
|
/** \brief Updates statistics for a profiler block.
|
|
|
|
|
|
|
|
\param _stats_map Storage of statistics for blocks.
|
|
|
|
\param _current Pointer to the current block.
|
|
|
|
\param _stats Reference to the variable where pointer to the block statistics must be written.
|
|
|
|
|
|
|
|
\note All blocks with similar name have the same pointer to statistics information.
|
|
|
|
|
|
|
|
\note As all profiler block keeps a pointer to it's statistics, all similar blocks
|
|
|
|
automatically receive statistics update.
|
|
|
|
|
|
|
|
*/
|
2016-08-04 22:38:45 +03:00
|
|
|
void update_statistics(StatsMap& _stats_map, const ::profiler::BlocksTree& _current, ::profiler::BlockStatistics*& _stats)
|
2016-06-26 02:15:39 +03:00
|
|
|
{
|
2016-08-04 22:38:45 +03:00
|
|
|
auto duration = _current.node->block()->duration();
|
|
|
|
StatsMap::key_type key(_current.node->getBlockName());
|
2016-06-26 02:15:39 +03:00
|
|
|
auto it = _stats_map.find(key);
|
|
|
|
if (it != _stats_map.end())
|
|
|
|
{
|
2016-07-10 01:21:11 +03:00
|
|
|
// Update already existing statistics
|
2016-06-26 02:15:39 +03:00
|
|
|
|
2016-07-10 01:21:11 +03:00
|
|
|
_stats = it->second; // write pointer to statistics into output (this is BlocksTree::total_statistics or BlocksTree::frame_statistics)
|
|
|
|
|
|
|
|
++_stats->calls_number; // update calls number of this block
|
|
|
|
_stats->total_duration += duration; // update summary duration of all block calls
|
2016-06-26 02:15:39 +03:00
|
|
|
|
|
|
|
//if (duration > _stats->max_duration_block->block()->duration())
|
|
|
|
if (duration > _stats->max_duration)
|
|
|
|
{
|
2016-07-10 01:21:11 +03:00
|
|
|
// update max duration
|
2016-08-04 22:38:45 +03:00
|
|
|
_stats->max_duration_block = _current.block_index;
|
2016-06-26 02:15:39 +03:00
|
|
|
_stats->max_duration = duration;
|
|
|
|
}
|
|
|
|
|
|
|
|
//if (duration < _stats->min_duration_block->block()->duration())
|
|
|
|
if (duration < _stats->min_duration)
|
|
|
|
{
|
2016-07-10 01:21:11 +03:00
|
|
|
// update min duraton
|
2016-08-04 22:38:45 +03:00
|
|
|
_stats->min_duration_block = _current.block_index;
|
2016-06-26 02:15:39 +03:00
|
|
|
_stats->min_duration = duration;
|
|
|
|
}
|
2016-07-10 01:21:11 +03:00
|
|
|
|
|
|
|
// average duration is calculated inside average_duration() method by dividing total_duration to the calls_number
|
2016-06-26 02:15:39 +03:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-07-10 01:21:11 +03:00
|
|
|
// This is first time the block appear in the file.
|
|
|
|
// Create new statistics.
|
2016-08-04 22:38:45 +03:00
|
|
|
_stats = new ::profiler::BlockStatistics(duration, _current.block_index);
|
2016-06-26 02:15:39 +03:00
|
|
|
_stats_map.insert(::std::make_pair(key, _stats));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2016-06-19 23:46:42 +03:00
|
|
|
extern "C"{
|
2016-08-03 23:00:04 +03:00
|
|
|
|
|
|
|
unsigned int fillTreesFromFile(const char* filename, ::profiler::thread_blocks_tree_t& threaded_trees, bool gather_statistics)
|
2016-06-19 23:46:42 +03:00
|
|
|
{
|
2016-08-03 23:00:04 +03:00
|
|
|
::std::ifstream inFile(filename, ::std::fstream::binary);
|
2016-06-19 23:46:42 +03:00
|
|
|
|
|
|
|
if (!inFile.is_open()){
|
2016-06-27 23:22:12 +03:00
|
|
|
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
|
|
|
|
2016-06-27 23:22:12 +03:00
|
|
|
unsigned int blocks_counter = 0;
|
2016-06-19 23:46:42 +03:00
|
|
|
|
|
|
|
while (!inFile.eof()){
|
2016-07-04 22:53:48 +03:00
|
|
|
PROFILER_BEGIN_BLOCK("Read block from file")
|
2016-06-19 23:46:42 +03:00
|
|
|
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);
|
2016-08-03 23:00:04 +03:00
|
|
|
::profiler::BaseBlockData* baseData = (::profiler::BaseBlockData*)data;
|
2016-06-19 23:46:42 +03:00
|
|
|
|
2016-08-03 23:00:04 +03:00
|
|
|
auto& root = threaded_trees[baseData->getThreadId()];
|
2016-06-19 23:46:42 +03:00
|
|
|
|
2016-08-03 23:00:04 +03:00
|
|
|
::profiler::BlocksTree tree;
|
|
|
|
tree.node = new ::profiler::SerilizedBlock(sz, data);
|
2016-08-06 14:48:01 +03:00
|
|
|
tree.block_index = blocks_counter++;
|
2016-06-19 23:46:42 +03:00
|
|
|
|
2016-08-03 23:00:04 +03:00
|
|
|
if (::profiler::BLOCK_TYPE_THREAD_SIGN == baseData->getType())
|
|
|
|
{
|
|
|
|
root.thread_name = tree.node->getBlockName();
|
|
|
|
}
|
2016-07-31 22:12:11 +03:00
|
|
|
|
2016-08-03 23:00:04 +03:00
|
|
|
if (!root.tree.children.empty())
|
2016-06-27 23:22:12 +03:00
|
|
|
{
|
2016-08-03 23:00:04 +03:00
|
|
|
auto& back = root.tree.children.back();
|
2016-06-27 23:22:12 +03:00
|
|
|
auto t1 = back.node->block()->getEnd();
|
|
|
|
auto mt0 = tree.node->block()->getBegin();
|
|
|
|
if (mt0 < t1)//parent - starts earlier than last ends
|
|
|
|
{
|
2016-08-03 23:00:04 +03:00
|
|
|
//auto lower = ::std::lower_bound(root.children.begin(), root.children.end(), tree);
|
2016-07-04 21:44:07 +03:00
|
|
|
/**/
|
2016-07-04 22:53:48 +03:00
|
|
|
PROFILER_BEGIN_BLOCK("Find children")
|
2016-08-03 23:00:04 +03:00
|
|
|
auto rlower1 = ++root.tree.children.rbegin();
|
|
|
|
for(; rlower1 != root.tree.children.rend(); ++rlower1){
|
2016-07-04 21:44:07 +03:00
|
|
|
if(mt0 > rlower1->node->block()->getBegin())
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
auto lower = rlower1.base();
|
2016-08-03 23:00:04 +03:00
|
|
|
::std::move(lower, root.tree.children.end(), ::std::back_inserter(tree.children));
|
2016-06-19 23:46:42 +03:00
|
|
|
|
2016-08-03 23:00:04 +03:00
|
|
|
root.tree.children.erase(lower, root.tree.children.end());
|
2016-06-19 23:46:42 +03:00
|
|
|
|
2016-08-03 23:00:04 +03:00
|
|
|
::profiler::timestamp_t children_duration = 0;
|
2016-06-26 02:15:39 +03:00
|
|
|
if (gather_statistics)
|
|
|
|
{
|
2016-07-04 22:53:48 +03:00
|
|
|
PROFILER_BEGIN_BLOCK("Gather statistic for frame")
|
2016-06-26 02:15:39 +03:00
|
|
|
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)
|
|
|
|
{
|
2016-08-04 22:38:45 +03:00
|
|
|
update_statistics(frame_statistics, child, child.frame_statistics);
|
2016-07-27 21:50:11 +03:00
|
|
|
|
2016-08-03 23:00:04 +03:00
|
|
|
children_duration += child.node->block()->duration();
|
|
|
|
tree.total_children_number += child.total_children_number;
|
2016-07-27 21:50:11 +03:00
|
|
|
if (tree.depth < child.depth)
|
|
|
|
tree.depth = child.depth;
|
2016-06-26 02:15:39 +03:00
|
|
|
}
|
2016-07-27 21:50:11 +03:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-08-03 23:00:04 +03:00
|
|
|
for (const auto& child : tree.children)
|
2016-07-27 21:50:11 +03:00
|
|
|
{
|
2016-08-03 23:00:04 +03:00
|
|
|
children_duration += child.node->block()->duration();
|
2016-07-27 21:50:11 +03:00
|
|
|
tree.total_children_number += child.total_children_number;
|
|
|
|
if (tree.depth < child.depth)
|
|
|
|
tree.depth = child.depth;
|
|
|
|
}
|
2016-06-26 02:15:39 +03:00
|
|
|
}
|
2016-08-03 23:00:04 +03:00
|
|
|
|
|
|
|
tree.total_children_number += static_cast<unsigned int>(tree.children.size());
|
|
|
|
++tree.depth;
|
2016-06-27 23:22:12 +03:00
|
|
|
}
|
|
|
|
}
|
2016-06-19 23:46:42 +03:00
|
|
|
|
2016-08-03 23:00:04 +03:00
|
|
|
root.tree.children.emplace_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-08-03 23:00:04 +03:00
|
|
|
PROFILER_BEGIN_BLOCK("Gather statistics")
|
|
|
|
auto& current = root.tree.children.back();
|
2016-08-04 22:38:45 +03:00
|
|
|
update_statistics(overall_statistics, current, current.total_statistics);
|
2016-06-26 00:56:24 +03:00
|
|
|
}
|
|
|
|
|
2016-06-26 02:15:39 +03:00
|
|
|
}
|
2016-07-27 21:50:11 +03:00
|
|
|
|
2016-08-03 23:00:04 +03:00
|
|
|
PROFILER_BEGIN_BLOCK("Gather statistics for roots")
|
2016-06-26 21:04:24 +03:00
|
|
|
if (gather_statistics)
|
|
|
|
{
|
2016-08-03 23:00:04 +03:00
|
|
|
for (auto& it : threaded_trees)
|
2016-06-26 21:04:24 +03:00
|
|
|
{
|
2016-08-03 23:00:04 +03:00
|
|
|
auto& root = it.second;
|
|
|
|
root.thread_id = it.first;
|
2016-07-10 01:21:11 +03:00
|
|
|
|
2016-06-26 21:04:24 +03:00
|
|
|
frame_statistics.clear();
|
2016-08-03 23:00:04 +03:00
|
|
|
for (auto& frame : root.tree.children)
|
2016-06-26 21:04:24 +03:00
|
|
|
{
|
2016-08-03 23:00:04 +03:00
|
|
|
root.tree.total_children_number += frame.total_children_number;
|
2016-08-04 22:38:45 +03:00
|
|
|
update_statistics(frame_statistics, frame, frame.frame_statistics);
|
2016-08-03 23:00:04 +03:00
|
|
|
if (root.tree.depth < frame.depth)
|
|
|
|
root.tree.depth = frame.depth;
|
2016-06-26 21:04:24 +03:00
|
|
|
}
|
2016-07-27 21:50:11 +03:00
|
|
|
|
2016-08-03 23:00:04 +03:00
|
|
|
root.tree.total_children_number += static_cast<unsigned int>(root.tree.children.size());
|
|
|
|
++root.tree.depth;
|
2016-06-26 21:04:24 +03:00
|
|
|
}
|
|
|
|
}
|
2016-06-29 20:31:17 +03:00
|
|
|
else
|
|
|
|
{
|
2016-08-03 23:00:04 +03:00
|
|
|
for (auto& it : threaded_trees)
|
2016-06-29 20:31:17 +03:00
|
|
|
{
|
2016-08-03 23:00:04 +03:00
|
|
|
auto& root = it.second;
|
|
|
|
root.thread_id = it.first;
|
2016-07-27 21:50:11 +03:00
|
|
|
|
2016-08-03 23:00:04 +03:00
|
|
|
for (auto& frame : root.tree.children)
|
|
|
|
{
|
|
|
|
root.tree.total_children_number += frame.total_children_number;
|
|
|
|
if (root.tree.depth < frame.depth)
|
|
|
|
root.tree.depth = frame.depth;
|
2016-06-29 20:31:17 +03:00
|
|
|
}
|
2016-07-27 21:50:11 +03:00
|
|
|
|
2016-08-03 23:00:04 +03:00
|
|
|
root.tree.total_children_number += static_cast<unsigned int>(root.tree.children.size());
|
|
|
|
++root.tree.depth;
|
2016-06-29 20:31:17 +03:00
|
|
|
}
|
|
|
|
}
|
2016-07-04 22:53:48 +03:00
|
|
|
PROFILER_END_BLOCK
|
2016-07-10 01:21:11 +03:00
|
|
|
// No need to delete BlockStatistics instances - they will be deleted inside BlocksTree destructors
|
2016-06-26 00:56:24 +03:00
|
|
|
|
2016-06-27 23:22:12 +03:00
|
|
|
return blocks_counter;
|
2016-06-19 23:46:42 +03:00
|
|
|
}
|
2016-08-03 23:00:04 +03:00
|
|
|
}
|