ppsspp/ext/native/profiler/profiler.cpp
2015-09-06 12:24:17 -07:00

181 lines
4.2 KiB
C++

// Ultra-lightweight category profiler with history.
#include <vector>
#include <string>
#include <map>
#include <string.h>
#include "base/logging.h"
#include "base/timeutil.h"
#include "gfx_es2/draw_buffer.h"
#include "profiler/profiler.h"
#define MAX_CATEGORIES 32 // Can be any number
#define MAX_DEPTH 16 // Can be any number
#define HISTORY_SIZE 256 // Must be power of 2
#ifndef _DEBUG
// If the compiler can collapse identical strings, we don't even need the strcmp.
#define UNIFIED_CONST_STR
#endif
struct Category {
const char *name;
};
struct CategoryFrame {
CategoryFrame() {
memset(time_taken, 0, sizeof(time_taken));
}
float time_taken[MAX_CATEGORIES];
int count[MAX_CATEGORIES];
};
struct Profiler {
int frameCount;
int historyPos;
int depth;
int parentCategory[MAX_DEPTH];
double eventStart[MAX_CATEGORIES];
double curFrameStart;
};
static Profiler profiler;
static Category categories[MAX_CATEGORIES];
static CategoryFrame *history;
void internal_profiler_init() {
memset(&profiler, 0, sizeof(profiler));
for (int i = 0; i < MAX_DEPTH; i++) {
profiler.parentCategory[i] = -1;
}
history = new CategoryFrame[HISTORY_SIZE];
}
int internal_profiler_find_cat(const char *category_name) {
int i;
for (i = 0; i < MAX_CATEGORIES; i++) {
const char *catname = categories[i].name;
if (!catname)
break;
#ifdef UNIFIED_CONST_STR
if (catname == category_name) {
#else
if (!strcmp(catname, category_name)) {
#endif
return i;
}
}
if (i < MAX_CATEGORIES && category_name) {
categories[i].name = category_name;
return i;
}
return -1;
}
// Suspend, also used to prepare for leaving.
static void internal_profiler_suspend(int category, double now) {
double diff = now - profiler.eventStart[category];
history[profiler.historyPos].time_taken[category] += (float)diff;
profiler.eventStart[category] = 0.0;
}
// Resume, also used as part of entering.
static void internal_profiler_resume(int category, double now) {
profiler.eventStart[category] = now;
}
int internal_profiler_enter(const char *category_name) {
int category = internal_profiler_find_cat(category_name);
if (category == -1 || !history) {
return category;
}
if (profiler.eventStart[category] == 0.0f) {
double now = real_time_now();
int parent = profiler.parentCategory[profiler.depth];
// Temporarily suspend the parent on entering a child.
if (parent != -1) {
internal_profiler_suspend(parent, now);
}
internal_profiler_resume(category, now);
} else {
DLOG("profiler: recursive enter (%i - %s)", category, category_name);
}
profiler.depth++;
profiler.parentCategory[profiler.depth] = category;
return category;
}
void internal_profiler_leave(int category) {
if (category == -1 || !history) {
return;
}
if (category < 0 || category >= MAX_CATEGORIES) {
ELOG("Bad category index %d", category);
profiler.depth--;
return;
}
double now = real_time_now();
profiler.depth--;
if (profiler.depth < 0) {
FLOG("Profiler enter/leave mismatch!");
}
int parent = profiler.parentCategory[profiler.depth];
// When there's recursion, we don't suspend or resume.
if (parent != category) {
internal_profiler_suspend(category, now);
history[profiler.historyPos].count[category]++;
if (parent != -1) {
// Resume tracking the parent.
internal_profiler_resume(parent, now);
}
}
}
void internal_profiler_end_frame() {
if (profiler.depth != 0) {
FLOG("Can't be inside a profiler scope at end of frame!");
}
profiler.curFrameStart = real_time_now();
profiler.historyPos++;
profiler.historyPos &= (HISTORY_SIZE - 1);
memset(&history[profiler.historyPos], 0, sizeof(history[profiler.historyPos]));
}
const char *Profiler_GetCategoryName(int i) {
return i >= 0 ? categories[i].name : "N/A";
}
int Profiler_GetHistoryLength() {
return HISTORY_SIZE;
}
int Profiler_GetNumCategories() {
for (int i = 0; i < MAX_CATEGORIES; i++) {
if (!categories[i].name)
return i;
}
return 0;
}
void Profiler_GetHistory(int category, float *data, int count) {
for (int i = 0; i < HISTORY_SIZE; i++) {
int x = i - count + profiler.historyPos + 1;
while (x < 0)
x += HISTORY_SIZE;
while (x >= HISTORY_SIZE)
x -= HISTORY_SIZE;
data[i] = history[x].time_taken[category];
}
}