TAMSVIZ
Visualization and annotation tool for ROS
profiler.h
1 // TAMSVIZ
2 // (c) 2020 Philipp Ruppel
3 
4 #pragma once
5 
6 #include <atomic>
7 #include <chrono>
8 #include <condition_variable>
9 #include <memory>
10 #include <mutex>
11 #include <string>
12 #include <thread>
13 #include <unordered_set>
14 #include <vector>
15 
16 struct ProfilerData {
17  uint64_t time = 0;
18  uint64_t count = 0;
19 };
20 
22  std::string _name;
23  std::atomic<int64_t> _time;
24  std::atomic<int64_t> _count;
25 
26 public:
27  ProfilerTrack(const std::string &name);
28  ProfilerTrack(const ProfilerTrack &) = delete;
29  ProfilerTrack &operator=(const ProfilerTrack &) = delete;
30  ~ProfilerTrack();
31  const std::string &name() const { return _name; }
32  inline void add(int64_t time) {
33  _time += time;
34  _count++;
35  }
36  ProfilerData swap();
37 };
38 
40  ProfilerTrack &_track;
41  std::chrono::steady_clock::time_point _start;
42 
43 public:
44  inline ProfilerScope(ProfilerTrack &track)
45  : _track(track), _start(std::chrono::steady_clock::now()) {}
46  inline ~ProfilerScope() {
47  _track.add(std::chrono::duration_cast<std::chrono::microseconds, int64_t>(
48  std::chrono::steady_clock::now() - _start)
49  .count());
50  }
51  ProfilerScope(const ProfilerScope &) = delete;
52  ProfilerScope &operator=(const ProfilerScope &) = delete;
53 };
54 
55 class Profiler {
56  mutable std::mutex _mutex;
57  mutable std::vector<std::weak_ptr<ProfilerTrack>> _tracks;
58 
59 public:
60  static std::shared_ptr<Profiler> instance();
61  std::shared_ptr<ProfilerTrack>
62  track(const std::shared_ptr<ProfilerTrack> &track);
63  std::vector<std::shared_ptr<ProfilerTrack>> tracks() const;
64 };
65 
67  std::thread _thread;
68  bool _exit = false;
69  std::mutex _mutex;
70  std::condition_variable _condition;
71 
72 public:
74  const std::shared_ptr<Profiler> &profiler = Profiler::instance());
75  ~ProfilerThread();
76 };
77 
78 #define PROFILER(...) \
79  static std::shared_ptr<ProfilerTrack> profiler_track = \
80  Profiler::instance()->track(std::make_shared<ProfilerTrack>( \
81  std::string() + __PRETTY_FUNCTION__ + " " __VA_ARGS__)); \
82  ProfilerScope profiler_scope(*profiler_track);