| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255 |
- //
- // Copyright (c) 2008-2014 the Urho3D project.
- //
- // Permission is hereby granted, free of charge, to any person obtaining a copy
- // of this software and associated documentation files (the "Software"), to deal
- // in the Software without restriction, including without limitation the rights
- // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
- // copies of the Software, and to permit persons to whom the Software is
- // furnished to do so, subject to the following conditions:
- //
- // The above copyright notice and this permission notice shall be included in
- // all copies or substantial portions of the Software.
- //
- // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
- // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
- // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
- // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
- // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
- // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
- // THE SOFTWARE.
- //
- #pragma once
- #include "Str.h"
- #include "Timer.h"
- namespace Urho3D
- {
- /// Profiling data for one block in the profiling tree.
- class URHO3D_API ProfilerBlock
- {
- public:
- /// Construct with the specified parent block and name.
- ProfilerBlock(ProfilerBlock* parent, const char* name) :
- name_(name),
- time_(0),
- maxTime_(0),
- count_(0),
- parent_(parent),
- frameTime_(0),
- frameMaxTime_(0),
- frameCount_(0),
- intervalTime_(0),
- intervalMaxTime_(0),
- intervalCount_(0),
- totalTime_(0),
- totalMaxTime_(0),
- totalCount_(0)
- {
- }
-
- /// Destruct. Free the child blocks.
- ~ProfilerBlock()
- {
- for (PODVector<ProfilerBlock*>::Iterator i = children_.Begin(); i != children_.End(); ++i)
- {
- delete *i;
- *i = 0;
- }
- }
-
- /// Begin timing.
- void Begin()
- {
- timer_.Reset();
- ++count_;
- }
-
- /// End timing.
- void End()
- {
- long long time = timer_.GetUSec(false);
- if (time > maxTime_)
- maxTime_ = time;
- time_ += time;
- }
-
- /// End profiling frame and update interval and total values.
- void EndFrame()
- {
- frameTime_ = time_;
- frameMaxTime_ = maxTime_;
- frameCount_ = count_;
- intervalTime_ += time_;
- if (maxTime_ > intervalMaxTime_)
- intervalMaxTime_ = maxTime_;
- intervalCount_ += count_;
- totalTime_ += time_;
- if (maxTime_ > totalMaxTime_)
- totalMaxTime_ = maxTime_;
- totalCount_ += count_;
- time_ = 0;
- maxTime_ = 0;
- count_ = 0;
-
- for (PODVector<ProfilerBlock*>::Iterator i = children_.Begin(); i != children_.End(); ++i)
- (*i)->EndFrame();
- }
-
- /// Begin new profiling interval.
- void BeginInterval()
- {
- intervalTime_ = 0;
- intervalMaxTime_ = 0;
- intervalCount_ = 0;
-
- for (PODVector<ProfilerBlock*>::Iterator i = children_.Begin(); i != children_.End(); ++i)
- (*i)->BeginInterval();
- }
-
- /// Return child block with the specified name.
- ProfilerBlock* GetChild(const char* name)
- {
- // First check using string pointers only, then resort to actual strcmp
- for (PODVector<ProfilerBlock*>::Iterator i = children_.Begin(); i != children_.End(); ++i)
- {
- if ((*i)->name_ == name)
- return *i;
- }
-
- for (PODVector<ProfilerBlock*>::Iterator i = children_.Begin(); i != children_.End(); ++i)
- {
- if (!String::Compare((*i)->name_, name, true))
- return *i;
- }
-
- ProfilerBlock* newBlock = new ProfilerBlock(this, name);
- children_.Push(newBlock);
-
- return newBlock;
- }
-
- /// Block name.
- const char* name_;
- /// High-resolution timer for measuring the block duration.
- HiresTimer timer_;
- /// Time on current frame.
- long long time_;
- /// Maximum time on current frame.
- long long maxTime_;
- /// Calls on current frame.
- unsigned count_;
- /// Parent block.
- ProfilerBlock* parent_;
- /// Child blocks.
- PODVector<ProfilerBlock*> children_;
- /// Time on the previous frame.
- long long frameTime_;
- /// Maximum time on the previous frame.
- long long frameMaxTime_;
- /// Calls on the previous frame.
- unsigned frameCount_;
- /// Time during current profiler interval.
- long long intervalTime_;
- /// Maximum time during current profiler interval.
- long long intervalMaxTime_;
- /// Calls during current profiler interval.
- unsigned intervalCount_;
- /// Total accumulated time.
- long long totalTime_;
- /// All-time maximum time.
- long long totalMaxTime_;
- /// Total accumulated calls.
- unsigned totalCount_;
- };
- /// Hierarchical performance profiler subsystem.
- class URHO3D_API Profiler : public Object
- {
- OBJECT(Profiler);
-
- public:
- /// Construct.
- Profiler(Context* context);
- /// Destruct.
- virtual ~Profiler();
-
- /// Begin timing a profiling block.
- void BeginBlock(const char* name)
- {
- current_ = current_->GetChild(name);
- current_->Begin();
- }
-
- /// End timing the current profiling block.
- void EndBlock()
- {
- if (current_ != root_)
- {
- current_->End();
- current_ = current_->parent_;
- }
- }
-
- /// Begin the profiling frame. Called by HandleBeginFrame().
- void BeginFrame();
- /// End the profiling frame. Called by HandleEndFrame().
- void EndFrame();
- /// Begin a new interval.
- void BeginInterval();
-
- /// Return profiling data as text output.
- String GetData(bool showUnused = false, bool showTotal = false, unsigned maxDepth = M_MAX_UNSIGNED) const;
- /// Return the current profiling block.
- const ProfilerBlock* GetCurrentBlock() { return current_; }
- /// Return the root profiling block.
- const ProfilerBlock* GetRootBlock() { return root_; }
-
- private:
- /// Return profiling data as text output for a specified profiling block.
- void GetData(ProfilerBlock* block, String& output, unsigned depth, unsigned maxDepth, bool showUnused, bool showTotal) const;
-
- /// Current profiling block.
- ProfilerBlock* current_;
- /// Root profiling block.
- ProfilerBlock* root_;
- /// Frames in the current interval.
- unsigned intervalFrames_;
- /// Total frames.
- unsigned totalFrames_;
- };
- /// Helper class for automatically beginning and ending a profiling block
- class URHO3D_API AutoProfileBlock
- {
- public:
- /// Construct. Begin a profiling block with the specified name and optional call count.
- AutoProfileBlock(Profiler* profiler, const char* name) :
- profiler_(profiler)
- {
- if (profiler_)
- profiler_->BeginBlock(name);
- }
-
- /// Destruct. End the profiling block.
- ~AutoProfileBlock()
- {
- if (profiler_)
- profiler_->EndBlock();
- }
-
- private:
- /// Profiler.
- Profiler* profiler_;
- };
- #ifdef ENABLE_PROFILING
- #define PROFILE(name) AutoProfileBlock profile_ ## name (GetSubsystem<Profiler>(), #name)
- #else
- #define PROFILE(name)
- #endif
- }
|