OpenTTD Source 20260208-master-g43af8e94d0
newgrf_profiling.cpp
Go to the documentation of this file.
1/*
2 * This file is part of OpenTTD.
3 * OpenTTD 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, version 2.
4 * OpenTTD 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.
5 * See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with OpenTTD. If not, see <https://www.gnu.org/licenses/old-licenses/gpl-2.0>.
6 */
7
9
10#include "stdafx.h"
11
12#include "newgrf_profiling.h"
13#include "fileio_func.h"
14#include "string_func.h"
15#include "console_func.h"
16#include "spritecache.h"
17#include "3rdparty/fmt/chrono.h"
18#include "timer/timer.h"
20
21#include <chrono>
22
23#include "safeguards.h"
24
25std::vector<NewGRFProfiler> _newgrf_profilers;
26
27
35
42
48{
49 using namespace std::chrono;
50 this->cur_call.root_sprite = resolver.root_spritegroup->nfo_line;
51 this->cur_call.subs = 0;
52 this->cur_call.time = (uint32_t)time_point_cast<microseconds>(high_resolution_clock::now()).time_since_epoch().count();
54 this->cur_call.cb = resolver.callback;
55 this->cur_call.feat = resolver.GetFeature();
56 this->cur_call.item = resolver.GetDebugID();
57}
58
63{
64 using namespace std::chrono;
65 this->cur_call.time = (uint32_t)time_point_cast<microseconds>(high_resolution_clock::now()).time_since_epoch().count() - this->cur_call.time;
66
67 struct visitor {
68 uint32_t operator()(std::monostate)
69 {
70 return 0;
71 }
72 uint32_t operator()(CallbackResult cb_result)
73 {
74 return cb_result;
75 }
76 uint32_t operator()(const ResultSpriteGroup *group)
77 {
78 return group == nullptr ? 0 : GetSpriteLocalID(group->sprite);
79 }
80 uint32_t operator()(const TileLayoutSpriteGroup *group)
81 {
82 return group == nullptr ? 0 : group->nfo_line;
83 }
84 uint32_t operator()(const IndustryProductionSpriteGroup *group)
85 {
86 return group == nullptr ? 0 : group->nfo_line;
87 }
88 };
89 this->cur_call.result = std::visit(visitor{}, result);
90
91 this->calls.push_back(this->cur_call);
92}
93
98{
99 this->cur_call.subs += 1;
100}
101
102void NewGRFProfiler::Start()
103{
104 this->Abort();
105 this->active = true;
107}
108
109uint32_t NewGRFProfiler::Finish()
110{
111 if (!this->active) return 0;
112
113 if (this->calls.empty()) {
114 IConsolePrint(CC_DEBUG, "Finished profile of NewGRF [{:08X}], no events collected, not writing a file.", std::byteswap(this->grffile->grfid));
115
116 this->Abort();
117 return 0;
118 }
119
120 std::string filename = this->GetOutputFilename();
121 IConsolePrint(CC_DEBUG, "Finished profile of NewGRF [{:08X}], writing {} events to '{}'.", std::byteswap(this->grffile->grfid), this->calls.size(), filename);
122
123 uint32_t total_microseconds = 0;
124
125 auto f = FioFOpenFile(filename, "wt", Subdirectory::NO_DIRECTORY);
126
127 if (!f.has_value()) {
128 IConsolePrint(CC_ERROR, "Failed to open '{}' for writing.", filename);
129 } else {
130 fmt::print(*f, "Tick,Sprite,Feature,Item,CallbackID,Microseconds,Depth,Result\n");
131 for (const Call &c : this->calls) {
132 fmt::print(*f, "{},{},0x{:X},{},0x{:X},{},{},{}\n", c.tick, c.root_sprite, c.feat, c.item, (uint)c.cb, c.time, c.subs, c.result);
133 total_microseconds += c.time;
134 }
135 }
136
137 this->Abort();
138 return total_microseconds;
139}
140
141void NewGRFProfiler::Abort()
142{
143 this->active = false;
144 this->calls.clear();
145}
146
152{
153 return fmt::format("{}grfprofile-{:%Y%m%d-%H%M}-{:08X}.csv", FiosGetScreenshotDir(), fmt::localtime(time(nullptr)), std::byteswap(this->grffile->grfid));
154}
155
156/* static */ uint32_t NewGRFProfiler::FinishAll()
157{
159
160 uint64_t max_ticks = 0;
161 uint32_t total_microseconds = 0;
162 for (NewGRFProfiler &pr : _newgrf_profilers) {
163 if (pr.active) {
164 total_microseconds += pr.Finish();
165 max_ticks = std::max(max_ticks, TimerGameTick::counter - pr.start_tick);
166 }
167 }
168
169 if (total_microseconds > 0 && max_ticks > 0) {
170 IConsolePrint(CC_DEBUG, "Total NewGRF callback processing: {} microseconds over {} ticks.", total_microseconds, max_ticks);
171 }
172
173 return total_microseconds;
174}
175
180{
181 NewGRFProfiler::FinishAll();
182});
183
187/* static */ void NewGRFProfiler::StartTimer(uint64_t ticks)
188{
189 _profiling_finish_timeout.Reset({ TimerGameTick::Priority::None, static_cast<uint>(ticks) });
190}
191
195/* static */ void NewGRFProfiler::AbortTimer()
196{
198}
constexpr enable_if_t< is_integral_v< T >, T > byteswap(T x) noexcept
Custom implementation of std::byteswap; remove once we build with C++23.
A timeout timer will fire once after the interval.
Definition timer.h:116
static TickCounter counter
Monotonic counter, in ticks, since start of game.
@ None
These timers can be executed in any order; the order is not relevant.
void IConsolePrint(TextColour colour_code, const std::string &string)
Handle the printing of text entered into the console or redirected there by any other means.
Definition console.cpp:90
Console functions used outside of the console code.
static const TextColour CC_DEBUG
Colour for debug output.
static const TextColour CC_ERROR
Colour for error lines.
std::optional< FileHandle > FioFOpenFile(std::string_view filename, std::string_view mode, Subdirectory subdir, size_t *filesize)
Opens a OpenTTD file somewhere in a personal or global directory.
Definition fileio.cpp:244
Functions for standard in/out file operations.
std::string_view FiosGetScreenshotDir()
Get the directory for screenshots.
Definition fios.cpp:579
@ NO_DIRECTORY
A path without any base directory.
static TimeoutTimer< TimerGameTick > _profiling_finish_timeout({ TimerGameTick::Priority::None, 0 }, []() { NewGRFProfiler::FinishAll();})
Check whether profiling is active and should be finished.
Profiling of NewGRF action 2 handling.
std::variant< std::monostate, CallbackResult, const ResultSpriteGroup *, const TileLayoutSpriteGroup *, const IndustryProductionSpriteGroup * > ResolverResult
Result of resolving sprite groups:
A number of safeguards to prevent using unsafe methods.
uint32_t GetSpriteLocalID(SpriteID sprite)
Get the GRF-local sprite id of a given sprite.
Functions to cache sprites in memory.
Definition of base types and functions in a cross-platform compatible way.
Functions related to low-level strings.
Dynamic data of a loaded NewGRF.
Definition newgrf.h:117
Measurement of a single sprite group resolution.
Callback profiler for NewGRF development.
void RecursiveResolve()
Capture a recursive sprite group resolution.
static void StartTimer(uint64_t ticks)
Start the timeout timer that will finish all profiling sessions.
void EndResolve(const ResolverResult &result)
Capture the completion of a sprite group resolution.
~NewGRFProfiler()
Complete profiling session and write data to file.
Call cur_call
Data for current call in progress.
bool active
Is this profiler collecting data.
const GRFFile * grffile
Which GRF is being profiled.
uint64_t start_tick
Tick number this profiler was started on.
static void AbortTimer()
Abort the timeout timer, so the timer callback is never called.
void BeginResolve(const ResolverObject &resolver)
Capture the start of a sprite group resolution.
NewGRFProfiler(const GRFFile *grffile)
Create profiler object and begin profiling session.
std::vector< Call > calls
All calls collected so far.
std::string GetOutputFilename() const
Get name of the file that will be written.
Interface for SpriteGroup-s to access the gamestate.
virtual uint32_t GetDebugID() const
Get an identifier for the item being resolved.
CallbackID callback
Callback being resolved.
virtual GrfSpecFeature GetFeature() const
Get the feature number being resolved for.
const SpriteGroup * root_spritegroup
Root SpriteGroup to use for resolving.
Action 2 sprite layout for houses, industry tiles, objects and airport tiles.
Definition of Interval and OneShot timers.
Definition of the tick-based game-timer.