mirror of
https://github.com/dolphin-emu/dolphin.git
synced 2026-01-30 11:03:31 +00:00
Yellow squiggly lines begone! Done automatically on .cpp files through `run-clang-tidy`, with manual corrections to the mistakes. If an import is directly used, but is technically unnecessary since it's recursively imported by something else, it is *not* removed. The tool doesn't touch .h files, so I did some of them by hand while fixing errors due to old recursive imports. Not everything is removed, but the cleanup should be substantial enough. Because this done on Linux, code that isn't used on it is mostly untouched. (Hopefully no open PR is depending on these imports...)
164 lines
4.1 KiB
C++
164 lines
4.1 KiB
C++
// Copyright 2014 Dolphin Emulator Project
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
#include "Common/Profiler.h"
|
|
|
|
#include <algorithm>
|
|
#include <cmath>
|
|
#include <iomanip>
|
|
#include <ios>
|
|
#include <sstream>
|
|
|
|
#include "Common/Timer.h"
|
|
|
|
namespace Common
|
|
{
|
|
static const u32 PROFILER_FIELD_LENGTH = 8;
|
|
static const u32 PROFILER_FIELD_LENGTH_FP = PROFILER_FIELD_LENGTH + 3;
|
|
static const int PROFILER_LAZY_DELAY = 60; // in frames
|
|
|
|
std::list<Profiler*> Profiler::s_all_profilers;
|
|
std::mutex Profiler::s_mutex;
|
|
u32 Profiler::s_max_length = 0;
|
|
u64 Profiler::s_frame_time;
|
|
u64 Profiler::s_usecs_frame;
|
|
|
|
std::string Profiler::s_lazy_result;
|
|
int Profiler::s_lazy_delay = 0;
|
|
|
|
Profiler::Profiler(const std::string& name)
|
|
: m_name(name), m_usecs(0), m_usecs_min(UINT64_MAX), m_usecs_max(0), m_usecs_quad(0), m_calls(0)
|
|
{
|
|
s_max_length = std::max<u32>(s_max_length, u32(m_name.length()));
|
|
|
|
std::lock_guard<std::mutex> lk(s_mutex);
|
|
s_all_profilers.push_back(this);
|
|
}
|
|
|
|
Profiler::~Profiler()
|
|
{
|
|
std::lock_guard<std::mutex> lk(s_mutex);
|
|
s_all_profilers.remove(this);
|
|
}
|
|
|
|
bool Profiler::operator<(const Profiler& b) const
|
|
{
|
|
return m_usecs < b.m_usecs;
|
|
}
|
|
|
|
std::string Profiler::ToString()
|
|
{
|
|
if (s_lazy_delay > 0)
|
|
{
|
|
s_lazy_delay--;
|
|
return s_lazy_result;
|
|
}
|
|
s_lazy_delay = PROFILER_LAZY_DELAY - 1;
|
|
|
|
// don't write anything if no profilation is enabled
|
|
std::lock_guard<std::mutex> lk(s_mutex);
|
|
if (s_all_profilers.empty())
|
|
return "";
|
|
|
|
u64 end = Common::Timer::NowUs();
|
|
s_usecs_frame = end - s_frame_time;
|
|
s_frame_time = end;
|
|
|
|
std::ostringstream buffer;
|
|
buffer << std::setw(s_max_length) << std::left << ""
|
|
<< " ";
|
|
buffer << std::setw(PROFILER_FIELD_LENGTH) << std::right << "calls"
|
|
<< " ";
|
|
buffer << std::setw(PROFILER_FIELD_LENGTH) << std::right << "sum"
|
|
<< " ";
|
|
buffer << std::setw(PROFILER_FIELD_LENGTH_FP) << std::right << "rel"
|
|
<< " ";
|
|
buffer << std::setw(PROFILER_FIELD_LENGTH) << std::right << "min"
|
|
<< " ";
|
|
buffer << std::setw(PROFILER_FIELD_LENGTH_FP) << std::right << "avg"
|
|
<< " ";
|
|
buffer << std::setw(PROFILER_FIELD_LENGTH_FP) << std::right << "stdev"
|
|
<< " ";
|
|
buffer << std::setw(PROFILER_FIELD_LENGTH) << std::right << "max"
|
|
<< " ";
|
|
buffer << "/ usec" << std::endl;
|
|
|
|
s_all_profilers.sort([](Profiler* a, Profiler* b) { return *b < *a; });
|
|
|
|
for (auto profiler : s_all_profilers)
|
|
{
|
|
buffer << profiler->Read() << std::endl;
|
|
}
|
|
s_lazy_result = buffer.str();
|
|
return s_lazy_result;
|
|
}
|
|
|
|
void Profiler::Start(u64* time, int* depth)
|
|
{
|
|
if ((*depth)++ == 0)
|
|
{
|
|
*time = Common::Timer::NowUs();
|
|
}
|
|
}
|
|
|
|
void Profiler::Stop(u64* time, int* depth)
|
|
{
|
|
if (--(*depth) == 0)
|
|
{
|
|
u64 end = Common::Timer::NowUs();
|
|
|
|
u64 diff = end - *time;
|
|
|
|
std::lock_guard lk(m_mutex);
|
|
m_usecs += diff;
|
|
m_usecs_min = std::min(m_usecs_min, diff);
|
|
m_usecs_max = std::max(m_usecs_max, diff);
|
|
m_usecs_quad += diff * diff;
|
|
m_calls++;
|
|
}
|
|
}
|
|
|
|
std::string Profiler::Read()
|
|
{
|
|
std::lock_guard lk(m_mutex);
|
|
|
|
double avg = 0;
|
|
double stdev = 0;
|
|
double time_rel = 0;
|
|
if (m_calls)
|
|
{
|
|
avg = double(m_usecs) / m_calls;
|
|
stdev = std::sqrt(double(m_usecs_quad) / m_calls - avg * avg);
|
|
}
|
|
else
|
|
{
|
|
m_usecs_min = 0;
|
|
}
|
|
if (s_usecs_frame)
|
|
{
|
|
time_rel = double(m_usecs) * 100 / s_usecs_frame;
|
|
}
|
|
|
|
std::ostringstream buffer;
|
|
|
|
buffer << std::setw(s_max_length) << std::left << m_name << " ";
|
|
buffer << std::setw(PROFILER_FIELD_LENGTH) << std::right << m_calls << " ";
|
|
buffer << std::setw(PROFILER_FIELD_LENGTH) << std::right << m_usecs << " ";
|
|
buffer << std::setw(PROFILER_FIELD_LENGTH_FP) << std::right << time_rel << " ";
|
|
buffer << std::setw(PROFILER_FIELD_LENGTH) << std::right << m_usecs_min << " ";
|
|
buffer << std::setw(PROFILER_FIELD_LENGTH_FP) << std::right << std::fixed << std::setprecision(2)
|
|
<< avg << " ";
|
|
buffer << std::setw(PROFILER_FIELD_LENGTH_FP) << std::right << std::fixed << std::setprecision(2)
|
|
<< stdev << " ";
|
|
buffer << std::setw(PROFILER_FIELD_LENGTH) << std::right << m_usecs_max;
|
|
|
|
m_usecs = 0;
|
|
m_usecs_min = UINT64_MAX;
|
|
m_usecs_max = 0;
|
|
m_usecs_quad = 0;
|
|
m_calls = 0;
|
|
|
|
return buffer.str();
|
|
}
|
|
} // namespace Common
|