2016-09-06 22:15:50 +03:00
|
|
|
/************************************************************************
|
|
|
|
* file name : profile_manager.cpp
|
|
|
|
* ----------------- :
|
|
|
|
* creation time : 2016/02/16
|
|
|
|
* authors : Sergey Yagovtsev, Victor Zarubkin
|
|
|
|
* emails : yse.sey@gmail.com, v.s.zarubkin@gmail.com
|
|
|
|
* ----------------- :
|
|
|
|
* description : The file contains implementation of Profile manager and implement access c-function
|
|
|
|
* :
|
|
|
|
* license : Lightweight profiler library for c++
|
|
|
|
* : Copyright(C) 2016 Sergey Yagovtsev, Victor Zarubkin
|
|
|
|
* :
|
|
|
|
* : This program is free software : you can redistribute it and / or modify
|
|
|
|
* : it under the terms of the GNU General Public License as published by
|
|
|
|
* : the Free Software Foundation, either version 3 of the License, or
|
|
|
|
* : (at your option) any later version.
|
|
|
|
* :
|
|
|
|
* : This program is distributed in the hope that it will be useful,
|
|
|
|
* : but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* : MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.See the
|
|
|
|
* : GNU General Public License for more details.
|
|
|
|
* :
|
|
|
|
* : You should have received a copy of the GNU General Public License
|
|
|
|
* : along with this program.If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
************************************************************************/
|
2016-08-28 21:06:23 +03:00
|
|
|
#include "profile_manager.h"
|
|
|
|
#include "profiler/serialized_block.h"
|
2016-09-04 14:48:35 +03:00
|
|
|
#include "event_trace_win.h"
|
2016-08-28 21:06:23 +03:00
|
|
|
|
2016-09-08 21:03:05 +03:00
|
|
|
#include "profiler/easy_net.h"
|
|
|
|
|
2016-08-28 21:06:23 +03:00
|
|
|
#include <thread>
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include <fstream>
|
|
|
|
|
2016-09-06 21:52:56 +03:00
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2016-08-28 21:06:23 +03:00
|
|
|
using namespace profiler;
|
|
|
|
|
2016-09-04 14:48:35 +03:00
|
|
|
#ifdef _WIN32
|
|
|
|
extern decltype(LARGE_INTEGER::QuadPart) CPU_FREQUENCY;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
//auto& MANAGER = ProfileManager::instance();
|
|
|
|
#define MANAGER ProfileManager::instance()
|
2016-08-28 21:06:23 +03:00
|
|
|
|
2016-09-06 23:03:05 +03:00
|
|
|
extern "C" {
|
|
|
|
|
|
|
|
PROFILER_API block_id_t registerDescription(const char* _name, const char* _filename, int _line, block_type_t _block_type, color_t _color)
|
|
|
|
{
|
|
|
|
return MANAGER.addBlockDescriptor(_name, _filename, _line, _block_type, _color);
|
|
|
|
}
|
2016-08-28 21:06:23 +03:00
|
|
|
|
2016-09-06 23:03:05 +03:00
|
|
|
PROFILER_API void endBlock()
|
2016-08-28 21:06:23 +03:00
|
|
|
{
|
|
|
|
MANAGER.endBlock();
|
|
|
|
}
|
|
|
|
|
2016-09-06 23:03:05 +03:00
|
|
|
PROFILER_API void setEnabled(bool isEnable)
|
2016-08-28 21:06:23 +03:00
|
|
|
{
|
|
|
|
MANAGER.setEnabled(isEnable);
|
2016-09-05 22:11:03 +03:00
|
|
|
#ifdef _WIN32
|
2016-09-04 14:48:35 +03:00
|
|
|
if (isEnable)
|
|
|
|
EasyEventTracer::instance().enable(true);
|
|
|
|
else
|
|
|
|
EasyEventTracer::instance().disable();
|
2016-09-05 22:11:03 +03:00
|
|
|
#endif
|
2016-08-28 21:06:23 +03:00
|
|
|
}
|
2016-09-04 14:48:35 +03:00
|
|
|
|
2016-09-06 23:03:05 +03:00
|
|
|
PROFILER_API void beginBlock(Block& _block)
|
2016-09-07 21:32:14 +03:00
|
|
|
{
|
2016-08-28 21:06:23 +03:00
|
|
|
MANAGER.beginBlock(_block);
|
2016-09-07 21:32:14 +03:00
|
|
|
}
|
2016-08-28 21:06:23 +03:00
|
|
|
|
2016-09-07 21:32:14 +03:00
|
|
|
PROFILER_API uint32_t dumpBlocksToFile(const char* filename)
|
|
|
|
{
|
2016-08-28 21:06:23 +03:00
|
|
|
return MANAGER.dumpBlocksToFile(filename);
|
2016-09-07 21:32:14 +03:00
|
|
|
}
|
2016-08-28 21:06:23 +03:00
|
|
|
|
2016-09-06 23:03:05 +03:00
|
|
|
PROFILER_API const char* setThreadName(const char* name, const char* filename, const char* _funcname, int line)
|
2016-09-07 21:32:14 +03:00
|
|
|
{
|
2016-08-28 23:40:23 +03:00
|
|
|
return MANAGER.setThreadName(name, filename, _funcname, line);
|
2016-09-07 21:32:14 +03:00
|
|
|
}
|
2016-09-04 14:48:35 +03:00
|
|
|
|
2016-09-06 23:03:05 +03:00
|
|
|
PROFILER_API void setContextSwitchLogFilename(const char* name)
|
2016-09-05 22:11:03 +03:00
|
|
|
{
|
|
|
|
return MANAGER.setContextSwitchLogFilename(name);
|
|
|
|
}
|
|
|
|
|
2016-09-06 00:22:26 +03:00
|
|
|
PROFILER_API const char* getContextSwitchLogFilename()
|
2016-09-05 22:11:03 +03:00
|
|
|
{
|
|
|
|
return MANAGER.getContextSwitchLogFilename();
|
|
|
|
}
|
|
|
|
|
2016-09-08 21:03:05 +03:00
|
|
|
PROFILER_API void startListenSignalToCapture()
|
|
|
|
{
|
|
|
|
return MANAGER.startListenSignalToCapture();
|
|
|
|
}
|
|
|
|
|
|
|
|
PROFILER_API void stopListenSignalToCapture()
|
|
|
|
{
|
|
|
|
return MANAGER.stopListenSignalToCapture();
|
|
|
|
}
|
|
|
|
|
2016-08-28 21:06:23 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
SerializedBlock::SerializedBlock(const Block& block, uint16_t name_length)
|
|
|
|
: BaseBlockData(block)
|
|
|
|
{
|
|
|
|
auto pName = const_cast<char*>(name());
|
|
|
|
if (name_length) strncpy(pName, block.name(), name_length);
|
|
|
|
pName[name_length] = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
BaseBlockDescriptor::BaseBlockDescriptor(int _line, block_type_t _block_type, color_t _color)
|
|
|
|
: m_line(_line)
|
|
|
|
, m_type(_block_type)
|
|
|
|
, m_color(_color)
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
BlockDescriptor::BlockDescriptor(uint64_t& _used_mem, const char* _name, const char* _filename, int _line, block_type_t _block_type, color_t _color)
|
|
|
|
: BaseBlockDescriptor(_line, _block_type, _color)
|
|
|
|
, m_name(_name)
|
|
|
|
, m_filename(_filename)
|
|
|
|
{
|
2016-08-28 23:40:23 +03:00
|
|
|
_used_mem += sizeof(profiler::SerializedBlockDescriptor) + strlen(_name) + strlen(_filename) + 2;
|
2016-08-28 21:06:23 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2016-09-04 14:48:35 +03:00
|
|
|
void ThreadStorage::storeBlock(const profiler::Block& block)
|
|
|
|
{
|
|
|
|
auto name_length = static_cast<uint16_t>(strlen(block.name()));
|
|
|
|
auto size = static_cast<uint16_t>(sizeof(BaseBlockData) + name_length + 1);
|
|
|
|
auto data = blocks.alloc.allocate(size);
|
|
|
|
::new (static_cast<void*>(data)) SerializedBlock(block, name_length);
|
|
|
|
blocks.usedMemorySize += size;
|
|
|
|
blocks.closedList.emplace_back(reinterpret_cast<SerializedBlock*>(data));
|
|
|
|
}
|
|
|
|
|
|
|
|
void ThreadStorage::storeCSwitch(const profiler::Block& block)
|
2016-08-28 21:06:23 +03:00
|
|
|
{
|
|
|
|
auto name_length = static_cast<uint16_t>(strlen(block.name()));
|
|
|
|
auto size = static_cast<uint16_t>(sizeof(BaseBlockData) + name_length + 1);
|
2016-09-04 14:48:35 +03:00
|
|
|
auto data = sync.alloc.allocate(size);
|
2016-08-28 21:06:23 +03:00
|
|
|
::new (static_cast<void*>(data)) SerializedBlock(block, name_length);
|
2016-09-04 14:48:35 +03:00
|
|
|
sync.usedMemorySize += size;
|
|
|
|
sync.closedList.emplace_back(reinterpret_cast<SerializedBlock*>(data));
|
2016-08-28 21:06:23 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void ThreadStorage::clearClosed()
|
|
|
|
{
|
2016-09-04 14:48:35 +03:00
|
|
|
blocks.clearClosed();
|
|
|
|
sync.clearClosed();
|
2016-08-28 21:06:23 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2016-09-06 23:03:05 +03:00
|
|
|
EASY_THREAD_LOCAL static ::ThreadStorage* THREAD_STORAGE = nullptr;
|
|
|
|
|
2016-09-04 14:48:35 +03:00
|
|
|
// #ifdef _WIN32
|
|
|
|
// LPTOP_LEVEL_EXCEPTION_FILTER PREVIOUS_FILTER = NULL;
|
|
|
|
// LONG WINAPI easyTopLevelExceptionFilter(struct _EXCEPTION_POINTERS* ExceptionInfo)
|
|
|
|
// {
|
|
|
|
// std::ofstream testexp("TEST_EXP.txt", std::fstream::binary);
|
|
|
|
// testexp.write("APPLICATION CRASHED!", strlen("APPLICATION CRASHED!"));
|
|
|
|
//
|
|
|
|
// EasyEventTracer::instance().disable();
|
|
|
|
// if (PREVIOUS_FILTER)
|
|
|
|
// return PREVIOUS_FILTER(ExceptionInfo);
|
|
|
|
// return EXCEPTION_CONTINUE_SEARCH;
|
|
|
|
// }
|
|
|
|
// #endif
|
|
|
|
|
2016-08-28 21:06:23 +03:00
|
|
|
ProfileManager::ProfileManager()
|
|
|
|
{
|
2016-09-04 14:48:35 +03:00
|
|
|
// #ifdef _WIN32
|
|
|
|
// PREVIOUS_FILTER = SetUnhandledExceptionFilter(easyTopLevelExceptionFilter);
|
2016-09-08 21:03:05 +03:00
|
|
|
// #endif
|
2016-09-08 23:15:01 +03:00
|
|
|
|
|
|
|
m_stopListen = ATOMIC_VAR_INIT(false);
|
2016-08-28 21:06:23 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
ProfileManager::~ProfileManager()
|
|
|
|
{
|
2016-09-08 21:03:05 +03:00
|
|
|
stopListenSignalToCapture();
|
|
|
|
if(m_listenThread.joinable()){
|
|
|
|
m_listenThread.join();
|
|
|
|
}
|
2016-08-28 21:06:23 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
ProfileManager& ProfileManager::instance()
|
|
|
|
{
|
|
|
|
///C++11 makes possible to create Singleton without any warry about thread-safeness
|
|
|
|
///http://preshing.com/20130930/double-checked-locking-is-fixed-in-cpp11/
|
|
|
|
static ProfileManager m_profileManager;
|
|
|
|
return m_profileManager;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ProfileManager::beginBlock(Block& _block)
|
|
|
|
{
|
|
|
|
if (!m_isEnabled)
|
|
|
|
return;
|
|
|
|
|
2016-09-06 23:03:05 +03:00
|
|
|
if (THREAD_STORAGE == nullptr)
|
|
|
|
THREAD_STORAGE = &threadStorage(getCurrentThreadId());
|
2016-08-28 21:06:23 +03:00
|
|
|
|
2016-09-06 23:03:05 +03:00
|
|
|
if (!_block.isFinished())
|
|
|
|
THREAD_STORAGE->blocks.openedList.emplace(_block);
|
|
|
|
else
|
|
|
|
THREAD_STORAGE->storeBlock(_block);
|
2016-09-04 14:48:35 +03:00
|
|
|
}
|
|
|
|
|
2016-09-06 23:03:05 +03:00
|
|
|
void ProfileManager::beginContextSwitch(profiler::thread_id_t _thread_id, profiler::timestamp_t _time, profiler::block_id_t _id)
|
2016-09-04 14:48:35 +03:00
|
|
|
{
|
2016-09-06 23:03:05 +03:00
|
|
|
auto ts = findThreadStorage(_thread_id);
|
|
|
|
if (ts != nullptr)
|
|
|
|
ts->sync.openedList.emplace(_time, profiler::BLOCK_TYPE_CONTEXT_SWITCH, _id, "");
|
2016-08-28 21:06:23 +03:00
|
|
|
}
|
|
|
|
|
2016-09-06 23:03:05 +03:00
|
|
|
void ProfileManager::storeContextSwitch(profiler::thread_id_t _thread_id, profiler::timestamp_t _time, profiler::block_id_t _id)
|
2016-09-05 22:11:03 +03:00
|
|
|
{
|
2016-09-06 23:03:05 +03:00
|
|
|
auto ts = findThreadStorage(_thread_id);
|
|
|
|
if (ts != nullptr)
|
|
|
|
{
|
2016-09-05 22:11:03 +03:00
|
|
|
profiler::Block lastBlock(_time, profiler::BLOCK_TYPE_CONTEXT_SWITCH, _id, "");
|
|
|
|
lastBlock.finish(_time);
|
2016-09-06 23:03:05 +03:00
|
|
|
ts->storeCSwitch(lastBlock);
|
2016-09-05 22:11:03 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-28 21:06:23 +03:00
|
|
|
void ProfileManager::endBlock()
|
|
|
|
{
|
|
|
|
if (!m_isEnabled)
|
|
|
|
return;
|
|
|
|
|
2016-09-06 23:03:05 +03:00
|
|
|
if (THREAD_STORAGE->blocks.openedList.empty())
|
2016-08-28 21:06:23 +03:00
|
|
|
return;
|
|
|
|
|
2016-09-06 23:03:05 +03:00
|
|
|
Block& lastBlock = THREAD_STORAGE->blocks.openedList.top();
|
2016-08-28 21:06:23 +03:00
|
|
|
if (!lastBlock.isFinished())
|
|
|
|
lastBlock.finish();
|
|
|
|
|
2016-09-06 23:03:05 +03:00
|
|
|
THREAD_STORAGE->storeBlock(lastBlock);
|
|
|
|
THREAD_STORAGE->blocks.openedList.pop();
|
2016-09-04 14:48:35 +03:00
|
|
|
}
|
|
|
|
|
2016-09-06 23:03:05 +03:00
|
|
|
void ProfileManager::endContextSwitch(profiler::thread_id_t _thread_id, profiler::timestamp_t _endtime)
|
2016-09-04 14:48:35 +03:00
|
|
|
{
|
2016-09-06 23:03:05 +03:00
|
|
|
auto ts = findThreadStorage(_thread_id);
|
|
|
|
if (ts == nullptr || ts->sync.openedList.empty())
|
2016-09-04 14:48:35 +03:00
|
|
|
return;
|
|
|
|
|
2016-09-06 23:03:05 +03:00
|
|
|
Block& lastBlock = ts->sync.openedList.top();
|
2016-09-04 14:48:35 +03:00
|
|
|
lastBlock.finish(_endtime);
|
|
|
|
|
2016-09-06 23:03:05 +03:00
|
|
|
ts->storeCSwitch(lastBlock);
|
|
|
|
ts->sync.openedList.pop();
|
2016-08-28 21:06:23 +03:00
|
|
|
}
|
|
|
|
|
2016-09-08 21:03:05 +03:00
|
|
|
void ProfileManager::startListenSignalToCapture()
|
2016-08-28 21:06:23 +03:00
|
|
|
{
|
2016-09-08 21:03:05 +03:00
|
|
|
if(!m_isAlreadyListened)
|
|
|
|
{
|
2016-09-08 23:15:01 +03:00
|
|
|
m_stopListen.store(false);
|
2016-09-08 21:03:05 +03:00
|
|
|
m_listenThread = std::thread(&ProfileManager::startListen, this);
|
|
|
|
m_isAlreadyListened = true;
|
2016-09-08 23:15:01 +03:00
|
|
|
|
2016-09-08 21:03:05 +03:00
|
|
|
}
|
2016-08-28 21:06:23 +03:00
|
|
|
}
|
|
|
|
|
2016-09-08 21:03:05 +03:00
|
|
|
void ProfileManager::stopListenSignalToCapture()
|
2016-08-28 21:06:23 +03:00
|
|
|
{
|
2016-09-08 23:15:01 +03:00
|
|
|
m_stopListen.store(true);
|
|
|
|
m_isAlreadyListened = false;
|
2016-09-08 21:03:05 +03:00
|
|
|
}
|
2016-08-28 21:06:23 +03:00
|
|
|
|
2016-09-08 21:03:05 +03:00
|
|
|
void ProfileManager::setEnabled(bool isEnable)
|
|
|
|
{
|
|
|
|
m_isEnabled = isEnable;
|
|
|
|
}
|
2016-09-04 14:48:35 +03:00
|
|
|
|
2016-09-08 21:03:05 +03:00
|
|
|
//////////////////////////////////////////////////////////////////////////
|
2016-08-28 21:06:23 +03:00
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2016-09-04 14:48:35 +03:00
|
|
|
#define STORE_CSWITCHES_SEPARATELY
|
|
|
|
|
2016-09-08 21:03:05 +03:00
|
|
|
uint32_t ProfileManager::dumpBlocksToStream(StreamWriter& of)
|
2016-08-28 21:06:23 +03:00
|
|
|
{
|
2016-09-04 14:48:35 +03:00
|
|
|
const bool wasEnabled = m_isEnabled;
|
|
|
|
if (wasEnabled)
|
|
|
|
::profiler::setEnabled(false);
|
|
|
|
|
2016-09-05 22:11:03 +03:00
|
|
|
#ifndef _WIN32
|
|
|
|
uint64_t timestamp;
|
|
|
|
uint32_t thread_from, thread_to;
|
|
|
|
|
|
|
|
std::ifstream infile(m_csInfoFilename.c_str());
|
|
|
|
|
|
|
|
if(infile.is_open())
|
|
|
|
{
|
2016-09-06 23:03:05 +03:00
|
|
|
static const auto desc = addBlockDescriptor("OS.ContextSwitch", __FILE__, __LINE__, ::profiler::BLOCK_TYPE_CONTEXT_SWITCH, ::profiler::colors::White);
|
|
|
|
while (infile >> timestamp >> thread_from >> thread_to)
|
2016-09-05 22:11:03 +03:00
|
|
|
{
|
2016-09-06 23:03:05 +03:00
|
|
|
beginContextSwitch(thread_from, timestamp, desc);
|
|
|
|
endContextSwitch(thread_to, timestamp);
|
2016-09-05 22:11:03 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
2016-08-28 21:06:23 +03:00
|
|
|
uint64_t usedMemorySize = 0;
|
|
|
|
uint32_t blocks_number = 0;
|
|
|
|
for (const auto& thread_storage : m_threads)
|
|
|
|
{
|
2016-09-04 14:48:35 +03:00
|
|
|
const auto& t = thread_storage.second;
|
|
|
|
usedMemorySize += t.blocks.usedMemorySize + t.sync.usedMemorySize;
|
|
|
|
blocks_number += static_cast<uint32_t>(t.blocks.closedList.size()) + static_cast<uint32_t>(t.sync.closedList.size());
|
2016-08-28 21:06:23 +03:00
|
|
|
}
|
|
|
|
|
2016-09-04 14:48:35 +03:00
|
|
|
#ifdef _WIN32
|
|
|
|
of.write(CPU_FREQUENCY);
|
|
|
|
#else
|
|
|
|
of.write(0LL);
|
|
|
|
#endif
|
|
|
|
|
2016-08-28 21:06:23 +03:00
|
|
|
of.write(blocks_number);
|
|
|
|
of.write(usedMemorySize);
|
|
|
|
of.write(static_cast<uint32_t>(m_descriptors.size()));
|
|
|
|
of.write(m_usedMemorySize);
|
|
|
|
|
|
|
|
for (const auto& descriptor : m_descriptors)
|
|
|
|
{
|
|
|
|
const auto name_size = static_cast<uint16_t>(strlen(descriptor.name()) + 1);
|
|
|
|
const auto filename_size = static_cast<uint16_t>(strlen(descriptor.file()) + 1);
|
2016-08-28 23:40:23 +03:00
|
|
|
const auto size = static_cast<uint16_t>(sizeof(profiler::SerializedBlockDescriptor) + name_size + filename_size);
|
2016-08-28 21:06:23 +03:00
|
|
|
|
|
|
|
of.write(size);
|
|
|
|
of.write<profiler::BaseBlockDescriptor>(descriptor);
|
|
|
|
of.write(name_size);
|
|
|
|
of.write(descriptor.name(), name_size);
|
|
|
|
of.write(descriptor.file(), filename_size);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (auto& thread_storage : m_threads)
|
|
|
|
{
|
2016-09-04 14:48:35 +03:00
|
|
|
auto& t = thread_storage.second;
|
|
|
|
|
2016-08-28 21:06:23 +03:00
|
|
|
of.write(thread_storage.first);
|
2016-09-04 14:48:35 +03:00
|
|
|
#ifdef STORE_CSWITCHES_SEPARATELY
|
|
|
|
of.write(static_cast<uint32_t>(t.blocks.closedList.size()));
|
|
|
|
#else
|
|
|
|
of.write(static_cast<uint32_t>(t.blocks.closedList.size()) + static_cast<uint32_t>(t.sync.closedList.size()));
|
|
|
|
uint32_t i = 0;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
for (auto b : t.blocks.closedList)
|
|
|
|
{
|
|
|
|
#ifndef STORE_CSWITCHES_SEPARATELY
|
|
|
|
if (i < t.sync.closedList.size())
|
|
|
|
{
|
|
|
|
auto s = t.sync.closedList[i];
|
|
|
|
if (s->end() <= b->end())// || s->begin() >= b->begin())
|
|
|
|
//if (((s->end() <= b->end() && s->end() >= b->begin()) || (s->begin() >= b->begin() && s->begin() <= b->end())))
|
|
|
|
{
|
|
|
|
if (s->m_begin < b->m_begin)
|
|
|
|
s->m_begin = b->m_begin;
|
|
|
|
if (s->m_end > b->m_end)
|
|
|
|
s->m_end = b->m_end;
|
|
|
|
of.writeBlock(s);
|
|
|
|
++i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
of.writeBlock(b);
|
|
|
|
}
|
2016-08-28 21:06:23 +03:00
|
|
|
|
2016-09-04 14:48:35 +03:00
|
|
|
#ifdef STORE_CSWITCHES_SEPARATELY
|
|
|
|
of.write(static_cast<uint32_t>(t.sync.closedList.size()));
|
|
|
|
for (auto b : t.sync.closedList)
|
|
|
|
{
|
|
|
|
#else
|
|
|
|
for (; i < t.sync.closedList.size(); ++i)
|
2016-08-28 21:06:23 +03:00
|
|
|
{
|
2016-09-04 14:48:35 +03:00
|
|
|
auto b = t.sync.closedList[i];
|
|
|
|
#endif
|
|
|
|
|
|
|
|
of.writeBlock(b);
|
2016-08-28 21:06:23 +03:00
|
|
|
}
|
|
|
|
|
2016-09-04 14:48:35 +03:00
|
|
|
t.clearClosed();
|
2016-08-28 21:06:23 +03:00
|
|
|
}
|
|
|
|
|
2016-09-04 14:48:35 +03:00
|
|
|
// if (wasEnabled)
|
|
|
|
// ::profiler::setEnabled(true);
|
|
|
|
|
2016-08-28 21:06:23 +03:00
|
|
|
return blocks_number;
|
|
|
|
}
|
|
|
|
|
2016-09-08 21:03:05 +03:00
|
|
|
uint32_t ProfileManager::dumpBlocksToFile(const char* filename)
|
|
|
|
{
|
|
|
|
StreamWriter os;
|
|
|
|
auto res = dumpBlocksToStream(os);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
std::ofstream of(filename, std::fstream::binary);
|
|
|
|
of << os.stream().str();
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2016-09-06 23:03:05 +03:00
|
|
|
const char* ProfileManager::setThreadName(const char* name, const char* filename, const char* _funcname, int line)
|
2016-08-28 21:06:23 +03:00
|
|
|
{
|
2016-09-06 23:03:05 +03:00
|
|
|
if (THREAD_STORAGE == nullptr)
|
|
|
|
{
|
|
|
|
THREAD_STORAGE = &threadStorage(getCurrentThreadId());
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!THREAD_STORAGE->named)
|
|
|
|
{
|
2016-09-07 21:48:50 +03:00
|
|
|
const auto id = addBlockDescriptor(_funcname, filename, line, profiler::BLOCK_TYPE_THREAD_SIGN, profiler::colors::Black);
|
2016-09-06 23:03:05 +03:00
|
|
|
THREAD_STORAGE->storeBlock(profiler::Block(profiler::BLOCK_TYPE_THREAD_SIGN, id, name));
|
|
|
|
THREAD_STORAGE->name = name;
|
|
|
|
THREAD_STORAGE->named = true;
|
|
|
|
}
|
2016-08-28 21:06:23 +03:00
|
|
|
|
2016-09-06 23:03:05 +03:00
|
|
|
return THREAD_STORAGE->name.c_str();
|
2016-08-28 21:06:23 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2016-09-08 21:03:05 +03:00
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/socket.h>
|
|
|
|
#include <netdb.h>
|
|
|
|
#include <stdio.h>
|
2016-09-08 23:15:01 +03:00
|
|
|
#include <unistd.h>
|
|
|
|
#include <fcntl.h>
|
2016-09-08 21:03:05 +03:00
|
|
|
|
|
|
|
|
|
|
|
void error(const char *msg)
|
|
|
|
{
|
|
|
|
perror(msg);
|
|
|
|
exit(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ProfileManager::startListen()
|
|
|
|
{
|
|
|
|
int sockfd, portno, n;
|
|
|
|
struct sockaddr_in serv_addr;
|
|
|
|
struct hostent *server;
|
|
|
|
|
|
|
|
portno = 28077;
|
|
|
|
|
|
|
|
sockfd = socket(AF_INET, SOCK_STREAM, 0);
|
|
|
|
if (sockfd < 0)
|
|
|
|
error("ERROR opening socket");
|
|
|
|
server = gethostbyname("127.0.0.1");
|
|
|
|
if (server == NULL) {
|
|
|
|
fprintf(stderr,"ERROR, no such host\n");
|
|
|
|
exit(0);
|
|
|
|
}
|
|
|
|
bzero((char *) &serv_addr, sizeof(serv_addr));
|
|
|
|
serv_addr.sin_family = AF_INET;
|
|
|
|
bcopy((char *)server->h_addr,
|
|
|
|
(char *)&serv_addr.sin_addr.s_addr,
|
|
|
|
server->h_length);
|
|
|
|
serv_addr.sin_port = htons(portno);
|
2016-09-08 23:15:01 +03:00
|
|
|
|
|
|
|
struct timeval tv;
|
|
|
|
|
|
|
|
tv.tv_sec = 1; /* 30 Secs Timeout */
|
|
|
|
tv.tv_usec = 0; // Not init'ing this can cause strange errors
|
|
|
|
|
|
|
|
setsockopt(sockfd, SOL_SOCKET, SO_RCVTIMEO, (char *)&tv,sizeof(struct timeval));
|
|
|
|
|
|
|
|
while(!m_stopListen.load() && (connect(sockfd,(struct sockaddr *) &serv_addr,sizeof(serv_addr)) < 0 ) ) {}
|
|
|
|
//if (connect(sockfd,(struct sockaddr *) &serv_addr,sizeof(serv_addr)) < 0)
|
|
|
|
// error("ERROR connecting");
|
|
|
|
|
|
|
|
|
2016-09-08 21:03:05 +03:00
|
|
|
|
|
|
|
profiler::net::Message replyMessage(profiler::net::MESSAGE_TYPE_REPLY_START_CAPTURING);
|
|
|
|
|
|
|
|
char buffer[256];
|
|
|
|
bzero(buffer,256);
|
2016-09-08 23:15:01 +03:00
|
|
|
while(!m_stopListen.load())
|
2016-09-08 21:03:05 +03:00
|
|
|
{
|
|
|
|
n = read(sockfd,buffer,255);
|
|
|
|
|
|
|
|
char *buf = &buffer[0];
|
|
|
|
int bytes = n;
|
|
|
|
if(bytes > 0)
|
|
|
|
{
|
|
|
|
profiler::net::Message* message = (profiler::net::Message*)buf;
|
|
|
|
if(!message->isEasyNetMessage()){
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (message->type) {
|
|
|
|
case profiler::net::MESSAGE_TYPE_REQUEST_START_CAPTURE:
|
|
|
|
{
|
|
|
|
//printf ("RECEIVED MESSAGE_TYPE_REQUEST_START_CAPTURE\n");
|
|
|
|
profiler::setEnabled(true);
|
|
|
|
|
|
|
|
replyMessage.type = profiler::net::MESSAGE_TYPE_REPLY_START_CAPTURING;
|
|
|
|
write(sockfd,&replyMessage,sizeof(replyMessage));
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case profiler::net::MESSAGE_TYPE_REQUEST_STOP_CAPTURE:
|
|
|
|
{
|
|
|
|
//printf ("RECEIVED MESSAGE_TYPE_REQUEST_STOP_CAPTURE\n");
|
|
|
|
profiler::setEnabled(false);
|
|
|
|
|
|
|
|
replyMessage.type = profiler::net::MESSAGE_TYPE_REPLY_PREPARE_BLOCKS;
|
|
|
|
int send_bytes = write(sockfd,&replyMessage,sizeof(replyMessage));
|
|
|
|
|
|
|
|
|
|
|
|
profiler::net::DataMessage dm;
|
|
|
|
StreamWriter os;
|
|
|
|
dumpBlocksToStream(os);
|
|
|
|
dm.size = os.stream().str().length();
|
|
|
|
|
|
|
|
int packet_size = int(sizeof(dm))+int(dm.size);
|
|
|
|
|
|
|
|
char *sendbuf = new char[packet_size];
|
|
|
|
|
|
|
|
memset(sendbuf,0,packet_size);
|
|
|
|
memcpy(sendbuf,&dm,sizeof(dm));
|
|
|
|
memcpy(sendbuf + sizeof(dm),os.stream().str().c_str(),dm.size);
|
|
|
|
|
|
|
|
send_bytes = write(sockfd,sendbuf,packet_size);
|
|
|
|
|
|
|
|
delete [] sendbuf;
|
|
|
|
|
|
|
|
replyMessage.type = profiler::net::MESSAGE_TYPE_REPLY_END_SEND_BLOCKS;
|
|
|
|
send_bytes = write(sockfd,&replyMessage,sizeof(replyMessage));
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//nn_freemsg (buf);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|