SRB2/src/m_perfstats.c
2023-07-27 15:38:42 +02:00

842 lines
23 KiB
C

// SONIC ROBO BLAST 2
//-----------------------------------------------------------------------------
// Copyright (C) 2020-2023 by Sonic Team Junior.
//
// This program is free software distributed under the
// terms of the GNU General Public License, version 2.
// See the 'LICENSE' file for more details.
//-----------------------------------------------------------------------------
/// \file m_perfstats.c
/// \brief Performance measurement tools.
#include "m_perfstats.h"
#include "v_video.h"
#include "i_video.h"
#include "netcode/d_netcmd.h"
#include "r_main.h"
#include "i_system.h"
#include "z_zone.h"
#include "p_local.h"
#include "r_fps.h"
#ifdef HWRENDER
#include "hardware/hw_main.h"
#endif
struct perfstatrow;
typedef struct perfstatrow perfstatrow_t;
struct perfstatrow {
const char * lores_label;
const char * hires_label;
ps_metric_t * metric;
UINT8 flags;
};
// perfstatrow_t flags
#define PS_TIME 1 // metric measures time (uses precise_t instead of INT32)
#define PS_LEVEL 2 // metric is valid only when a level is active
#define PS_SW 4 // metric is valid only in software mode
#define PS_HW 8 // metric is valid only in opengl mode
#define PS_BATCHING 16 // metric is valid only when opengl batching is active
#define PS_HIDE_ZERO 32 // hide metric if its value is zero
static ps_metric_t ps_frametime = {0};
ps_metric_t ps_tictime = {0};
ps_metric_t ps_playerthink_time = {0};
ps_metric_t ps_thinkertime = {0};
ps_metric_t ps_thlist_times[NUM_THINKERLISTS];
static ps_metric_t ps_thinkercount = {0};
static ps_metric_t ps_polythcount = {0};
static ps_metric_t ps_mainthcount = {0};
static ps_metric_t ps_mobjcount = {0};
static ps_metric_t ps_regularcount = {0};
static ps_metric_t ps_scenerycount = {0};
static ps_metric_t ps_nothinkcount = {0};
static ps_metric_t ps_dynslopethcount = {0};
static ps_metric_t ps_precipcount = {0};
static ps_metric_t ps_removecount = {0};
ps_metric_t ps_checkposition_calls = {0};
ps_metric_t ps_lua_thinkframe_time = {0};
ps_metric_t ps_lua_mobjhooks = {0};
ps_metric_t ps_otherlogictime = {0};
// Columns for perfstats pages.
// Position on screen is determined separately in the drawing functions.
// New columns must also be added to the drawing and update functions.
// Drawing functions: PS_DrawRenderStats, PS_DrawGameLogicStats, etc.
// Update functions:
// - PS_UpdateFrameStats for frame-dependent values
// - PS_UpdateTickStats for tick-dependent values
// Rendering stats columns
perfstatrow_t rendertime_rows[] = {
{"frmtime", "Frame time: ", &ps_frametime, PS_TIME},
{"drwtime", "3d rendering: ", &ps_rendercalltime, PS_TIME|PS_LEVEL},
#ifdef HWRENDER
{" skybox ", " Skybox render: ", &ps_hw_skyboxtime, PS_TIME|PS_LEVEL|PS_HW},
{" bsptime", " RenderBSPNode: ", &ps_bsptime, PS_TIME|PS_LEVEL|PS_HW},
{" batsort", " Batch sort: ", &ps_hw_batchsorttime, PS_TIME|PS_LEVEL|PS_HW|PS_BATCHING},
{" batdraw", " Batch render: ", &ps_hw_batchdrawtime, PS_TIME|PS_LEVEL|PS_HW|PS_BATCHING},
{" sprsort", " Sprite sort: ", &ps_hw_spritesorttime, PS_TIME|PS_LEVEL|PS_HW},
{" sprdraw", " Sprite render: ", &ps_hw_spritedrawtime, PS_TIME|PS_LEVEL|PS_HW},
{" nodesrt", " Drwnode sort: ", &ps_hw_nodesorttime, PS_TIME|PS_LEVEL|PS_HW},
{" nodedrw", " Drwnode render:", &ps_hw_nodedrawtime, PS_TIME|PS_LEVEL|PS_HW},
{" other ", " Other: ", &ps_otherrendertime, PS_TIME|PS_LEVEL|PS_HW},
#endif
{" bsptime", " RenderBSPNode: ", &ps_bsptime, PS_TIME|PS_LEVEL|PS_SW},
{" sprclip", " R_ClipSprites: ", &ps_sw_spritecliptime, PS_TIME|PS_LEVEL|PS_SW},
{" portals", " Portals+Skybox:", &ps_sw_portaltime, PS_TIME|PS_LEVEL|PS_SW},
{" planes ", " R_DrawPlanes: ", &ps_sw_planetime, PS_TIME|PS_LEVEL|PS_SW},
{" masked ", " R_DrawMasked: ", &ps_sw_maskedtime, PS_TIME|PS_LEVEL|PS_SW},
{" other ", " Other: ", &ps_otherrendertime, PS_TIME|PS_LEVEL|PS_SW},
{"ui ", "UI render: ", &ps_uitime, PS_TIME},
{"finupdt", "I_FinishUpdate:", &ps_swaptime, PS_TIME},
{0}
};
perfstatrow_t gamelogicbrief_row[] = {
{"logic ", "Game logic: ", &ps_tictime, PS_TIME},
{0}
};
perfstatrow_t commoncounter_rows[] = {
{"bspcall", "BSP calls: ", &ps_numbspcalls, 0},
{"sprites", "Sprites: ", &ps_numsprites, 0},
{"drwnode", "Drawnodes: ", &ps_numdrawnodes, 0},
{"plyobjs", "Polyobjects: ", &ps_numpolyobjects, 0},
{0}
};
perfstatrow_t interpolation_rows[] = {
{"intpfrc", "Interp frac: ", &ps_interp_frac, PS_TIME},
{"intplag", "Interp lag: ", &ps_interp_lag, PS_TIME},
{0}
};
#ifdef HWRENDER
perfstatrow_t batchcount_rows[] = {
{"polygon", "Polygons: ", &ps_hw_numpolys, 0},
{"vertex ", "Vertices: ", &ps_hw_numverts, 0},
{0}
};
perfstatrow_t batchcalls_rows[] = {
{"drwcall", "Draw calls:", &ps_hw_numcalls, 0},
{"shaders", "Shaders: ", &ps_hw_numshaders, 0},
{"texture", "Textures: ", &ps_hw_numtextures, 0},
{"polyflg", "Polyflags: ", &ps_hw_numpolyflags, 0},
{"colors ", "Colors: ", &ps_hw_numcolors, 0},
{0}
};
#endif
// Game logic stats columns
perfstatrow_t gamelogic_rows[] = {
{"logic ", "Game logic: ", &ps_tictime, PS_TIME},
{" plrthnk", " P_PlayerThink: ", &ps_playerthink_time, PS_TIME|PS_LEVEL},
{" thnkers", " P_RunThinkers: ", &ps_thinkertime, PS_TIME|PS_LEVEL},
{" plyobjs", " Polyobjects: ", &ps_thlist_times[THINK_POLYOBJ], PS_TIME|PS_LEVEL},
{" main ", " Main: ", &ps_thlist_times[THINK_MAIN], PS_TIME|PS_LEVEL},
{" mobjs ", " Mobjs: ", &ps_thlist_times[THINK_MOBJ], PS_TIME|PS_LEVEL},
{" dynslop", " Dynamic slopes: ", &ps_thlist_times[THINK_DYNSLOPE], PS_TIME|PS_LEVEL},
{" precip ", " Precipitation: ", &ps_thlist_times[THINK_PRECIP], PS_TIME|PS_LEVEL},
{" lthinkf", " LUAh_ThinkFrame:", &ps_lua_thinkframe_time, PS_TIME|PS_LEVEL},
{" other ", " Other: ", &ps_otherlogictime, PS_TIME|PS_LEVEL},
{0}
};
perfstatrow_t thinkercount_rows[] = {
{"thnkers", "Thinkers: ", &ps_thinkercount, PS_LEVEL},
{" plyobjs", " Polyobjects: ", &ps_polythcount, PS_LEVEL},
{" main ", " Main: ", &ps_mainthcount, PS_LEVEL},
{" mobjs ", " Mobjs: ", &ps_mobjcount, PS_LEVEL},
{" regular", " Regular: ", &ps_regularcount, PS_LEVEL},
{" scenery", " Scenery: ", &ps_scenerycount, PS_LEVEL},
{" nothink", " Nothink: ", &ps_nothinkcount, PS_HIDE_ZERO|PS_LEVEL},
{" dynslop", " Dynamic slopes: ", &ps_dynslopethcount, PS_LEVEL},
{" precip ", " Precipitation: ", &ps_precipcount, PS_LEVEL},
{" remove ", " Pending removal:", &ps_removecount, PS_LEVEL},
{0}
};
perfstatrow_t misc_calls_rows[] = {
{"lmhook", "Lua mobj hooks: ", &ps_lua_mobjhooks, PS_LEVEL},
{"chkpos", "P_CheckPosition:", &ps_checkposition_calls, PS_LEVEL},
{0}
};
// Sample collection status for averaging.
// Maximum of these two is shown to user if nonzero to tell that
// the reported averages are not correct yet.
int ps_frame_samples_left = 0;
int ps_tick_samples_left = 0;
// History writing positions for frame and tick based metrics
int ps_frame_index = 0;
int ps_tick_index = 0;
// dynamically allocated resizeable array for thinkframe hook stats
ps_hookinfo_t *thinkframe_hooks = NULL;
int thinkframe_hooks_length = 0;
int thinkframe_hooks_capacity = 16;
void PS_SetThinkFrameHookInfo(int index, precise_t time_taken, char* short_src)
{
if (!thinkframe_hooks)
{
// array needs to be initialized
thinkframe_hooks = Z_Calloc(sizeof(ps_hookinfo_t) * thinkframe_hooks_capacity, PU_STATIC, NULL);
}
if (index >= thinkframe_hooks_capacity)
{
// array needs more space, realloc with double size
int new_capacity = thinkframe_hooks_capacity * 2;
thinkframe_hooks = Z_Realloc(thinkframe_hooks,
sizeof(ps_hookinfo_t) * new_capacity, PU_STATIC, NULL);
// initialize new memory with zeros so the pointers in the structs are null
memset(&thinkframe_hooks[thinkframe_hooks_capacity], 0,
sizeof(ps_hookinfo_t) * thinkframe_hooks_capacity);
thinkframe_hooks_capacity = new_capacity;
}
thinkframe_hooks[index].time_taken.value.p = time_taken;
memcpy(thinkframe_hooks[index].short_src, short_src, LUA_IDSIZE * sizeof(char));
// since the values are set sequentially from begin to end, the last call should leave
// the correct value to this variable
thinkframe_hooks_length = index + 1;
}
static boolean PS_HighResolution(void)
{
return (vid.width >= 640 && vid.height >= 400);
}
static boolean PS_IsLevelActive(void)
{
return gamestate == GS_LEVEL ||
(gamestate == GS_TITLESCREEN && titlemapinaction);
}
// Is the row valid in the current context?
static boolean PS_IsRowValid(perfstatrow_t *row)
{
return !((row->flags & PS_LEVEL && !PS_IsLevelActive())
|| (row->flags & PS_SW && rendermode != render_soft)
|| (row->flags & PS_HW && rendermode != render_opengl)
#ifdef HWRENDER
|| (row->flags & PS_BATCHING && !cv_glbatching.value)
#endif
);
}
// Should the row be visible on the screen?
static boolean PS_IsRowVisible(perfstatrow_t *row)
{
boolean value_is_zero;
if (row->flags & PS_TIME)
value_is_zero = row->metric->value.p == 0;
else
value_is_zero = row->metric->value.i == 0;
return !(!PS_IsRowValid(row) ||
(row->flags & PS_HIDE_ZERO && value_is_zero));
}
static INT32 PS_GetMetricAverage(ps_metric_t *metric, boolean time_metric)
{
char* history_read_pos = metric->history; // char* used for pointer arithmetic
INT64 sum = 0;
int i;
int value_size = time_metric ? sizeof(precise_t) : sizeof(INT32);
for (i = 0; i < cv_ps_samplesize.value; i++)
{
if (time_metric)
sum += (*((precise_t*)history_read_pos)) / (I_GetPrecisePrecision() / 1000000);
else
sum += *((INT32*)history_read_pos);
history_read_pos += value_size;
}
return sum / cv_ps_samplesize.value;
}
static INT32 PS_GetMetricMinOrMax(ps_metric_t *metric, boolean time_metric, boolean get_max)
{
char* history_read_pos = metric->history; // char* used for pointer arithmetic
INT32 found_value = get_max ? INT32_MIN : INT32_MAX;
int i;
int value_size = time_metric ? sizeof(precise_t) : sizeof(INT32);
for (i = 0; i < cv_ps_samplesize.value; i++)
{
INT32 value;
if (time_metric)
value = (*((precise_t*)history_read_pos)) / (I_GetPrecisePrecision() / 1000000);
else
value = *((INT32*)history_read_pos);
if ((get_max && value > found_value) ||
(!get_max && value < found_value))
{
found_value = value;
}
history_read_pos += value_size;
}
return found_value;
}
// Calculates the standard deviation for metric.
static INT32 PS_GetMetricSD(ps_metric_t *metric, boolean time_metric)
{
char* history_read_pos = metric->history; // char* used for pointer arithmetic
INT64 sum = 0;
int i;
int value_size = time_metric ? sizeof(precise_t) : sizeof(INT32);
INT32 avg = PS_GetMetricAverage(metric, time_metric);
for (i = 0; i < cv_ps_samplesize.value; i++)
{
INT64 value;
if (time_metric)
value = (*((precise_t*)history_read_pos)) / (I_GetPrecisePrecision() / 1000000);
else
value = *((INT32*)history_read_pos);
value -= avg;
sum += value * value;
history_read_pos += value_size;
}
return round(sqrt(sum / cv_ps_samplesize.value));
}
// Returns the value to show on screen for metric.
static INT32 PS_GetMetricScreenValue(ps_metric_t *metric, boolean time_metric)
{
if (cv_ps_samplesize.value > 1 && metric->history)
{
if (cv_ps_descriptor.value == 1)
return PS_GetMetricAverage(metric, time_metric);
else if (cv_ps_descriptor.value == 2)
return PS_GetMetricSD(metric, time_metric);
else if (cv_ps_descriptor.value == 3)
return PS_GetMetricMinOrMax(metric, time_metric, false);
else
return PS_GetMetricMinOrMax(metric, time_metric, true);
}
else
{
if (time_metric)
return (metric->value.p) / (I_GetPrecisePrecision() / 1000000);
else
return metric->value.i;
}
}
static int PS_DrawPerfRows(int x, int y, int color, perfstatrow_t *rows)
{
const boolean hires = PS_HighResolution();
INT32 draw_flags = V_MONOSPACE | color;
perfstatrow_t * row;
int draw_y = y;
if (hires)
draw_flags |= V_ALLOWLOWERCASE;
for (row = rows; row->lores_label; ++row)
{
const char *label;
INT32 value;
char *final_str;
if (!PS_IsRowVisible(row))
continue;
label = hires ? row->hires_label : row->lores_label;
value = PS_GetMetricScreenValue(row->metric, !!(row->flags & PS_TIME));
final_str = va("%s %d", label, value);
if (hires)
{
V_DrawSmallString(x, draw_y, draw_flags, final_str);
draw_y += 5;
}
else
{
V_DrawThinString(x, draw_y, draw_flags, final_str);
draw_y += 8;
}
}
return draw_y;
}
static void PS_UpdateMetricHistory(ps_metric_t *metric, boolean time_metric, boolean frame_metric, boolean set_user)
{
int index = frame_metric ? ps_frame_index : ps_tick_index;
if (!metric->history)
{
// allocate history table
int value_size = time_metric ? sizeof(precise_t) : sizeof(INT32);
void** memory_user = set_user ? &metric->history : NULL;
metric->history = Z_Calloc(value_size * cv_ps_samplesize.value, PU_PERFSTATS,
memory_user);
// reset "samples left" counter since this history table needs to be filled
if (frame_metric)
ps_frame_samples_left = cv_ps_samplesize.value;
else
ps_tick_samples_left = cv_ps_samplesize.value;
}
if (time_metric)
{
precise_t *history = (precise_t*)metric->history;
history[index] = metric->value.p;
}
else
{
INT32 *history = (INT32*)metric->history;
history[index] = metric->value.i;
}
}
static void PS_UpdateRowHistories(perfstatrow_t *rows, boolean frame_metric)
{
perfstatrow_t *row;
for (row = rows; row->lores_label; row++)
{
if (PS_IsRowValid(row))
PS_UpdateMetricHistory(row->metric, !!(row->flags & PS_TIME), frame_metric, true);
}
}
// Update all metrics that are calculated on every frame.
static void PS_UpdateFrameStats(void)
{
// update frame time
precise_t currenttime = I_GetPreciseTime();
ps_frametime.value.p = currenttime - ps_prevframetime;
ps_prevframetime = currenttime;
// update 3d rendering stats
if (PS_IsLevelActive())
{
// Remember to update this calculation when adding more 3d rendering stats!
ps_otherrendertime.value.p = ps_rendercalltime.value.p - ps_bsptime.value.p;
#ifdef HWRENDER
if (rendermode == render_opengl)
{
ps_otherrendertime.value.p -=
ps_hw_skyboxtime.value.p +
ps_hw_nodesorttime.value.p +
ps_hw_nodedrawtime.value.p +
ps_hw_spritesorttime.value.p +
ps_hw_spritedrawtime.value.p;
if (cv_glbatching.value)
{
ps_otherrendertime.value.p -=
ps_hw_batchsorttime.value.p +
ps_hw_batchdrawtime.value.p;
}
}
else
#endif
{
ps_otherrendertime.value.p -=
ps_sw_spritecliptime.value.p +
ps_sw_portaltime.value.p +
ps_sw_planetime.value.p +
ps_sw_maskedtime.value.p;
}
}
if (cv_ps_samplesize.value > 1)
{
PS_UpdateRowHistories(rendertime_rows, true);
if (PS_IsLevelActive())
PS_UpdateRowHistories(commoncounter_rows, true);
if (R_UsingFrameInterpolation())
PS_UpdateRowHistories(interpolation_rows, true);
#ifdef HWRENDER
if (rendermode == render_opengl && cv_glbatching.value)
{
PS_UpdateRowHistories(batchcount_rows, true);
PS_UpdateRowHistories(batchcalls_rows, true);
}
#endif
ps_frame_index++;
if (ps_frame_index >= cv_ps_samplesize.value)
ps_frame_index = 0;
if (ps_frame_samples_left)
ps_frame_samples_left--;
}
}
// Update thinker counters by iterating the thinker lists.
static void PS_CountThinkers(void)
{
int i;
thinker_t *thinker;
ps_thinkercount.value.i = 0;
ps_polythcount.value.i = 0;
ps_mainthcount.value.i = 0;
ps_mobjcount.value.i = 0;
ps_regularcount.value.i = 0;
ps_scenerycount.value.i = 0;
ps_nothinkcount.value.i = 0;
ps_dynslopethcount.value.i = 0;
ps_precipcount.value.i = 0;
ps_removecount.value.i = 0;
for (i = 0; i < NUM_THINKERLISTS; i++)
{
for (thinker = thlist[i].next; thinker != &thlist[i]; thinker = thinker->next)
{
ps_thinkercount.value.i++;
if (thinker->function.acp1 == (actionf_p1)P_RemoveThinkerDelayed)
ps_removecount.value.i++;
else if (i == THINK_POLYOBJ)
ps_polythcount.value.i++;
else if (i == THINK_MAIN)
ps_mainthcount.value.i++;
else if (i == THINK_MOBJ)
{
if (thinker->function.acp1 == (actionf_p1)P_MobjThinker)
{
mobj_t *mobj = (mobj_t*)thinker;
ps_mobjcount.value.i++;
if (mobj->flags & MF_NOTHINK)
ps_nothinkcount.value.i++;
else if (mobj->flags & MF_SCENERY)
ps_scenerycount.value.i++;
else
ps_regularcount.value.i++;
}
}
else if (i == THINK_DYNSLOPE)
ps_dynslopethcount.value.i++;
else if (i == THINK_PRECIP)
ps_precipcount.value.i++;
}
}
}
// Update all metrics that are calculated on every tick.
void PS_UpdateTickStats(void)
{
if (cv_perfstats.value == 1 && cv_ps_samplesize.value > 1)
{
PS_UpdateRowHistories(gamelogicbrief_row, false);
}
if (cv_perfstats.value == 2)
{
if (PS_IsLevelActive())
{
ps_otherlogictime.value.p =
ps_tictime.value.p -
ps_playerthink_time.value.p -
ps_thinkertime.value.p -
ps_lua_thinkframe_time.value.p;
PS_CountThinkers();
}
if (cv_ps_samplesize.value > 1)
{
PS_UpdateRowHistories(gamelogic_rows, false);
PS_UpdateRowHistories(thinkercount_rows, false);
PS_UpdateRowHistories(misc_calls_rows, false);
}
}
if (cv_perfstats.value == 3 && cv_ps_samplesize.value > 1 && PS_IsLevelActive())
{
int i;
for (i = 0; i < thinkframe_hooks_length; i++)
{
PS_UpdateMetricHistory(&thinkframe_hooks[i].time_taken, true, false, false);
}
}
if (cv_perfstats.value && cv_ps_samplesize.value > 1)
{
ps_tick_index++;
if (ps_tick_index >= cv_ps_samplesize.value)
ps_tick_index = 0;
if (ps_tick_samples_left)
ps_tick_samples_left--;
}
}
static void PS_DrawDescriptorHeader(void)
{
if (cv_ps_samplesize.value > 1)
{
const char* descriptor_names[] = {
"average",
"standard deviation",
"minimum",
"maximum"
};
const boolean hires = PS_HighResolution();
char* str;
INT32 flags = V_MONOSPACE | V_ALLOWLOWERCASE;
int samples_left = max(ps_frame_samples_left, ps_tick_samples_left);
int x, y;
if (cv_perfstats.value == 3)
{
x = 2;
y = 0;
}
else
{
x = 20;
y = hires ? 5 : 2;
}
if (samples_left)
{
str = va("Samples needed for correct results: %d", samples_left);
flags |= V_REDMAP;
}
else
{
str = va("Showing the %s of %d samples.",
descriptor_names[cv_ps_descriptor.value - 1], cv_ps_samplesize.value);
flags |= V_GREENMAP;
}
if (hires)
V_DrawSmallString(x, y, flags, str);
else
V_DrawThinString(x, y, flags, str);
}
}
static void PS_DrawRenderStats(void)
{
const boolean hires = PS_HighResolution();
const int half_row = hires ? 5 : 4;
int x, y, cy = 10;
PS_DrawDescriptorHeader();
y = PS_DrawPerfRows(20, 10, V_YELLOWMAP, rendertime_rows);
PS_DrawPerfRows(20, y + half_row, V_GRAYMAP, gamelogicbrief_row);
if (PS_IsLevelActive())
{
x = hires ? 115 : 90;
cy = PS_DrawPerfRows(x, 10, V_BLUEMAP, commoncounter_rows) + half_row;
#ifdef HWRENDER
if (rendermode == render_opengl && cv_glbatching.value)
{
x = hires ? 200 : 155;
y = PS_DrawPerfRows(x, 10, V_PURPLEMAP, batchcount_rows);
x = hires ? 200 : 220;
y = hires ? y + half_row : 10;
PS_DrawPerfRows(x, y, V_PURPLEMAP, batchcalls_rows);
}
#endif
}
if (R_UsingFrameInterpolation())
{
x = hires ? 115 : 90;
PS_DrawPerfRows(x, cy, V_ROSYMAP, interpolation_rows);
}
}
static void PS_DrawGameLogicStats(void)
{
const boolean hires = PS_HighResolution();
int x, y;
PS_DrawDescriptorHeader();
PS_DrawPerfRows(20, 10, V_YELLOWMAP, gamelogic_rows);
x = hires ? 115 : 90;
PS_DrawPerfRows(x, 10, V_BLUEMAP, thinkercount_rows);
if (hires)
V_DrawSmallString(212, 10, V_MONOSPACE | V_ALLOWLOWERCASE | V_PURPLEMAP, "Calls:");
x = hires ? 216 : 170;
y = hires ? 15 : 10;
PS_DrawPerfRows(x, y, V_PURPLEMAP, misc_calls_rows);
}
static void PS_DrawThinkFrameStats(void)
{
char s[100];
int i;
// text writing position
int x = 2;
int y = 4;
UINT32 text_color;
char tempbuffer[LUA_IDSIZE];
char last_mod_name[LUA_IDSIZE];
last_mod_name[0] = '\0';
PS_DrawDescriptorHeader();
for (i = 0; i < thinkframe_hooks_length; i++)
{
#define NEXT_ROW() \
y += 4; \
if (y > 192) \
{ \
y = 4; \
x += 106; \
if (x > 214) \
break; \
}
char* str = thinkframe_hooks[i].short_src;
char* tempstr = tempbuffer;
int len = (int)strlen(str);
char* str_ptr;
if (strcmp(".lua", str + len - 4) == 0)
{
str[len-4] = '\0'; // remove .lua at end
len -= 4;
}
// we locate the wad/pk3 name in the string and compare it to
// what we found on the previous iteration.
// if the name has changed, print it out on the screen
strcpy(tempstr, str);
str_ptr = strrchr(tempstr, '|');
if (str_ptr)
{
*str_ptr = '\0';
str = str_ptr + 1; // this is the name of the hook without the mod file name
str_ptr = strrchr(tempstr, PATHSEP[0]);
if (str_ptr)
tempstr = str_ptr + 1;
// tempstr should now point to the mod name, (wad/pk3) possibly truncated
if (strcmp(tempstr, last_mod_name) != 0)
{
strcpy(last_mod_name, tempstr);
len = (int)strlen(tempstr);
if (len > 25)
tempstr += len - 25;
snprintf(s, sizeof s - 1, "%s", tempstr);
V_DrawSmallString(x, y, V_MONOSPACE | V_ALLOWLOWERCASE | V_GRAYMAP, s);
NEXT_ROW()
}
text_color = V_YELLOWMAP;
}
else
{
// probably a standalone lua file
// cut off the folder if it's there
str_ptr = strrchr(tempstr, PATHSEP[0]);
if (str_ptr)
str = str_ptr + 1;
text_color = 0; // white
}
len = (int)strlen(str);
if (len > 20)
str += len - 20;
snprintf(s, sizeof s - 1, "%20s: %d", str,
PS_GetMetricScreenValue(&thinkframe_hooks[i].time_taken, true));
V_DrawSmallString(x, y, V_MONOSPACE | V_ALLOWLOWERCASE | text_color, s);
NEXT_ROW()
#undef NEXT_ROW
}
}
void M_DrawPerfStats(void)
{
if (cv_perfstats.value == 1) // rendering
{
PS_UpdateFrameStats();
PS_DrawRenderStats();
}
else if (cv_perfstats.value == 2) // logic
{
// PS_UpdateTickStats is called in TryRunTics, since otherwise it would miss
// tics when frame skips happen
PS_DrawGameLogicStats();
}
else if (cv_perfstats.value == 3) // lua thinkframe
{
if (!PS_IsLevelActive())
return;
if (!PS_HighResolution())
{
// Low resolutions can't really use V_DrawSmallString that is used by thinkframe stats.
// A low-res version using V_DrawThinString could be implemented,
// but it would have much less space for information.
V_DrawThinString(80, 92, V_MONOSPACE | V_ALLOWLOWERCASE | V_YELLOWMAP, "Perfstats 3 is not available");
V_DrawThinString(80, 100, V_MONOSPACE | V_ALLOWLOWERCASE | V_YELLOWMAP, "for resolutions below 640x400.");
}
else
{
PS_DrawThinkFrameStats();
}
}
}
// remove and unallocate history from all metrics
static void PS_ClearHistory(void)
{
int i;
Z_FreeTag(PU_PERFSTATS);
// thinkframe hook metric history pointers need to be cleared manually
for (i = 0; i < thinkframe_hooks_length; i++)
{
thinkframe_hooks[i].time_taken.history = NULL;
}
ps_frame_index = ps_tick_index = 0;
// PS_UpdateMetricHistory will set these correctly when it runs
ps_frame_samples_left = ps_tick_samples_left = 0;
}
void PS_PerfStats_OnChange(void)
{
if (cv_perfstats.value && cv_ps_samplesize.value > 1)
PS_ClearHistory();
}
void PS_SampleSize_OnChange(void)
{
if (cv_ps_samplesize.value > 1)
PS_ClearHistory();
}