timing_flamegraph.cpp
Go to the documentation of this file.
1 /*
2  * OpenVINS: An Open Platform for Visual-Inertial Research
3  * Copyright (C) 2018-2023 Patrick Geneva
4  * Copyright (C) 2018-2023 Guoquan Huang
5  * Copyright (C) 2018-2023 OpenVINS Contributors
6  * Copyright (C) 2018-2019 Kevin Eckenhoff
7  *
8  * This program is free software: you can redistribute it and/or modify
9  * it under the terms of the GNU General Public License as published by
10  * the Free Software Foundation, either version 3 of the License, or
11  * (at your option) any later version.
12  *
13  * This program is distributed in the hope that it will be useful,
14  * but WITHOUT ANY WARRANTY; without even the implied warranty of
15  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16  * GNU General Public License for more details.
17  *
18  * You should have received a copy of the GNU General Public License
19  * along with this program. If not, see <https://www.gnu.org/licenses/>.
20  */
21 
22 #include <Eigen/Eigen>
23 #include <boost/algorithm/string/predicate.hpp>
24 #include <boost/filesystem.hpp>
25 #include <boost/foreach.hpp>
26 #include <fstream>
27 #include <iostream>
28 #include <string>
29 
30 #include "utils/Loader.h"
31 #include "utils/Statistics.h"
32 #include "utils/colors.h"
33 #include "utils/print.h"
34 
35 #ifdef HAVE_PYTHONLIBS
36 
37 // import the c++ wrapper for matplot lib
38 // https://github.com/lava/matplotlib-cpp
39 // sudo apt-get install python-matplotlib python-numpy python2.7-dev
40 #include "plot/matplotlibcpp.h"
41 
42 #endif
43 
44 int main(int argc, char **argv) {
45 
46  // Verbosity setting
48 
49  // Ensure we have a path
50  if (argc < 2) {
51  PRINT_ERROR(RED "ERROR: Please specify a timing file\n" RESET);
52  PRINT_ERROR(RED "ERROR: ./timing_flamegraph <file_times.txt> <subsample>\n" RESET);
53  PRINT_ERROR(RED "ERROR: rosrun ov_eval timing_flamegraph <file_times.txt> <subsample>\n" RESET);
54  std::exit(EXIT_FAILURE);
55  }
56  int keep_every = 10;
57  if (argc == 3) {
58  keep_every = atoi(argv[2]);
59  }
60 
61  // Load it!!
62  std::vector<std::string> names;
63  std::vector<double> times;
64  std::vector<Eigen::VectorXd> timing_values;
65  ov_eval::Loader::load_timing_flamegraph(argv[1], names, times, timing_values);
66  PRINT_INFO("[TIME]: loaded %d timestamps from file (%d categories)!!\n", (int)times.size(), (int)names.size());
67 
68  // Our categories
69  std::vector<ov_eval::Statistics> stats;
70  for (size_t i = 0; i < names.size(); i++)
71  stats.push_back(ov_eval::Statistics());
72 
73  // Loop through each and report the average timing information
74  for (size_t i = 0; i < times.size(); i++) {
75  for (size_t c = 0; c < names.size(); c++) {
76  stats.at(c).timestamps.push_back(times.at(i));
77  stats.at(c).values.push_back(timing_values.at(i)(c));
78  }
79  }
80 
81  // Now print the statistic for this run
82  for (size_t i = 0; i < names.size(); i++) {
83  stats.at(i).calculate();
84  PRINT_INFO("mean_time = %.4f | std = %.4f | 99th = %.4f | max = %.4f (%s)\n", stats.at(i).mean, stats.at(i).std,
85  stats.at(i).ninetynine, stats.at(i).max, names.at(i).c_str());
86  }
87 
88 #ifdef HAVE_PYTHONLIBS
89 
90  // Sub-sample the time
91  std::vector<double> times_skipped;
92  for (size_t t = 0; t < times.size(); t++) {
93  if (t % keep_every == 0) {
94  times_skipped.push_back(times.at(t));
95  }
96  }
97 
98  // Zero our time arrays
99  double starttime1 = (times_skipped.empty()) ? 0 : times_skipped.at(0);
100  double endtime1 = (times_skipped.empty()) ? 0 : times_skipped.at(times_skipped.size() - 1);
101  for (size_t j = 0; j < times_skipped.size(); j++) {
102  times_skipped.at(j) -= starttime1;
103  }
104 
105  // Valid colors
106  // https://matplotlib.org/stable/tutorials/colors/colors.html
107  // std::vector<std::string> colors_valid = {"blue","aqua","lightblue","lightgreen","yellowgreen","green"};
108  std::vector<std::string> colors_valid = {"navy", "blue", "lightgreen", "green", "gold", "goldenrod"};
109 
110  // Create vector for each category
111  // NOTE we skip the last category since it is the "total" time by convention
112  std::vector<std::string> labels;
113  std::vector<std::string> colors;
114  std::vector<std::vector<double>> timings;
115  for (size_t i = 0; i < names.size() - 1; i++) {
116  labels.push_back(names.at(i));
117  colors.push_back(colors_valid.at(i % colors_valid.size()));
118  std::vector<double> values_skipped;
119  for (size_t t = 0; t < stats.at(i).values.size(); t++) {
120  if (t % keep_every == 0) {
121  values_skipped.push_back(stats.at(i).values.at(t));
122  }
123  }
124  timings.push_back(values_skipped);
125  }
126 
127  // Plot this figure
128  matplotlibcpp::figure_size(1200, 400);
129  matplotlibcpp::stackplot(times_skipped, timings, labels, colors, "zero");
130  matplotlibcpp::ylabel("execution time (s)");
131  matplotlibcpp::xlim(0.0, endtime1 - starttime1);
132  // matplotlibcpp::ylim(0.0,stats.at(stats.size()-1).ninetynine);
133  matplotlibcpp::ylim(0.0, stats.at(stats.size() - 1).max);
134  matplotlibcpp::xlabel("dataset time (s)");
137 
138  // Display to the user
139  matplotlibcpp::show(true);
140 
141 #endif
142 
143  // Done!
144  return EXIT_SUCCESS;
145 }
Statistics object for a given set scalar time series values.
Definition: Statistics.h:39
void ylim(Numeric left, Numeric right)
#define RESET
int main(int argc, char **argv)
RED
void figure_size(size_t w, size_t h)
void show(const bool block=true)
static void setPrintLevel(const std::string &level)
void xlim(Numeric left, Numeric right)
bool stackplot(const std::vector< NumericX > &x, const std::vector< std::vector< NumericY >> &ys, const std::vector< std::string > &labels, const std::vector< std::string > &colors, const std::string &baseline="zero")
void xlabel(const std::string &str, const std::map< std::string, std::string > &keywords={})
void ylabel(const std::string &str, const std::map< std::string, std::string > &keywords={})
void tight_layout()
static void load_timing_flamegraph(std::string path, std::vector< std::string > &names, std::vector< double > &times, std::vector< Eigen::VectorXd > &timing_values)
Load comma separated timing file from pid_ros.py file.
Definition: Loader.cpp:240


ov_eval
Author(s): Patrick Geneva , Kevin Eckenhoff , Guoquan Huang
autogenerated on Wed Jun 21 2023 03:05:40