chromium/third_party/mediapipe/src/mediapipe/util/tracking/measure_time.h

// Copyright 2019 The MediaPipe Authors.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
//      http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//
// Helper class and macro to take time measurements within current scope.
// Takes time measurement within current scope. Outputs to ABSL_LOG(INFO) if
// flag --measure_time is set or if build flag SET_FLAG_MEASURE_TIME is
// defined (add --copt=-DSET_FLAG_MEASURE_TIME to your build command).
// Additionally you can limit time measurements to specific files,
// via the flag
// --measure_time_filter="<comma separated list of substrings of file names>"
// Example:
// {     // Scope to be measured
//   MEASURE_TIME << "Some additional logging and answers : " << 42;
//   ...  // instructions.
// }

#ifndef MEDIAPIPE_UTIL_TRACKING_MEASURE_TIME_H_
#define MEDIAPIPE_UTIL_TRACKING_MEASURE_TIME_H_

#include <cstdint>
#include <memory>
#include <sstream>

#include "absl/log/absl_check.h"
#include "absl/log/absl_log.h"
#include "absl/strings/str_split.h"
#include "absl/strings/string_view.h"
#include "absl/synchronization/mutex.h"
#include "absl/time/clock.h"

extern bool flags_measure_time;

namespace mediapipe {
class MeasureTimeFilter;
}  // namespace mediapipe

#define MEASURE_TIME \
  MEASURE_TIME_PRE_IMPL(flags_measure_time, __LINE__, __FILE__)

// Add level of indirection to trigger __LINE__ macro expansion.
#define MEASURE_TIME_PRE_IMPL(show_output, line, file) \
  MEASURE_TIME_IMPL(show_output, line, file)

#define MEASURE_TIME_IMPL(show_output, line, file)                      \
  std::unique_ptr<mediapipe::ScopedWallTimer> scoped_wall_timer_##line; \
  const bool activated##line = show_output;                             \
  if (activated##line) {                                                \
    static mediapipe::ScopedWallTimer::Accumulator*                     \
        scoped_wall_timer_accum_##line =                                \
            new mediapipe::ScopedWallTimer::Accumulator;                \
    scoped_wall_timer_##line.reset(new mediapipe::ScopedWallTimer(      \
        file, line, show_output, scoped_wall_timer_accum_##line));      \
  }                                                                     \
  if (activated##line) /* NOLINT */                                     \
  scoped_wall_timer_##line->stream()

namespace mediapipe {

class ScopedWallTimer {
 public:
  // Helper class for accumulating time across multiple calls to a scoped wall
  // timer. Thread-safe (except on ANDROID, for which Mutex is mocked out).
  class Accumulator {
   public:
    Accumulator() : accum_time_(0.0f), count_(0) {}
    Accumulator(const Accumulator&) = delete;
    Accumulator& operator=(const Accumulator&) = delete;

    // Accumulates passed_time into accumulators. Returns total time and number
    // of calls made.
    void Accumulate(double passed_time, double* accum_time, int* count) {
      absl::MutexLock lock(&mutex_);
      accum_time_ += passed_time;
      ++count_;
      *accum_time = accum_time_;
      *count = count_;
    }

   private:
    double accum_time_;
    int count_;
    absl::Mutex mutex_;
  };

  // Creates a new ScopedWallTimer for current file and line. LogMessage is only
  // initialized if show_output is set to true.
  ScopedWallTimer(const char* file, int line, bool show_output,
                  Accumulator* accumulator)
      : file_(file),
        line_(line),
        show_output_(show_output),
        accumulator_(accumulator) {
    if (show_output_) {
      ABSL_CHECK(accumulator_);
      start_time_ = GetWallTime();
    }
  }
  ScopedWallTimer(const ScopedWallTimer&) = delete;
  ScopedWallTimer& operator=(const ScopedWallTimer&) = delete;

  // Destructor measures time and outputs to stream.
  ~ScopedWallTimer() {
    if (show_output_) {
      double passed_time = GetWallTime() - start_time_;
      double accum_time = 0.0;
      int count = 0;
      accumulator_->Accumulate(passed_time, &accum_time, &count);
      ABSL_LOG(INFO) << stream_.str() << " TIMES: [Curr: " << passed_time * 1e-6
                     << " ms, "
                     << "Avg: " << accum_time * 1e-6 / std::max(1, count)
                     << " ms, " << count << " calls]";
    }
  }

  std::ostream& stream() { return stream_; }

 private:
  const char* file_;
  int line_;
  bool show_output_;
  // We need to buffer information passed via stream operator <<
  // While LogMessage is adequate for this, no good equivalent exists on
  // Android, so we employ a portable ostringstream for buffering.
  std::ostringstream stream_;
  int64_t start_time_;
  Accumulator* accumulator_;

  int64_t GetWallTime() { return absl::GetCurrentTimeNanos(); }
};

class MeasureTimeFilter {
 public:
  static const MeasureTimeFilter* get() {
    static MeasureTimeFilter instance;
    return &instance;
  }
  MeasureTimeFilter(const MeasureTimeFilter&) = delete;
  MeasureTimeFilter& operator=(const MeasureTimeFilter&) = delete;

  bool Matches(const std::string& item) const {
    for (const std::string& match_item : match_items_) {
      if (item.find(match_item) != std::string::npos) {
        return true;
      }
    }
    return false;
  }

 private:
  explicit MeasureTimeFilter() {}
  explicit MeasureTimeFilter(const std::string& filter) {
    match_items_ = absl::StrSplit(filter, absl::ByChar(','));
  }
  std::vector<std::string> match_items_;
};

}  // namespace mediapipe

#endif  // MEDIAPIPE_UTIL_TRACKING_MEASURE_TIME_H_