192 lines
No EOL
5.2 KiB
C++
192 lines
No EOL
5.2 KiB
C++
#include <ctime>
|
|
#include <cstring>
|
|
#include <iostream>
|
|
#include <fstream>
|
|
#include <time.h>
|
|
#include <vector>
|
|
#include <unistd.h>
|
|
#include <stdexcept>
|
|
|
|
#include "chunk.hpp"
|
|
#include "event.hpp"
|
|
#include "profiler.hpp"
|
|
|
|
// #define MAC_OS
|
|
|
|
namespace GC
|
|
{
|
|
/**
|
|
* Records an event independent of a chunk.
|
|
*
|
|
* @param type The type of event to record.
|
|
*/
|
|
void Profiler::record(GCEventType type)
|
|
{
|
|
auto event = new GCEvent(type);
|
|
auto profiler = Profiler::the();
|
|
profiler->m_events.push_back(event);
|
|
}
|
|
|
|
/**
|
|
* This overload is only used with an AllocStart
|
|
* event.
|
|
*
|
|
* @param type The type of event to record.
|
|
*
|
|
* @param size The size of requested to alloc().
|
|
*/
|
|
void Profiler::record(GCEventType type, size_t size)
|
|
{
|
|
auto event = new GCEvent(type, size);
|
|
auto profiler = Profiler::the();
|
|
profiler->m_events.push_back(event);
|
|
}
|
|
|
|
/**
|
|
* Records an event related to a chunk.
|
|
*
|
|
* @param type The type of event to record.
|
|
*
|
|
* @param chunk The chunk the event is connected
|
|
* to.
|
|
*/
|
|
void Profiler::record(GCEventType type, Chunk *chunk)
|
|
{
|
|
// Create a copy of chunk to store in the profiler
|
|
// because in free() chunks are deleted and cannot
|
|
// be referenced by the profiler. These copied
|
|
// chunks are deleted by the profiler on dispose().
|
|
auto chunk_copy = new Chunk(chunk);
|
|
auto event = new GCEvent(type, chunk_copy);
|
|
auto profiler = Profiler::the();
|
|
profiler->m_events.push_back(event);
|
|
}
|
|
|
|
/**
|
|
* Prints the history of the recorded events
|
|
* to a log file in the /tests/logs folder.
|
|
*/
|
|
void Profiler::dump_trace()
|
|
{
|
|
auto profiler = Profiler::the();
|
|
auto start = profiler->m_events.begin();
|
|
auto end = profiler->m_events.end();
|
|
|
|
// File output stream
|
|
std::ofstream fstr = profiler->create_file_stream();
|
|
// Buffer for timestamp
|
|
char buffer[22];
|
|
// Time variables
|
|
std::tm *btm;
|
|
std::time_t tt;
|
|
const Chunk *chunk;
|
|
|
|
while (start != end)
|
|
{
|
|
auto event = *start++;
|
|
|
|
tt = event->get_time_stamp();
|
|
btm = std::localtime(&tt);
|
|
std::strftime(buffer, 22, "%a %T", btm);
|
|
|
|
fstr << "--------------------------------\n"
|
|
<< buffer
|
|
<< "\nEvent:\t" << event->type_to_string();
|
|
|
|
|
|
|
|
chunk = event->get_chunk();
|
|
|
|
if (event->get_type() == AllocStart)
|
|
{
|
|
fstr << "\nSize: " << event->get_size();
|
|
}
|
|
else if (chunk)
|
|
{
|
|
fstr << "\nChunk: " << chunk->m_start
|
|
<< "\n Size: " << chunk->m_size
|
|
<< "\n Mark: " << chunk->m_marked;
|
|
}
|
|
fstr << "\n";
|
|
}
|
|
fstr << "--------------------------------" << std::endl;
|
|
}
|
|
|
|
/**
|
|
* Deletes the profiler singleton and all
|
|
* the events recorded after recording
|
|
* the ProfilerDispose event and dumping
|
|
* the history to a log file.
|
|
*/
|
|
void Profiler::dispose()
|
|
{
|
|
Profiler::record(ProfilerDispose);
|
|
Profiler::dump_trace();
|
|
auto profiler = Profiler::the();
|
|
delete profiler;
|
|
}
|
|
|
|
/**
|
|
* Creates a filestream for the future
|
|
* log file to print the history to in
|
|
* dump_trace().
|
|
*
|
|
* @returns The output stream to the file.
|
|
*/
|
|
std::ofstream Profiler::create_file_stream()
|
|
{
|
|
// get current time
|
|
std::time_t tt = std::time(NULL);
|
|
std::tm *ptm = std::localtime(&tt);
|
|
|
|
// format to string
|
|
char buffer[32];
|
|
std::strftime(buffer, 32, "/log_%a_%H_%M_%S.txt", ptm);
|
|
std::string filename(buffer);
|
|
|
|
// const std::string ABS_PATH = "/home/virre/dev/systemF/org/language/src/GC/";
|
|
// // const std::string ABS_PATH = "/Users/valtermiari/Desktop/DV/Bachelors/code/language/src/GC";
|
|
// std::string fullpath = ABS_PATH + filename;
|
|
|
|
const std::string fullpath = get_log_folder() + filename;
|
|
|
|
std::ofstream fstr(fullpath);
|
|
return fstr;
|
|
}
|
|
|
|
/**
|
|
* This function retrieves the path to the folder
|
|
* of the executable to use for log files.
|
|
*
|
|
* @returns The path to the logs folder.
|
|
*
|
|
* @throws A runtime error if the call
|
|
* to readlink() fails.
|
|
*/
|
|
std::string Profiler::get_log_folder()
|
|
{
|
|
#ifndef MAC_OS
|
|
char buffer[1024];
|
|
// chars read from path
|
|
ssize_t len = readlink("/proc/self/exe", buffer, sizeof(buffer)-1);
|
|
|
|
// if readlink fails
|
|
if (len == -1)
|
|
{
|
|
throw std::runtime_error(std::string("Error: readlink failed on '/proc/self/exe/'"));
|
|
}
|
|
|
|
buffer[len] = '\0';
|
|
|
|
// convert to string for string operators
|
|
auto path = std::string(buffer);
|
|
|
|
// remove filename
|
|
size_t last_slash = path.find_last_of('/');
|
|
std::string folder = path.substr(0, last_slash);
|
|
#else
|
|
auto folder = std::string("/Users/valtermiari/Desktop/DV/Bachelors/code/language/src/GC/tests");
|
|
#endif
|
|
return folder + "/logs";
|
|
}
|
|
} |