folly/folly/test/ConstexprMathBenchmark.cpp

/*
 * Copyright (c) Meta Platforms, Inc. and affiliates.
 *
 * 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.
 */

#include <folly/ConstexprMath.h>

#include <limits>
#include <type_traits>

#include <glog/logging.h>

#include <folly/Benchmark.h>

template <typename ValueT>
constexpr ValueT UBSafeAdd(ValueT a, ValueT b) {
  using UnsignedT = typename std::make_unsigned<ValueT>::type;
  return static_cast<ValueT>(
      static_cast<UnsignedT>(a) + static_cast<UnsignedT>(b));
}

template <typename ValueT>
constexpr ValueT UBSafeSub(ValueT a, ValueT b) {
  using UnsignedT = typename std::make_unsigned<ValueT>::type;
  return static_cast<ValueT>(
      static_cast<UnsignedT>(a) - static_cast<UnsignedT>(b));
}

template <typename ValueT, typename Op>
void Run(size_t iterations, ValueT kMin, ValueT kMax, Op&& op) {
  auto kMid = (kMin + kMax) / 2;

  for (size_t round = 0; round < iterations; round++) {
    for (ValueT a = kMin; a < kMin + 100; a++) {
      for (ValueT b = kMin; b < kMin + 100; b++) {
        auto a1 = a, b1 = b;
        folly::makeUnpredictable(a1);
        folly::makeUnpredictable(b1);
        ValueT c = op(a1, b1);
        folly::doNotOptimizeAway(c);
      }
    }
    for (ValueT a = kMin; a < kMin + 100; a++) {
      for (ValueT b = kMid - 50; b < kMid + 50; b++) {
        auto a1 = a, b1 = b;
        folly::makeUnpredictable(a1);
        folly::makeUnpredictable(b1);
        ValueT c = op(a1, b1);
        folly::doNotOptimizeAway(c);
      }
    }
    for (ValueT a = kMin; a < kMin + 100; a++) {
      for (ValueT b = kMax - 100; b < kMax; b++) {
        auto a1 = a, b1 = b;
        folly::makeUnpredictable(a1);
        folly::makeUnpredictable(b1);
        ValueT c = op(a1, b1);
        folly::doNotOptimizeAway(c);
      }
    }
    for (ValueT a = kMid - 50; a < kMid + 50; a++) {
      for (ValueT b = kMin; b < kMin + 100; b++) {
        auto a1 = a, b1 = b;
        folly::makeUnpredictable(a1);
        folly::makeUnpredictable(b1);
        ValueT c = op(a1, b1);
        folly::doNotOptimizeAway(c);
      }
    }
    for (ValueT a = kMid - 50; a < kMid + 50; a++) {
      for (ValueT b = kMid - 50; b < kMid + 50; b++) {
        auto a1 = a, b1 = b;
        folly::makeUnpredictable(a1);
        folly::makeUnpredictable(b1);
        ValueT c = op(a1, b1);
        folly::doNotOptimizeAway(c);
      }
    }
    for (ValueT a = kMid - 50; a < kMid + 50; a++) {
      for (ValueT b = kMax - 100; b < kMax; b++) {
        auto a1 = a, b1 = b;
        folly::makeUnpredictable(a1);
        folly::makeUnpredictable(b1);
        ValueT c = op(a1, b1);
        folly::doNotOptimizeAway(c);
      }
    }
    for (ValueT a = kMax - 100; a < kMax; a++) {
      for (ValueT b = kMin; b < kMin + 100; b++) {
        auto a1 = a, b1 = b;
        folly::makeUnpredictable(a1);
        folly::makeUnpredictable(b1);
        ValueT c = op(a1, b1);
        folly::doNotOptimizeAway(c);
      }
    }
    for (ValueT a = kMax - 100; a < kMax; a++) {
      for (ValueT b = kMid - 50; b < kMid + 50; b++) {
        auto a1 = a, b1 = b;
        folly::makeUnpredictable(a1);
        folly::makeUnpredictable(b1);
        ValueT c = op(a1, b1);
        folly::doNotOptimizeAway(c);
      }
    }
    for (ValueT a = kMax - 100; a < kMax; a++) {
      for (ValueT b = kMax - 100; b < kMax; b++) {
        auto a1 = a, b1 = b;
        folly::makeUnpredictable(a1);
        folly::makeUnpredictable(b1);
        ValueT c = op(a1, b1);
        folly::doNotOptimizeAway(c);
      }
    }
  }
}

template <typename ValueT>
void Add(size_t iterations, ValueT kMin, ValueT kMax) {
  Run<ValueT>(iterations, kMin, kMax, [](ValueT a, ValueT b) {
    return UBSafeAdd(a, b);
  });
}

template <typename ValueT>
void NoOverflowAdd(size_t iterations, ValueT kMin, ValueT kMax) {
  Run<ValueT>(iterations, kMin, kMax, [](ValueT a, ValueT b) {
    return folly::constexpr_add_overflow_clamped(a, b);
  });
}

template <typename ValueT>
void Sub(size_t iterations, ValueT kMin, ValueT kMax) {
  Run<ValueT>(iterations, kMin, kMax, [](ValueT a, ValueT b) {
    return UBSafeSub(a, b);
  });
}

template <typename ValueT>
void NoOverflowSub(size_t iterations, ValueT kMin, ValueT kMax) {
  Run<ValueT>(iterations, kMin, kMax, [](ValueT a, ValueT b) {
    return folly::constexpr_sub_overflow_clamped(a, b);
  });
}

#define GENERATE_BENCHMARKS_FOR_TYPE(ValueT) \
  BENCHMARK_NAMED_PARAM(                     \
      Add,                                   \
      ValueT,                                \
      std::numeric_limits<ValueT>::min(),    \
      std::numeric_limits<ValueT>::max())    \
  BENCHMARK_RELATIVE_NAMED_PARAM(            \
      NoOverflowAdd,                         \
      ValueT,                                \
      std::numeric_limits<ValueT>::min(),    \
      std::numeric_limits<ValueT>::max())    \
  BENCHMARK_NAMED_PARAM(                     \
      Sub,                                   \
      ValueT,                                \
      std::numeric_limits<ValueT>::min(),    \
      std::numeric_limits<ValueT>::max())    \
  BENCHMARK_RELATIVE_NAMED_PARAM(            \
      NoOverflowSub,                         \
      ValueT,                                \
      std::numeric_limits<ValueT>::min(),    \
      std::numeric_limits<ValueT>::max())

GENERATE_BENCHMARKS_FOR_TYPE(int8_t)
BENCHMARK_DRAW_LINE();
GENERATE_BENCHMARKS_FOR_TYPE(uint8_t)
BENCHMARK_DRAW_LINE();
GENERATE_BENCHMARKS_FOR_TYPE(int16_t)
BENCHMARK_DRAW_LINE();
GENERATE_BENCHMARKS_FOR_TYPE(uint16_t)
BENCHMARK_DRAW_LINE();
GENERATE_BENCHMARKS_FOR_TYPE(int32_t)
BENCHMARK_DRAW_LINE();
GENERATE_BENCHMARKS_FOR_TYPE(uint32_t)
BENCHMARK_DRAW_LINE();
GENERATE_BENCHMARKS_FOR_TYPE(int64_t)
BENCHMARK_DRAW_LINE();
GENERATE_BENCHMARKS_FOR_TYPE(uint64_t)

int main(int argc, char** argv) {
  gflags::ParseCommandLineFlags(&argc, &argv, true);
  folly::runBenchmarks();
  return 0;
}