ScummVM API documentation
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Macros Modules Pages
as_debug.h
1 /*
2  AngelCode Scripting Library
3  Copyright (c) 2003-2016 Andreas Jonsson
4 
5  This software is provided 'as-is', without any express or implied
6  warranty. In no event will the authors be held liable for any
7  damages arising from the use of this software.
8 
9  Permission is granted to anyone to use this software for any
10  purpose, including commercial applications, and to alter it and
11  redistribute it freely, subject to the following restrictions:
12 
13  1. The origin of this software must not be misrepresented; you
14  must not claim that you wrote the original software. If you use
15  this software in a product, an acknowledgment in the product
16  documentation would be appreciated but is not required.
17 
18  2. Altered source versions must be plainly marked as such, and
19  must not be misrepresented as being the original software.
20 
21  3. This notice may not be removed or altered from any source
22  distribution.
23 
24  The original version of this library can be located at:
25  http://www.angelcode.com/angelscript/
26 
27  Andreas Jonsson
28  andreas@angelcode.com
29 */
30 
31 
32 //
33 // as_debug.h
34 //
35 
36 #ifndef AS_DEBUG_H
37 #define AS_DEBUG_H
38 
39 #include "as_config.h"
40 
41 #if defined(AS_DEBUG)
42 
43 #ifndef AS_WII
44 // The Wii SDK doesn't have these, we'll survive without AS_DEBUG
45 
46 #ifndef _WIN32_WCE
47 // Neither does WinCE
48 
49 #ifndef AS_PSVITA
50 // Possible on PSVita, but requires SDK access
51 
52 #if !defined(_MSC_VER) && (defined(__GNUC__) || defined(AS_MARMALADE))
53 
54 #ifdef __ghs__
55 // WIIU defines __GNUC__ but types are not defined here in 'conventional' way
56 #include <types.h>
57 typedef signed char int8_t;
58 typedef unsigned char uint8_t;
59 typedef signed short int16_t;
60 typedef unsigned short uint16_t;
61 typedef signed int int32_t;
62 typedef unsigned int uint32_t;
63 typedef signed long long int64_t;
64 typedef unsigned long long uint64_t;
65 typedef float float32_t;
66 typedef double float64_t;
67 #else
68 // Define mkdir for GNUC
69 #include <sys/stat.h>
70 #include <sys/types.h>
71 #define _mkdir(dirname) mkdir(dirname, S_IRWXU)
72 #endif
73 #else
74 #include <direct.h>
75 #endif
76 
77 #endif // AS_PSVITA
78 #endif // _WIN32_WCE
79 #endif // AS_WII
80 
81 #endif // !defined(AS_DEBUG)
82 
83 
84 
85 #if defined(_MSC_VER) && defined(AS_PROFILE)
86 // Currently only do profiling with MSVC++
87 
88 #include <mmsystem.h>
89 #include <direct.h>
90 #include "as_string.h"
91 #include "as_map.h"
92 #include "as_string_util.h"
93 
94 BEGIN_AS_NAMESPACE
95 
96 struct TimeCount {
97  double time;
98  int count;
99  double max;
100  double min;
101 };
102 
103 class CProfiler {
104 public:
105  CProfiler() {
106  // We need to know how often the clock is updated
107  __int64 tps;
108  if (!QueryPerformanceFrequency((LARGE_INTEGER *)&tps))
109  usePerformance = false;
110  else {
111  usePerformance = true;
112  ticksPerSecond = double(tps);
113  }
114 
115  timeOffset = GetTime();
116  }
117 
118  ~CProfiler() {
119  WriteSummary();
120  }
121 
122  double GetTime() {
123  if (usePerformance) {
124  __int64 ticks;
125  QueryPerformanceCounter((LARGE_INTEGER *)&ticks);
126 
127  return double(ticks) / ticksPerSecond - timeOffset;
128  }
129 
130  return double(timeGetTime()) / 1000.0 - timeOffset;
131  }
132 
133  double Begin(const char *name) {
134  double time = GetTime();
135 
136  // Add the scope to the key
137  if (key.GetLength())
138  key += "|";
139  key += name;
140 
141  // Compensate for the time spent writing to the file
142  timeOffset += GetTime() - time;
143 
144  return time;
145  }
146 
147  void End(const char * /*name*/, double beginTime) {
148  double time = GetTime();
149 
150  double elapsed = time - beginTime;
151 
152  // Update the profile info for this scope
154  if (map.MoveTo(&cursor, key)) {
155  cursor->value.time += elapsed;
156  cursor->value.count++;
157  if (cursor->value.max < elapsed)
158  cursor->value.max = elapsed;
159  if (cursor->value.min > elapsed)
160  cursor->value.min = elapsed;
161  } else {
162  TimeCount tc = {elapsed, 1, elapsed, elapsed};
163  map.Insert(key, tc);
164  }
165 
166  // Remove the inner most scope from the key
167  int n = key.FindLast("|");
168  if (n > 0)
169  key.SetLength(n);
170  else
171  key.SetLength(0);
172 
173  // Compensate for the time spent writing to the file
174  timeOffset += GetTime() - time;
175  }
176 
177 protected:
178  void WriteSummary() {
179  // Write the analyzed info into a file for inspection
180  _mkdir("AS_DEBUG");
181  FILE *fp;
182 #if _MSC_VER >= 1500 && !defined(AS_MARMALADE)
183  fopen_s(&fp, "AS_DEBUG/profiling_summary.txt", "wt");
184 #else
185  fp = fopen("AS_DEBUG/profiling_summary.txt", "wt");
186 #endif
187  if (fp == 0)
188  return;
189 
190  fprintf(fp, "%-60s %10s %15s %15s %15s %15s\n\n", "Scope", "Count", "Tot time", "Avg time", "Max time", "Min time");
191 
193  map.MoveLast(&cursor);
194  while (cursor) {
195  asCString key = cursor->key;
196  int count;
197  int n = key.FindLast("|", &count);
198  if (count) {
199  key = asCString(" ", count) + key.SubString(n + 1);
200  }
201 
202  fprintf(fp, "%-60s %10d %15.6f %15.6f %15.6f %15.6f\n", key.AddressOf(), cursor->value.count, cursor->value.time, cursor->value.time / cursor->value.count, cursor->value.max, cursor->value.min);
203 
204  map.MovePrev(&cursor, cursor);
205  }
206 
207  fclose(fp);
208  }
209 
210  double timeOffset;
211  double ticksPerSecond;
212  bool usePerformance;
213 
214  asCString key;
216 };
217 
218 extern CProfiler g_profiler;
219 
220 class CProfilerScope {
221 public:
222  CProfilerScope(const char *name) {
223  this->name = name;
224  beginTime = g_profiler.Begin(name);
225  }
226 
227  ~CProfilerScope() {
228  g_profiler.End(name, beginTime);
229  }
230 
231 protected:
232  const char *name;
233  double beginTime;
234 };
235 
236 #define TimeIt(x) CProfilerScope profilescope(x)
237 
238 END_AS_NAMESPACE
239 
240 #else // !(_MSC_VER && AS_PROFILE)
241 
242 // Define it so nothing is done
243 #define TimeIt(x)
244 
245 #endif // !(_MSC_VER && AS_PROFILE)
246 
247 
248 
249 
250 #endif // defined(AS_DEBUG_H)
251 
252 
Definition: as_map.h:44
Definition: as_map.h:42
Definition: as_string.h:41