/*
 * Copyright (C) 2012 The Android Open Source Project
 *
 * 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.
 */

#ifndef BENCHMARKS_BENCHMARK_H_
#define BENCHMARKS_BENCHMARK_H_

#include <regex.h>
#include <stdint.h>

#include <string>
#include <vector>

namespace testing {

class Benchmark {
public:
  Benchmark() {
    List().push_back(this);
  }
  virtual ~Benchmark() {}

  virtual std::string Name() = 0;

  virtual size_t RunAllArgs(std::vector<regex_t*>&) = 0;

  void SetBenchmarkBytesProcessed(uint64_t bytes) { bytes_processed_ += bytes; }
  void StopBenchmarkTiming();
  void StartBenchmarkTiming();

  // Run all of the benchmarks that have registered.
  static size_t RunAll(std::vector<regex_t*>&);

  static std::vector<Benchmark*>& List();

  static int MaxNameColumnWidth();

protected:
  virtual size_t NameColumnWidth() = 0;

  uint64_t bytes_processed_;
  uint64_t total_time_ns_;
  uint64_t start_time_ns_;

  static bool header_printed_;

  static void PrintHeader();
};

template <typename T>
class BenchmarkT : public Benchmark {
public:
  BenchmarkT() {}
  virtual ~BenchmarkT() {}

protected:
  bool ShouldRun(std::vector<regex_t*>&, T arg);
  void RunWithArg(T arg);
  virtual void RunIterations(int, T) = 0;
  virtual std::string GetNameStr(T) = 0;
};

class BenchmarkWithoutArg : public BenchmarkT<void*> {
public:
  BenchmarkWithoutArg() {}
  virtual ~BenchmarkWithoutArg() {}

protected:
  virtual size_t RunAllArgs(std::vector<regex_t*>& regs) override {
    size_t benchmarks_run = 0;
    if (BenchmarkT<void*>::ShouldRun(regs, nullptr)) {
      PrintHeader();
      RunWithArg(nullptr);
      benchmarks_run++;
    }
    return benchmarks_run;
  }

  virtual void RunIterations(int iters, void*) override {
    Run(iters);
  }

  virtual void Run(int) = 0;

  virtual size_t NameColumnWidth() override {
    return Name().size();
  }

  virtual std::string GetNameStr(void *) override;
};

template<typename T>
class BenchmarkWithArg : public BenchmarkT<T> {
public:
  BenchmarkWithArg() {}
  virtual ~BenchmarkWithArg() {}

  BenchmarkWithArg* Arg(T arg) {
    args_.push_back(arg);
    return this;
  }

protected:
  virtual size_t NameColumnWidth() override {
    size_t max = 0;
    for (const auto& arg : args_) {
      max = std::max(max, GetNameStr(arg).size());
    }
    return max;
  }

  std::string GetNameStr(T arg) override;

  virtual size_t RunAllArgs(std::vector<regex_t*>& regs) override {
    size_t benchmarks_run = 0;
    for (T& arg : args_) {
      if (BenchmarkT<T>::ShouldRun(regs, arg)) {
        Benchmark::PrintHeader();
        BenchmarkT<T>::RunWithArg(arg);
        benchmarks_run++;
      }
    }
    return benchmarks_run;
  }

  virtual void RunIterations(int iters, T arg) override {
    Run(iters, arg);
  }

  virtual void Run(int iters, T arg) = 0;

private:
  std::vector<T> args_;
};

}  // namespace testing

#define BENCHMARK_START(f, super_class) \
  class f : public super_class { \
  public: \
    f() {} \
    virtual ~f() {} \
    virtual std::string Name() override { return #f; } \

#define BENCHMARK_NO_ARG(f) \
  BENCHMARK_START(f, ::testing::BenchmarkWithoutArg) \
    virtual void Run(int) override; \
  }; \
  static ::testing::Benchmark* __benchmark_##f = new f()

#define BENCHMARK_WITH_ARG(f, arg_type) \
  BENCHMARK_START(f, ::testing::BenchmarkWithArg<arg_type>) \
    virtual void Run(int, arg_type) override; \
  }; \
  static ::testing::BenchmarkWithArg<arg_type>* __benchmark_##f = (new f())

#endif  // BENCHMARKS_BENCHMARK_H_