2014-02-18 21:40:52 +02:00
|
|
|
#include "perf.h"
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <math.h>
|
|
|
|
#ifdef NANOVG_GLEW
|
|
|
|
# include <GL/glew.h>
|
|
|
|
#endif
|
|
|
|
#include <GLFW/glfw3.h>
|
|
|
|
#include "nanovg.h"
|
|
|
|
|
|
|
|
#ifdef _MSC_VER
|
|
|
|
#define snprintf _snprintf
|
2014-04-22 14:58:44 +08:00
|
|
|
#elif !defined(__MINGW32__)
|
2014-02-18 21:40:52 +02:00
|
|
|
#include <iconv.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// timer query support
|
|
|
|
#ifndef GL_ARB_timer_query
|
|
|
|
#define GL_TIME_ELAPSED 0x88BF
|
2014-04-14 17:49:21 +03:00
|
|
|
//typedef void (APIENTRY *pfnGLGETQUERYOBJECTUI64V)(GLuint id, GLenum pname, GLuint64* params);
|
|
|
|
//pfnGLGETQUERYOBJECTUI64V glGetQueryObjectui64v = 0;
|
2014-02-18 21:40:52 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
void initGPUTimer(struct GPUtimer* timer)
|
|
|
|
{
|
|
|
|
memset(timer, 0, sizeof(*timer));
|
|
|
|
|
2014-04-14 17:49:21 +03:00
|
|
|
/* timer->supported = glfwExtensionSupported("GL_ARB_timer_query");
|
2014-02-18 21:40:52 +02:00
|
|
|
if (timer->supported) {
|
|
|
|
#ifndef GL_ARB_timer_query
|
|
|
|
glGetQueryObjectui64v = (pfnGLGETQUERYOBJECTUI64V)glfwGetProcAddress("glGetQueryObjectui64v");
|
2014-04-14 17:49:21 +03:00
|
|
|
printf("glGetQueryObjectui64v=%p\n", glGetQueryObjectui64v);
|
2014-02-18 21:40:52 +02:00
|
|
|
if (!glGetQueryObjectui64v) {
|
|
|
|
timer->supported = GL_FALSE;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
glGenQueries(GPU_QUERY_COUNT, timer->queries);
|
2014-04-14 17:49:21 +03:00
|
|
|
}*/
|
2014-02-18 21:40:52 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void startGPUTimer(struct GPUtimer* timer)
|
|
|
|
{
|
|
|
|
if (!timer->supported)
|
|
|
|
return;
|
|
|
|
glBeginQuery(GL_TIME_ELAPSED, timer->queries[timer->cur % GPU_QUERY_COUNT] );
|
|
|
|
timer->cur++;
|
|
|
|
}
|
|
|
|
|
|
|
|
int stopGPUTimer(struct GPUtimer* timer, float* times, int maxTimes)
|
|
|
|
{
|
|
|
|
GLint available = 1;
|
|
|
|
int n = 0;
|
|
|
|
if (!timer->supported)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
glEndQuery(GL_TIME_ELAPSED);
|
|
|
|
while (available && timer->ret <= timer->cur) {
|
|
|
|
// check for results if there are any
|
|
|
|
glGetQueryObjectiv(timer->queries[timer->ret % GPU_QUERY_COUNT], GL_QUERY_RESULT_AVAILABLE, &available);
|
|
|
|
if (available) {
|
2014-04-14 17:49:21 +03:00
|
|
|
/* GLuint64 timeElapsed = 0;
|
2014-02-18 21:40:52 +02:00
|
|
|
glGetQueryObjectui64v(timer->queries[timer->ret % GPU_QUERY_COUNT], GL_QUERY_RESULT, &timeElapsed);
|
|
|
|
timer->ret++;
|
|
|
|
if (n < maxTimes) {
|
|
|
|
times[n] = (float)((double)timeElapsed * 1e-9);
|
|
|
|
n++;
|
2014-04-14 17:49:21 +03:00
|
|
|
}*/
|
2014-02-18 21:40:52 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return n;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-02-18 21:51:28 +02:00
|
|
|
void initGraph(struct PerfGraph* fps, int style, const char* name)
|
2014-02-18 21:40:52 +02:00
|
|
|
{
|
2014-02-18 21:51:28 +02:00
|
|
|
memset(fps, 0, sizeof(struct PerfGraph));
|
2014-02-18 21:40:52 +02:00
|
|
|
fps->style = style;
|
|
|
|
strncpy(fps->name, name, sizeof(fps->name));
|
|
|
|
fps->name[sizeof(fps->name)-1] = '\0';
|
|
|
|
}
|
|
|
|
|
2014-02-18 21:51:28 +02:00
|
|
|
void updateGraph(struct PerfGraph* fps, float frameTime)
|
2014-02-18 21:40:52 +02:00
|
|
|
{
|
2014-02-18 21:51:28 +02:00
|
|
|
fps->head = (fps->head+1) % GRAPH_HISTORY_COUNT;
|
2014-02-18 21:40:52 +02:00
|
|
|
fps->values[fps->head] = frameTime;
|
|
|
|
}
|
|
|
|
|
2014-02-18 21:51:28 +02:00
|
|
|
float getGraphAverage(struct PerfGraph* fps)
|
2014-02-18 21:40:52 +02:00
|
|
|
{
|
2014-02-18 21:51:28 +02:00
|
|
|
int i;
|
2014-02-18 21:40:52 +02:00
|
|
|
float avg = 0;
|
2014-02-18 21:51:28 +02:00
|
|
|
for (i = 0; i < GRAPH_HISTORY_COUNT; i++) {
|
|
|
|
avg += fps->values[i];
|
2014-02-18 21:40:52 +02:00
|
|
|
}
|
2014-02-18 21:51:28 +02:00
|
|
|
return avg / (float)GRAPH_HISTORY_COUNT;
|
2014-02-18 21:40:52 +02:00
|
|
|
}
|
|
|
|
|
2014-02-18 21:51:28 +02:00
|
|
|
void renderGraph(struct NVGcontext* vg, float x, float y, struct PerfGraph* fps)
|
2014-02-18 21:40:52 +02:00
|
|
|
{
|
|
|
|
int i;
|
|
|
|
float avg, w, h;
|
|
|
|
char str[64];
|
|
|
|
|
2014-02-18 21:51:28 +02:00
|
|
|
avg = getGraphAverage(fps);
|
2014-02-18 21:40:52 +02:00
|
|
|
|
|
|
|
w = 200;
|
|
|
|
h = 35;
|
|
|
|
|
|
|
|
nvgBeginPath(vg);
|
|
|
|
nvgRect(vg, x,y, w,h);
|
|
|
|
nvgFillColor(vg, nvgRGBA(0,0,0,128));
|
|
|
|
nvgFill(vg);
|
|
|
|
|
|
|
|
nvgBeginPath(vg);
|
|
|
|
nvgMoveTo(vg, x, y+h);
|
2014-02-18 21:51:28 +02:00
|
|
|
if (fps->style == GRAPH_RENDER_FPS) {
|
|
|
|
for (i = 0; i < GRAPH_HISTORY_COUNT; i++) {
|
|
|
|
float v = 1.0f / (0.00001f + fps->values[(fps->head+i) % GRAPH_HISTORY_COUNT]);
|
2014-02-18 16:53:48 -05:00
|
|
|
float vx, vy;
|
2014-02-18 21:40:52 +02:00
|
|
|
if (v > 80.0f) v = 80.0f;
|
2014-02-18 16:53:48 -05:00
|
|
|
vx = x + ((float)i/(GRAPH_HISTORY_COUNT-1)) * w;
|
|
|
|
vy = y + h - ((v / 80.0f) * h);
|
2014-02-18 21:40:52 +02:00
|
|
|
nvgLineTo(vg, vx, vy);
|
|
|
|
}
|
|
|
|
} else {
|
2014-02-18 21:51:28 +02:00
|
|
|
for (i = 0; i < GRAPH_HISTORY_COUNT; i++) {
|
|
|
|
float v = fps->values[(fps->head+i) % GRAPH_HISTORY_COUNT] * 1000.0f;
|
2014-02-18 16:53:48 -05:00
|
|
|
float vx, vy;
|
2014-02-18 21:40:52 +02:00
|
|
|
if (v > 20.0f) v = 20.0f;
|
2014-02-18 16:53:48 -05:00
|
|
|
vx = x + ((float)i/(GRAPH_HISTORY_COUNT-1)) * w;
|
|
|
|
vy = y + h - ((v / 20.0f) * h);
|
2014-02-18 21:40:52 +02:00
|
|
|
nvgLineTo(vg, vx, vy);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
nvgLineTo(vg, x+w, y+h);
|
|
|
|
nvgFillColor(vg, nvgRGBA(255,192,0,128));
|
|
|
|
nvgFill(vg);
|
|
|
|
|
|
|
|
nvgFontFace(vg, "sans");
|
|
|
|
|
|
|
|
if (fps->name[0] != '\0') {
|
|
|
|
nvgFontSize(vg, 14.0f);
|
|
|
|
nvgTextAlign(vg, NVG_ALIGN_LEFT|NVG_ALIGN_TOP);
|
|
|
|
nvgFillColor(vg, nvgRGBA(240,240,240,192));
|
|
|
|
nvgText(vg, x+3,y+1, fps->name, NULL);
|
|
|
|
}
|
|
|
|
|
2014-02-18 21:51:28 +02:00
|
|
|
if (fps->style == GRAPH_RENDER_FPS) {
|
2014-02-18 21:40:52 +02:00
|
|
|
nvgFontSize(vg, 18.0f);
|
|
|
|
nvgTextAlign(vg,NVG_ALIGN_RIGHT|NVG_ALIGN_TOP);
|
|
|
|
nvgFillColor(vg, nvgRGBA(240,240,240,255));
|
|
|
|
sprintf(str, "%.2f FPS", 1.0f / avg);
|
|
|
|
nvgText(vg, x+w-3,y+1, str, NULL);
|
|
|
|
|
|
|
|
nvgFontSize(vg, 15.0f);
|
|
|
|
nvgTextAlign(vg,NVG_ALIGN_RIGHT|NVG_ALIGN_BOTTOM);
|
|
|
|
nvgFillColor(vg, nvgRGBA(240,240,240,160));
|
|
|
|
sprintf(str, "%.2f ms", avg * 1000.0f);
|
|
|
|
nvgText(vg, x+w-3,y+h-1, str, NULL);
|
|
|
|
} else {
|
|
|
|
nvgFontSize(vg, 18.0f);
|
|
|
|
nvgTextAlign(vg,NVG_ALIGN_RIGHT|NVG_ALIGN_TOP);
|
|
|
|
nvgFillColor(vg, nvgRGBA(240,240,240,255));
|
|
|
|
sprintf(str, "%.2f ms", avg * 1000.0f);
|
|
|
|
nvgText(vg, x+w-3,y+1, str, NULL);
|
|
|
|
}
|
|
|
|
}
|