add profiling code

This commit is contained in:
Lion Kortlepel 2024-01-23 23:58:06 +01:00 committed by Lion
parent e2d7721438
commit cbc1483537
2 changed files with 93 additions and 0 deletions

52
include/Profiling.h Normal file
View File

@ -0,0 +1,52 @@
#pragma once
#include <boost/circular_buffer.hpp>
#include <boost/thread/synchronized_value.hpp>
#include <chrono>
#include <cstddef>
namespace prof {
using Duration = std::chrono::duration<double, std::milli>;
using TimePoint = std::chrono::high_resolution_clock::time_point;
/// Returns the current time.
TimePoint now();
/// Returns a sub-millisecond resolution duration between start and end.
Duration duration(const TimePoint& start, const TimePoint& end);
/// Calculates and stores the moving average over K samples of execution time data
/// for some single unit of code. Threadsafe.
struct UnitExecutionTime {
UnitExecutionTime();
/// Adds a sample to the collection, overriding the oldest sample if needed.
void add_sample(const Duration& dur);
/// Calculates the average duration over the `measurement_count()` measurements.
Duration average_duration() const;
/// Returns the number of elements the moving average is calculated over.
size_t measurement_count();
private:
boost::synchronized_value<boost::circular_buffer<Duration>> m_measurements;
};
/// Holds profiles for multiple units by name. Threadsafe.
struct UnitProfileCollection {
/// Adds a sample to the collection, overriding the oldest sample if needed.
void add_sample(const std::string& unit, const Duration& duration);
/// Calculates the average duration over the `measurement_count()` measurements.
Duration average_duration(const std::string& unit);
/// Returns the number of elements the moving average is calculated over.
size_t measurement_count(const std::string& unit);
private:
boost::synchronized_value<std::unordered_map<std::string, UnitExecutionTime>> m_map;
};
}

41
src/Profiling.cpp Normal file
View File

@ -0,0 +1,41 @@
#include "Profiling.h"
prof::Duration prof::duration(const TimePoint& start, const TimePoint& end) {
return end - start;
}
prof::TimePoint prof::now() {
return std::chrono::high_resolution_clock::now();
}
prof::Duration prof::UnitProfileCollection::average_duration(const std::string& unit) {
return m_map->operator[](unit).average_duration();
}
size_t prof::UnitProfileCollection::measurement_count(const std::string& unit) {
return m_map->operator[](unit).measurement_count();
}
void prof::UnitProfileCollection::add_sample(const std::string& unit, const Duration& duration) {
m_map->operator[](unit).add_sample(duration);
}
size_t prof::UnitExecutionTime::measurement_count() {
return m_measurements->size();
}
prof::Duration prof::UnitExecutionTime::average_duration() const {
auto measurements = m_measurements.synchronize();
Duration sum {};
for (const auto& measurement : *measurements) {
sum += measurement;
}
return sum / measurements->size();
}
void prof::UnitExecutionTime::add_sample(const Duration& dur) {
m_measurements->push_back(dur);
}
prof::UnitExecutionTime::UnitExecutionTime()
: m_measurements(boost::circular_buffer<Duration>(16)) {
}