mirror of
https://github.com/kevinbentley/Descent3.git
synced 2025-01-22 19:55:23 +00:00
339 lines
16 KiB
C++
339 lines
16 KiB
C++
/*
|
|
* Descent 3
|
|
* Copyright (C) 2024 Parallax Software
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#ifndef _RUN_TIME_PROFILING_
|
|
#define _RUN_TIME_PROFILING_
|
|
|
|
#include <cstdint>
|
|
|
|
// uncomment the following if you want to enable Run-time Profiling
|
|
#ifndef RELEASE
|
|
#define USE_RTP
|
|
#endif
|
|
|
|
#if defined(__LINUX__)
|
|
// #if defined(__LINUX__)
|
|
#ifdef USE_RTP
|
|
#undef USE_RTP // no rtp for now
|
|
#endif
|
|
#endif
|
|
|
|
// use INT64 for 64bit integers
|
|
#ifdef USE_RTP
|
|
#if defined(__LINUX__)
|
|
#define INT64 int64_t
|
|
#else
|
|
#define INT64 signed __int64
|
|
#endif
|
|
#else
|
|
#define INT64 uint8_t
|
|
#endif
|
|
|
|
|
|
// struct of information to be saved per frame (note: use INT64 for timer info)
|
|
// -----------------------------------------------------------------------------------
|
|
struct tRTFrameInfo {
|
|
INT64 frame_num; // the frame number, used internally
|
|
INT64 renderframe_time;
|
|
INT64 multiframe_time;
|
|
INT64 musicframe_time;
|
|
INT64 ambsound_frame_time;
|
|
INT64 weatherframe_time;
|
|
INT64 playerframe_time;
|
|
INT64 doorframe_time;
|
|
INT64 levelgoal_time;
|
|
INT64 matcenframe_time;
|
|
INT64 objframe_time;
|
|
INT64 aiframeall_time;
|
|
INT64 processkeys_time;
|
|
INT64 normalevent_time;
|
|
INT64 ct_flying_time;
|
|
INT64 ct_aidoframe_time;
|
|
INT64 ct_weaponframe_time;
|
|
INT64 ct_explosionframe_time;
|
|
INT64 ct_debrisframe_time;
|
|
INT64 ct_splinterframe_time;
|
|
INT64 mt_physicsframe_time;
|
|
INT64 mt_walkingframe_time;
|
|
INT64 mt_shockwave_time;
|
|
INT64 obj_doeffect_time;
|
|
INT64 obj_move_player_time;
|
|
INT64 obj_d3xint_time;
|
|
INT64 obj_objlight_time;
|
|
INT64 cycle_anim;
|
|
INT64 vis_eff_move;
|
|
INT64 phys_link;
|
|
INT64 obj_do_frm;
|
|
INT64 fvi_time;
|
|
|
|
int texture_uploads;
|
|
int polys_drawn;
|
|
int fvi_calls;
|
|
float frame_time; // how long the frame took. A float because it's already calc'd so we might as well save it
|
|
};
|
|
|
|
// Flags for Runtime Performance Counters (these are 64 bit)
|
|
// --------------------------------------------------------------------
|
|
#define RTI_FRAMETIME 0x00000001
|
|
#define RTI_RENDERFRAMETIME 0x00000002
|
|
#define RTI_MULTIFRAMETIME 0x00000004
|
|
#define RTI_MUSICFRAMETIME 0x00000008
|
|
#define RTI_AMBSOUNDFRAMETIME 0x00000010
|
|
#define RTI_WEATHERFRAMETIME 0x00000020
|
|
#define RTI_PLAYERFRAMETIME 0x00000040
|
|
#define RTI_DOORFRAMETIME 0x00000080
|
|
#define RTI_LEVELGOALTIME 0x00000100
|
|
#define RTI_MATCENFRAMETIME 0x00000200
|
|
#define RTI_OBJFRAMETIME 0x00000400
|
|
#define RTI_AIFRAMETIME 0x00000800
|
|
#define RTI_PROCESSKEYTIME 0x00001000
|
|
#define RTI_TEXTUREUPLOADS 0x00002000
|
|
#define RTI_POLYSDRAWN 0x00004000
|
|
|
|
// Macros to be used internally and externally
|
|
// --------------------------------------------------
|
|
|
|
#ifndef USE_RTP
|
|
|
|
#define RTP_CLOCKTOSECONDS(c1, c2, t)
|
|
#define RTP_CLOCKSECONDS(c, t)
|
|
#define RTP_RECORDVALUE(member, value)
|
|
#define RTP_INCRVALUE(member, amount)
|
|
#define RTP_DECRVALUE(member, amount)
|
|
#define RTP_GETVALUE(member, value)
|
|
#define RTP_STARTTIME(member)
|
|
#define RTP_tSTARTTIME(member, time)
|
|
#define RTP_ENDTIME(member)
|
|
#define RTP_tENDTIME(member, time)
|
|
#define RTP_GETCLOCK(time)
|
|
#define RTP_STARTINCTIME(member)
|
|
#define RTP_ENDINCTIME(member)
|
|
#define RTP_ENABLEFLAGS(flags)
|
|
#define RTP_DISABLEFLAGS(flags)
|
|
|
|
#else
|
|
|
|
#define RTP_ENABLEFLAGS(flags) rtp_EnableFlags(flags)
|
|
|
|
#define RTP_DISABLEFLAGS(flags) rtp_EnableFlags(flags)
|
|
|
|
// given 2 clocks (INT64's) it will return the time in seconds between them (double)
|
|
// (internal use)
|
|
// c1 = start time
|
|
// c2 = end time
|
|
// t = time in seconds
|
|
#define RTP_CLOCKTOSECONDS(c1, c2, t) \
|
|
do { \
|
|
if (Runtime_performance_enabled) { \
|
|
INT64 diff = (c2) - (c1); \
|
|
t = ((double)diff) / ((double)Runtime_performance_clockfreq); \
|
|
} \
|
|
} while (0)
|
|
|
|
// given a clock time, it will convert it to seconds
|
|
#define RTP_CLOCKSECONDS(c, t) \
|
|
do { \
|
|
if (Runtime_performance_enabled) { \
|
|
t = ((double)c) / ((double)Runtime_performance_clockfreq); \
|
|
} \
|
|
} while (0)
|
|
|
|
// sets a value of a member of the current tRTFrameInfo frame
|
|
// (external use)
|
|
// member = the member of the tRTFrameInfo to set
|
|
// value = the value to set it to
|
|
#define RTP_RECORDVALUE(member, value) \
|
|
do { \
|
|
if (Runtime_performance_enabled) { \
|
|
RTP_SingleFrame.member = (value); \
|
|
} \
|
|
} while (0)
|
|
|
|
// increments a value of a member of the current tRTFrameInfo frame
|
|
// (external use)
|
|
// member = the member of the tRTFrameInfo to increment
|
|
// amount = how much to increment it
|
|
#define RTP_INCRVALUE(member, amount) \
|
|
do { \
|
|
if (Runtime_performance_enabled) { \
|
|
RTP_SingleFrame.member += (amount); \
|
|
} \
|
|
} while (0)
|
|
|
|
// decrements a value of a member of the current tRTFrameInfo frame
|
|
// (external use)
|
|
// member = the member of the tRTFrameInfo to decrement
|
|
// amount = how much to decrement it
|
|
#define RTP_DECRVALUE(member, amount) \
|
|
do { \
|
|
if (Runtime_performance_enabled) { \
|
|
RTP_SingleFrame.member -= (amount); \
|
|
} \
|
|
} while (0)
|
|
|
|
// returns the value of a member of the current tRTFrameInfo frame
|
|
// (external use)
|
|
// member = the member to retrieve
|
|
// value = where to store the value
|
|
#define RTP_GETVALUE(member, value) \
|
|
do { \
|
|
if (Runtime_performance_enabled) { \
|
|
value = RTP_SingleFrame.member; \
|
|
} \
|
|
} while (0)
|
|
|
|
// starts a frame time calculation for a time member of the tRTFrameInfo frame
|
|
// this version calls the clock to get the current time
|
|
// member = member to start recording time on
|
|
#define RTP_STARTTIME(member) \
|
|
do { \
|
|
if (Runtime_performance_enabled) { \
|
|
RTP_SingleFrame.member = rtp_GetClock(); \
|
|
} \
|
|
} while (0)
|
|
|
|
// starts a frame time calculation given a time to start it with (similar to the above
|
|
// except you give it the starting time). This should be used (along with
|
|
// RTP_tENDTIME(member,time)) in places where you do a bunch of frame times in a row
|
|
// to reduce the number of calls to rtp_GetClock(), however, this way will give
|
|
// a few cycles off, compared to the RTP_STARTTIME(member)/RTP_ENDTIME(member) way
|
|
// member = member to start recording time on
|
|
#define RTP_tSTARTTIME(member, time) \
|
|
do { \
|
|
if (Runtime_performance_enabled) { \
|
|
RTP_SingleFrame.member = time; \
|
|
} \
|
|
} while (0)
|
|
|
|
// ends a frame time calculation for a member of tRTFrameInfo frame
|
|
// member = member to stop recording time for
|
|
#define RTP_ENDTIME(member) \
|
|
do { \
|
|
if (Runtime_performance_enabled) { \
|
|
RTP_SingleFrame.member = rtp_GetClock() - RTP_SingleFrame.member; \
|
|
} \
|
|
} while (0)
|
|
|
|
// same as above, but will return the time that it stopped the recording time
|
|
#define RTP_tENDTIME(member, time) \
|
|
do { \
|
|
if (Runtime_performance_enabled) { \
|
|
time = rtp_GetClock(); \
|
|
RTP_SingleFrame.member = time - RTP_SingleFrame.member; \
|
|
} \
|
|
} while (0)
|
|
|
|
// returns the current clock time
|
|
#define RTP_GETCLOCK(time) \
|
|
do { \
|
|
if (Runtime_performance_enabled) { \
|
|
time = rtp_GetClock(); \
|
|
} \
|
|
} while (0)
|
|
|
|
// Starts a frame time calculation, but when ending the frame, it will increment the time by
|
|
// the difference instead of just setting it
|
|
// RTP_STARTINCTIME and RTP_ENDINCTIME must be in the same scope
|
|
#define RTP_STARTINCTIME(member) \
|
|
INT64 __start_time_; \
|
|
do { \
|
|
if (Runtime_performance_enabled) { \
|
|
__start_time_ = rtp_GetClock(); \
|
|
} \
|
|
} while (0)
|
|
|
|
// Ends a frame time calculation and increments the member by the time for that frame
|
|
#define RTP_ENDINCTIME(member) \
|
|
do { \
|
|
if (Runtime_performance_enabled) { \
|
|
RTP_SingleFrame.member += (rtp_GetClock() - __start_time_); \
|
|
} \
|
|
} while (0)
|
|
|
|
#endif
|
|
|
|
// Internal Global Vars
|
|
// ------------------------------
|
|
extern tRTFrameInfo RTP_SingleFrame;
|
|
extern INT64 Runtime_performance_clockfreq;
|
|
extern uint8_t Runtime_performance_enabled;
|
|
|
|
/*
|
|
void rtp_Init
|
|
This function initilizes the runtime performance system so it's ready to be used
|
|
*/
|
|
void rtp_Init(void);
|
|
|
|
/*
|
|
void rtp_EnableFlags
|
|
This function enables the performance profiling of whatever flags you pass in
|
|
*/
|
|
void rtp_EnableFlags(INT64 flags);
|
|
|
|
/*
|
|
void rtp_DisableFlags
|
|
This function disables the performance profiling of whatever flags you pass in
|
|
*/
|
|
void rtp_DisableFlags(INT64 flags);
|
|
|
|
/*
|
|
void rtp_StartLog
|
|
Calling this function will reset the log and start a new log, recording immediatly
|
|
*/
|
|
void rtp_StartLog(void);
|
|
|
|
/*
|
|
void rtp_StopLog
|
|
Calling this function will stop the currently processing log and write it out to
|
|
file.
|
|
*/
|
|
void rtp_StopLog(void);
|
|
|
|
/*
|
|
void rtp_PauseLog
|
|
Calling this function will pause the log recording until rtp_ResumeLog is called
|
|
*/
|
|
void rtp_PauseLog(void);
|
|
|
|
/*
|
|
void rtp_ResumeLog
|
|
Calling this will resume a paused log, starting at where it left off
|
|
*/
|
|
void rtp_ResumeLog(void);
|
|
|
|
/*
|
|
void rtp_RecordFrame
|
|
Calling this will record the data of the frame into the internal log, and prepare for
|
|
the next frame
|
|
*/
|
|
void rtp_RecordFrame(void);
|
|
|
|
/*
|
|
INT64 rtp_GetClock
|
|
Returns the current hi-resolution clock value...no checking for overflow
|
|
*/
|
|
INT64 rtp_GetClock(void);
|
|
|
|
/*
|
|
void rtp_WriteBufferLog
|
|
Writes the buffer of frames to file...
|
|
*/
|
|
void rtp_WriteBufferLog(void);
|
|
|
|
#endif
|