Blender V4.3
stats.cpp
Go to the documentation of this file.
1/* SPDX-FileCopyrightText: 2011-2022 Blender Foundation
2 *
3 * SPDX-License-Identifier: Apache-2.0 */
4
5#include "scene/stats.h"
6#include "scene/object.h"
7#include "util/algorithm.h"
8#include "util/foreach.h"
9#include "util/string.h"
10
12
13static int kIndentNumSpaces = 2;
14
15/* Named size entry. */
16
17namespace {
18
19bool namedSizeEntryComparator(const NamedSizeEntry &a, const NamedSizeEntry &b)
20{
21 /* We sort in descending order. */
22 return a.size > b.size;
23}
24
25bool namedTimeEntryComparator(const NamedTimeEntry &a, const NamedTimeEntry &b)
26{
27 /* We sort in descending order. */
28 return a.time > b.time;
29}
30
31bool namedTimeSampleEntryComparator(const NamedNestedSampleStats &a,
33{
34 return a.sum_samples > b.sum_samples;
35}
36
37bool namedSampleCountPairComparator(const NamedSampleCountPair &a, const NamedSampleCountPair &b)
38{
39 return a.samples > b.samples;
40}
41
42} // namespace
43
44NamedSizeEntry::NamedSizeEntry() : name(""), size(0) {}
45
46NamedSizeEntry::NamedSizeEntry(const string &name, size_t size) : name(name), size(size) {}
47
48NamedTimeEntry::NamedTimeEntry() : name(""), time(0) {}
49
50NamedTimeEntry::NamedTimeEntry(const string &name, double time) : name(name), time(time) {}
51
52/* Named size statistics. */
53
55
57{
58 total_size += entry.size;
59 entries.push_back(entry);
60}
61
62string NamedSizeStats::full_report(int indent_level)
63{
64 const string indent(indent_level * kIndentNumSpaces, ' ');
65 const string double_indent = indent + indent;
66 string result = "";
67 result += string_printf("%sTotal memory: %s (%s)\n",
68 indent.c_str(),
71 sort(entries.begin(), entries.end(), namedSizeEntryComparator);
72 foreach (const NamedSizeEntry &entry, entries) {
73 result += string_printf("%s%-32s %s (%s)\n",
74 double_indent.c_str(),
75 entry.name.c_str(),
76 string_human_readable_size(entry.size).c_str(),
77 string_human_readable_number(entry.size).c_str());
78 }
79 return result;
80}
81
82string NamedTimeStats::full_report(int indent_level)
83{
84 const string indent(indent_level * kIndentNumSpaces, ' ');
85 const string double_indent = indent + indent;
86 string result = "";
87 result += string_printf("%sTotal time: %fs\n", indent.c_str(), total_time);
88 sort(entries.begin(), entries.end(), namedTimeEntryComparator);
89 foreach (const NamedTimeEntry &entry, entries) {
90 result += string_printf(
91 "%s%-40s %fs\n", double_indent.c_str(), entry.name.c_str(), entry.time);
92 }
93 return result;
94}
95
96/* Named time sample statistics. */
97
98NamedNestedSampleStats::NamedNestedSampleStats() : name(""), self_samples(0), sum_samples(0) {}
99
101 : name(name), self_samples(samples), sum_samples(samples)
102{
103}
104
106{
107 entries.push_back(NamedNestedSampleStats(name_, samples_));
108 return entries[entries.size() - 1];
109}
110
112{
114 foreach (NamedNestedSampleStats &entry, entries) {
115 entry.update_sum();
116 sum_samples += entry.sum_samples;
117 }
118}
119
120string NamedNestedSampleStats::full_report(int indent_level, uint64_t total_samples)
121{
122 update_sum();
123
124 if (total_samples == 0) {
125 total_samples = sum_samples;
126 }
127
128 const string indent(indent_level * kIndentNumSpaces, ' ');
129
130 const double sum_percent = 100 * ((double)sum_samples) / total_samples;
131 const double sum_seconds = sum_samples * 0.001;
132 const double self_percent = 100 * ((double)self_samples) / total_samples;
133 const double self_seconds = self_samples * 0.001;
134 string info = string_printf("%-32s: Total %3.2f%% (%.2fs), Self %3.2f%% (%.2fs)\n",
135 name.c_str(),
136 sum_percent,
137 sum_seconds,
138 self_percent,
139 self_seconds);
140 string result = indent + info;
141
142 sort(entries.begin(), entries.end(), namedTimeSampleEntryComparator);
143 foreach (NamedNestedSampleStats &entry, entries) {
144 result += entry.full_report(indent_level + 1, total_samples);
145 }
146 return result;
147}
148
149/* Named sample count pairs. */
150
152 : name(name), samples(samples), hits(hits)
153{
154}
155
157
158void NamedSampleCountStats::add(const ustring &name, uint64_t samples, uint64_t hits)
159{
160 entry_map::iterator entry = entries.find(name);
161 if (entry != entries.end()) {
162 entry->second.samples += samples;
163 entry->second.hits += hits;
164 return;
165 }
166 entries.emplace(name, NamedSampleCountPair(name, samples, hits));
167}
168
170{
171 const string indent(indent_level * kIndentNumSpaces, ' ');
172
173 vector<NamedSampleCountPair> sorted_entries;
174 sorted_entries.reserve(entries.size());
175
176 uint64_t total_hits = 0, total_samples = 0;
177 foreach (entry_map::const_reference entry, entries) {
178 const NamedSampleCountPair &pair = entry.second;
179
180 total_hits += pair.hits;
181 total_samples += pair.samples;
182
183 sorted_entries.push_back(pair);
184 }
185 const double avg_samples_per_hit = ((double)total_samples) / total_hits;
186
187 sort(sorted_entries.begin(), sorted_entries.end(), namedSampleCountPairComparator);
188
189 string result = "";
190 foreach (const NamedSampleCountPair &entry, sorted_entries) {
191 const double seconds = entry.samples * 0.001;
192 const double relative = ((double)entry.samples) / (entry.hits * avg_samples_per_hit);
193
194 result += indent +
196 "%-32s: %.2fs (Relative cost: %.2f)\n", entry.name.c_str(), seconds, relative);
197 }
198 return result;
199}
200
201/* Mesh statistics. */
202
204
205string MeshStats::full_report(int indent_level)
206{
207 const string indent(indent_level * kIndentNumSpaces, ' ');
208 string result = "";
209 result += indent + "Geometry:\n" + geometry.full_report(indent_level + 1);
210 return result;
211}
212
213/* Image statistics. */
214
216
217string ImageStats::full_report(int indent_level)
218{
219 const string indent(indent_level * kIndentNumSpaces, ' ');
220 string result = "";
221 result += indent + "Textures:\n" + textures.full_report(indent_level + 1);
222 return result;
223}
224
225/* Overall statistics. */
226
231
233{
234 has_profiling = true;
235
236 kernel = NamedNestedSampleStats("Total render time", prof.get_event(PROFILING_UNKNOWN));
237 kernel.add_entry("Ray setup", prof.get_event(PROFILING_RAY_SETUP));
238 kernel.add_entry("Intersect Closest", prof.get_event(PROFILING_INTERSECT_CLOSEST));
239 kernel.add_entry("Intersect Shadow", prof.get_event(PROFILING_INTERSECT_SHADOW));
240 kernel.add_entry("Intersect Subsurface", prof.get_event(PROFILING_INTERSECT_SUBSURFACE));
241 kernel.add_entry("Intersect Volume Stack", prof.get_event(PROFILING_INTERSECT_VOLUME_STACK));
242 kernel.add_entry("Intersect Blocked Light", prof.get_event(PROFILING_INTERSECT_DEDICATED_LIGHT));
243
244 NamedNestedSampleStats &surface = kernel.add_entry("Shade Surface", 0);
245 surface.add_entry("Setup", prof.get_event(PROFILING_SHADE_SURFACE_SETUP));
246 surface.add_entry("Shader Evaluation", prof.get_event(PROFILING_SHADE_SURFACE_EVAL));
247 surface.add_entry("Render Passes", prof.get_event(PROFILING_SHADE_SURFACE_PASSES));
248 surface.add_entry("Direct Light", prof.get_event(PROFILING_SHADE_SURFACE_DIRECT_LIGHT));
249 surface.add_entry("Indirect Light", prof.get_event(PROFILING_SHADE_SURFACE_INDIRECT_LIGHT));
250 surface.add_entry("Ambient Occlusion", prof.get_event(PROFILING_SHADE_SURFACE_AO));
251
252 NamedNestedSampleStats &volume = kernel.add_entry("Shade Volume", 0);
253 volume.add_entry("Setup", prof.get_event(PROFILING_SHADE_VOLUME_SETUP));
254 volume.add_entry("Integrate", prof.get_event(PROFILING_SHADE_VOLUME_INTEGRATE));
255 volume.add_entry("Direct Light", prof.get_event(PROFILING_SHADE_VOLUME_DIRECT_LIGHT));
256 volume.add_entry("Indirect Light", prof.get_event(PROFILING_SHADE_VOLUME_INDIRECT_LIGHT));
257
258 NamedNestedSampleStats &shadow = kernel.add_entry("Shade Shadow", 0);
259 shadow.add_entry("Setup", prof.get_event(PROFILING_SHADE_SHADOW_SETUP));
260 shadow.add_entry("Surface", prof.get_event(PROFILING_SHADE_SHADOW_SURFACE));
261 shadow.add_entry("Volume", prof.get_event(PROFILING_SHADE_SHADOW_VOLUME));
262 shadow.add_entry("Blocked Light", prof.get_event(PROFILING_SHADE_DEDICATED_LIGHT));
263
264 NamedNestedSampleStats &light = kernel.add_entry("Shade Light", 0);
265 light.add_entry("Setup", prof.get_event(PROFILING_SHADE_LIGHT_SETUP));
266 light.add_entry("Shader Evaluation", prof.get_event(PROFILING_SHADE_LIGHT_EVAL));
267
268 shaders.entries.clear();
269 foreach (Shader *shader, scene->shaders) {
270 uint64_t samples, hits;
271 if (prof.get_shader(shader->id, samples, hits)) {
272 shaders.add(shader->name, samples, hits);
273 }
274 }
275
276 objects.entries.clear();
277 foreach (Object *object, scene->objects) {
278 uint64_t samples, hits;
279 if (prof.get_object(object->get_device_index(), samples, hits)) {
280 objects.add(object->name, samples, hits);
281 }
282 }
283}
284
286{
287 string result = "";
288 result += "Mesh statistics:\n" + mesh.full_report(1);
289 result += "Image statistics:\n" + image.full_report(1);
290 if (has_profiling) {
291 result += "Kernel statistics:\n" + kernel.full_report(1);
292 result += "Shader statistics:\n" + shaders.full_report(1);
293 result += "Object statistics:\n" + objects.full_report(1);
294 }
295 else {
296 result += "Profiling information not available (only works with CPU rendering)";
297 }
298 return result;
299}
300
302
303string UpdateTimeStats::full_report(int indent_level)
304{
305 return times.full_report(indent_level + 1);
306}
307
309
311{
312 string result = "";
313 result += "Scene:\n" + scene.full_report(1);
314 result += "Geometry:\n" + geometry.full_report(1);
315 result += "Light:\n" + light.full_report(1);
316 result += "Object:\n" + object.full_report(1);
317 result += "Image:\n" + image.full_report(1);
318 result += "Background:\n" + background.full_report(1);
319 result += "Bake:\n" + bake.full_report(1);
320 result += "Camera:\n" + camera.full_report(1);
321 result += "Film:\n" + film.full_report(1);
322 result += "Integrator:\n" + integrator.full_report(1);
323 result += "OSL:\n" + osl.full_report(1);
324 result += "Particles:\n" + particles.full_report(1);
325 result += "SVM:\n" + svm.full_report(1);
326 result += "Tables:\n" + tables.full_report(1);
327 result += "Procedurals:\n" + procedurals.full_report(1);
328 return result;
329}
330
332{
333 geometry.times.clear();
334 image.times.clear();
335 light.times.clear();
336 object.times.clear();
337 background.times.clear();
338 bake.times.clear();
339 camera.times.clear();
340 film.times.clear();
342 osl.times.clear();
344 scene.times.clear();
345 svm.times.clear();
348}
349
typedef double(DMatrix)[4][4]
static DBVT_INLINE btDbvtNode * sort(btDbvtNode *n, btDbvtNode *&r)
Definition btDbvt.cpp:418
string full_report(int indent_level=0)
Definition stats.cpp:217
NamedSizeStats textures
MeshStats()
Definition stats.cpp:203
string full_report(int indent_level=0)
Definition stats.cpp:205
string full_report(int indent_level=0, uint64_t total_samples=0)
Definition stats.cpp:120
vector< NamedNestedSampleStats > entries
NamedNestedSampleStats & add_entry(const string &name, uint64_t samples)
Definition stats.cpp:105
NamedSampleCountPair(const ustring &name, uint64_t samples, uint64_t hits)
Definition stats.cpp:151
string full_report(int indent_level=0)
Definition stats.cpp:169
void add(const ustring &name, uint64_t samples, uint64_t hits)
Definition stats.cpp:158
void add_entry(const NamedSizeEntry &entry)
Definition stats.cpp:56
size_t total_size
Definition scene/stats.h:56
string full_report(int indent_level=0)
Definition stats.cpp:62
vector< NamedSizeEntry > entries
Definition scene/stats.h:61
double total_time
Definition scene/stats.h:79
string full_report(int indent_level=0)
Definition stats.cpp:82
vector< NamedTimeEntry > entries
Definition scene/stats.h:84
uint64_t get_event(ProfilingEvent event)
bool get_shader(int shader, uint64_t &samples, uint64_t &hits)
bool get_object(int object, uint64_t &samples, uint64_t &hits)
UpdateTimeStats tables
string full_report()
Definition stats.cpp:310
UpdateTimeStats particles
UpdateTimeStats procedurals
UpdateTimeStats integrator
UpdateTimeStats osl
UpdateTimeStats svm
NamedTimeStats times
string full_report(int indent_level=0)
Definition stats.cpp:303
local_group_size(16, 16) .push_constant(Type b
#define CCL_NAMESPACE_END
static CCL_NAMESPACE_BEGIN int kIndentNumSpaces
Definition stats.cpp:13
unsigned __int64 uint64_t
Definition stdint.h:90
string string_human_readable_size(size_t size)
Definition string.cpp:234
string string_human_readable_number(size_t num)
Definition string.cpp:255
CCL_NAMESPACE_BEGIN string string_printf(const char *format,...)
Definition string.cpp:23
void collect_profiling(Scene *scene, Profiler &prof)
Definition stats.cpp:232
bool has_profiling
string full_report()
Definition stats.cpp:285
NamedNestedSampleStats kernel
@ PROFILING_SHADE_SURFACE_SETUP
@ PROFILING_SHADE_SHADOW_SURFACE
@ PROFILING_SHADE_VOLUME_INTEGRATE
@ PROFILING_SHADE_SHADOW_VOLUME
@ PROFILING_INTERSECT_SUBSURFACE
@ PROFILING_INTERSECT_SHADOW
@ PROFILING_INTERSECT_CLOSEST
@ PROFILING_INTERSECT_DEDICATED_LIGHT
@ PROFILING_SHADE_VOLUME_SETUP
@ PROFILING_SHADE_SURFACE_EVAL
@ PROFILING_SHADE_SURFACE_INDIRECT_LIGHT
@ PROFILING_SHADE_SURFACE_AO
@ PROFILING_SHADE_SURFACE_DIRECT_LIGHT
@ PROFILING_SHADE_LIGHT_SETUP
@ PROFILING_RAY_SETUP
@ PROFILING_SHADE_SHADOW_SETUP
@ PROFILING_SHADE_DEDICATED_LIGHT
@ PROFILING_INTERSECT_VOLUME_STACK
@ PROFILING_SHADE_VOLUME_DIRECT_LIGHT
@ PROFILING_UNKNOWN
@ PROFILING_SHADE_SURFACE_PASSES
@ PROFILING_SHADE_LIGHT_EVAL
@ PROFILING_SHADE_VOLUME_INDIRECT_LIGHT
double total_time