Source code

Revision control

Copy as Markdown

Other Tools

// Copyright 2025 The Abseil 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
//
//
// 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 <string>
#include "absl/status/status.h"
#include "absl/status/statusor.h"
#include "benchmark/benchmark.h"
namespace {
void BM_StatusOrInt_CtorStatus(benchmark::State& state) {
for (auto _ : state) {
absl::StatusOr<int> status(absl::CancelledError());
benchmark::DoNotOptimize(status);
}
}
BENCHMARK(BM_StatusOrInt_CtorStatus);
void BM_StatusOrInt_CtorStatusWithMessage(benchmark::State& state) {
for (auto _ : state) {
absl::StatusOr<int> status(
absl::UnknownError("This string is 28 characters"));
benchmark::DoNotOptimize(status);
}
}
BENCHMARK(BM_StatusOrInt_CtorStatusWithMessage);
void BM_StatusOrInt_CopyCtor_Error(benchmark::State& state) {
for (auto _ : state) {
absl::StatusOr<int> original(
absl::UnknownError("This string is 28 characters"));
benchmark::DoNotOptimize(original);
absl::StatusOr<int> status(original);
benchmark::DoNotOptimize(status);
}
}
BENCHMARK(BM_StatusOrInt_CopyCtor_Error);
void BM_StatusOrInt_CopyCtor_Ok(benchmark::State& state) {
for (auto _ : state) {
absl::StatusOr<int> original(42);
benchmark::DoNotOptimize(original);
absl::StatusOr<int> status(original);
benchmark::DoNotOptimize(status);
}
}
BENCHMARK(BM_StatusOrInt_CopyCtor_Ok);
void BM_StatusOrInt_MoveCtor_Error(benchmark::State& state) {
for (auto _ : state) {
absl::StatusOr<int> original(
absl::UnknownError("This string is 28 characters"));
benchmark::DoNotOptimize(original);
absl::StatusOr<int> status(std::move(original));
benchmark::DoNotOptimize(status);
}
}
BENCHMARK(BM_StatusOrInt_MoveCtor_Error);
void BM_StatusOrInt_MoveCtor_Ok(benchmark::State& state) {
for (auto _ : state) {
absl::StatusOr<int> original(42);
benchmark::DoNotOptimize(original);
absl::StatusOr<int> status(std::move(original));
benchmark::DoNotOptimize(status);
}
}
BENCHMARK(BM_StatusOrInt_MoveCtor_Ok);
void BM_StatusOrInt_CopyAssign_Error(benchmark::State& state) {
for (auto _ : state) {
absl::StatusOr<int> original(
absl::UnknownError("This string is 28 characters"));
benchmark::DoNotOptimize(original);
absl::StatusOr<int> status(42);
benchmark::DoNotOptimize(status);
benchmark::DoNotOptimize(status = original);
benchmark::DoNotOptimize(status);
}
}
BENCHMARK(BM_StatusOrInt_CopyAssign_Error);
void BM_StatusOrInt_CopyAssign_Ok(benchmark::State& state) {
for (auto _ : state) {
absl::StatusOr<int> original(42);
benchmark::DoNotOptimize(original);
absl::StatusOr<int> status(42);
benchmark::DoNotOptimize(status);
benchmark::DoNotOptimize(status = original);
benchmark::DoNotOptimize(status);
}
}
BENCHMARK(BM_StatusOrInt_CopyAssign_Ok);
void BM_StatusOrInt_MoveAssign_Error(benchmark::State& state) {
for (auto _ : state) {
absl::StatusOr<int> original(
absl::UnknownError("This string is 28 characters"));
benchmark::DoNotOptimize(original);
absl::StatusOr<int> status(42);
benchmark::DoNotOptimize(status);
benchmark::DoNotOptimize(status = std::move(original));
benchmark::DoNotOptimize(status);
}
}
BENCHMARK(BM_StatusOrInt_MoveAssign_Error);
void BM_StatusOrInt_MoveAssign_Ok(benchmark::State& state) {
for (auto _ : state) {
absl::StatusOr<int> original(42);
benchmark::DoNotOptimize(original);
absl::StatusOr<int> status(42);
benchmark::DoNotOptimize(status);
benchmark::DoNotOptimize(status = std::move(original));
benchmark::DoNotOptimize(status);
}
}
BENCHMARK(BM_StatusOrInt_MoveAssign_Ok);
void BM_StatusOrInt_OkMethod_Error(benchmark::State& state) {
absl::StatusOr<int> status(
absl::UnknownError("This string is 28 characters"));
for (auto _ : state) {
benchmark::DoNotOptimize(status);
benchmark::DoNotOptimize(status.ok());
}
}
BENCHMARK(BM_StatusOrInt_OkMethod_Error);
void BM_StatusOrInt_OkMethod_Ok(benchmark::State& state) {
absl::StatusOr<int> status(42);
for (auto _ : state) {
benchmark::DoNotOptimize(status);
benchmark::DoNotOptimize(status.ok());
}
}
BENCHMARK(BM_StatusOrInt_OkMethod_Ok);
void BM_StatusOrInt_StatusMethod_Error(benchmark::State& state) {
for (auto _ : state) {
absl::StatusOr<int> status(
absl::UnknownError("This string is 28 characters"));
benchmark::DoNotOptimize(status);
benchmark::DoNotOptimize(status.status().ok());
}
}
BENCHMARK(BM_StatusOrInt_StatusMethod_Error);
void BM_StatusOrInt_StatusMethod_Ok(benchmark::State& state) {
for (auto _ : state) {
absl::StatusOr<int> status(42);
benchmark::DoNotOptimize(status);
benchmark::DoNotOptimize(std::move(status).status().ok());
}
}
BENCHMARK(BM_StatusOrInt_StatusMethod_Ok);
void BM_StatusOrInt_StatusMethodRvalue_Error(benchmark::State& state) {
for (auto _ : state) {
absl::StatusOr<int> status(
absl::UnknownError("This string is 28 characters"));
benchmark::DoNotOptimize(status);
benchmark::DoNotOptimize(std::move(status).status().ok());
}
}
BENCHMARK(BM_StatusOrInt_StatusMethodRvalue_Error);
void BM_StatusOrInt_StatusMethodRvalue_Ok(benchmark::State& state) {
for (auto _ : state) {
absl::StatusOr<int> status(42);
benchmark::DoNotOptimize(status);
benchmark::DoNotOptimize(std::move(status).status());
}
}
BENCHMARK(BM_StatusOrInt_StatusMethodRvalue_Ok);
void BM_StatusOrString_CtorStatus(benchmark::State& state) {
for (auto _ : state) {
absl::StatusOr<std::string> status(absl::CancelledError());
benchmark::DoNotOptimize(status);
}
}
BENCHMARK(BM_StatusOrString_CtorStatus);
void BM_StatusOrString_CtorStatusWithMessage(benchmark::State& state) {
for (auto _ : state) {
absl::StatusOr<std::string> status(
absl::UnknownError("This string is 28 characters"));
benchmark::DoNotOptimize(status);
}
}
BENCHMARK(BM_StatusOrString_CtorStatusWithMessage);
void BM_StatusOrString_CopyCtor_Error(benchmark::State& state) {
for (auto _ : state) {
absl::StatusOr<std::string> original(
absl::UnknownError("This string is 28 characters"));
benchmark::DoNotOptimize(original);
absl::StatusOr<std::string> status(original);
benchmark::DoNotOptimize(status);
}
}
BENCHMARK(BM_StatusOrString_CopyCtor_Error);
void BM_StatusOrString_CopyCtor_Ok(benchmark::State& state) {
for (auto _ : state) {
absl::StatusOr<std::string> original("This string is 28 characters");
benchmark::DoNotOptimize(original);
absl::StatusOr<std::string> status(original);
benchmark::DoNotOptimize(status);
}
}
BENCHMARK(BM_StatusOrString_CopyCtor_Ok);
void BM_StatusOrString_MoveCtor_Error(benchmark::State& state) {
for (auto _ : state) {
absl::StatusOr<std::string> original(
absl::UnknownError("This string is 28 characters"));
benchmark::DoNotOptimize(original);
absl::StatusOr<std::string> status(std::move(original));
benchmark::DoNotOptimize(status);
}
}
BENCHMARK(BM_StatusOrString_MoveCtor_Error);
void BM_StatusOrString_MoveCtor_Ok(benchmark::State& state) {
for (auto _ : state) {
absl::StatusOr<std::string> original("This string is 28 characters");
benchmark::DoNotOptimize(original);
absl::StatusOr<std::string> status(std::move(original));
benchmark::DoNotOptimize(status);
}
}
BENCHMARK(BM_StatusOrString_MoveCtor_Ok);
void BM_StatusOrString_CopyAssign_Error(benchmark::State& state) {
for (auto _ : state) {
absl::StatusOr<std::string> original(
absl::UnknownError("This string is 28 characters"));
benchmark::DoNotOptimize(original);
absl::StatusOr<std::string> status("This string is 28 characters");
benchmark::DoNotOptimize(status);
benchmark::DoNotOptimize(status = original);
benchmark::DoNotOptimize(status);
}
}
BENCHMARK(BM_StatusOrString_CopyAssign_Error);
void BM_StatusOrString_CopyAssign_Ok(benchmark::State& state) {
for (auto _ : state) {
absl::StatusOr<std::string> original("This string is 28 characters");
benchmark::DoNotOptimize(original);
absl::StatusOr<std::string> status("This string is 28 characters");
benchmark::DoNotOptimize(status);
benchmark::DoNotOptimize(status = original);
benchmark::DoNotOptimize(status);
}
}
BENCHMARK(BM_StatusOrString_CopyAssign_Ok);
void BM_StatusOrString_MoveAssign_Error(benchmark::State& state) {
for (auto _ : state) {
absl::StatusOr<std::string> original(
absl::UnknownError("This string is 28 characters"));
benchmark::DoNotOptimize(original);
absl::StatusOr<std::string> status("This string is 28 characters");
benchmark::DoNotOptimize(status);
benchmark::DoNotOptimize(status = std::move(original));
benchmark::DoNotOptimize(status);
}
}
BENCHMARK(BM_StatusOrString_MoveAssign_Error);
void BM_StatusOrString_MoveAssign_Ok(benchmark::State& state) {
for (auto _ : state) {
absl::StatusOr<std::string> original("This string is 28 characters");
benchmark::DoNotOptimize(original);
absl::StatusOr<std::string> status("This string is 28 characters");
benchmark::DoNotOptimize(status);
benchmark::DoNotOptimize(status = std::move(original));
benchmark::DoNotOptimize(status);
}
}
BENCHMARK(BM_StatusOrString_MoveAssign_Ok);
void BM_StatusOrString_OkMethod_Error(benchmark::State& state) {
for (auto _ : state) {
absl::StatusOr<std::string> status(
absl::UnknownError("This string is 28 characters"));
benchmark::DoNotOptimize(status);
benchmark::DoNotOptimize(status.ok());
}
}
BENCHMARK(BM_StatusOrString_OkMethod_Error);
void BM_StatusOrString_OkMethod_Ok(benchmark::State& state) {
for (auto _ : state) {
absl::StatusOr<std::string> status("This string is 28 characters");
benchmark::DoNotOptimize(status);
benchmark::DoNotOptimize(status.ok());
}
}
BENCHMARK(BM_StatusOrString_OkMethod_Ok);
void BM_StatusOrString_StatusMethod_Error(benchmark::State& state) {
for (auto _ : state) {
absl::StatusOr<std::string> status(
absl::UnknownError("This string is 28 characters"));
benchmark::DoNotOptimize(status);
benchmark::DoNotOptimize(status.status().ok());
}
}
BENCHMARK(BM_StatusOrString_StatusMethod_Error);
void BM_StatusOrString_StatusMethod_Ok(benchmark::State& state) {
for (auto _ : state) {
absl::StatusOr<std::string> status("This string is 28 characters");
benchmark::DoNotOptimize(status);
benchmark::DoNotOptimize(status.status().ok());
}
}
BENCHMARK(BM_StatusOrString_StatusMethod_Ok);
void BM_StatusOrString_StatusMethodRvalue_Error(benchmark::State& state) {
for (auto _ : state) {
absl::StatusOr<std::string> status(
absl::UnknownError("This string is 28 characters"));
benchmark::DoNotOptimize(status);
benchmark::DoNotOptimize(std::move(status).status());
}
}
BENCHMARK(BM_StatusOrString_StatusMethodRvalue_Error);
void BM_StatusOrString_StatusMethodRvalue_Ok(benchmark::State& state) {
for (auto _ : state) {
absl::StatusOr<std::string> status("This string is 28 characters");
benchmark::DoNotOptimize(status);
benchmark::DoNotOptimize(std::move(status).status());
}
}
BENCHMARK(BM_StatusOrString_StatusMethodRvalue_Ok);
// Benchmarks comparing a few alternative ways of structuring an interface
// for returning an int64 on success or an error. See (a), (b), (c), (d)
// below for the variants.
bool bm_cond = true;
bool SimpleIntInterface(int64_t* v) ABSL_ATTRIBUTE_NOINLINE;
bool SimpleIntInterfaceWithErrorMessage(int64_t* v, std::string* msg)
ABSL_ATTRIBUTE_NOINLINE;
absl::Status SimpleIntInterfaceWithErrorStatus(int64_t* v)
ABSL_ATTRIBUTE_NOINLINE;
absl::StatusOr<int64_t> SimpleIntStatusOrInterface() ABSL_ATTRIBUTE_NOINLINE;
// (a): Just a boolean return value with an out int64* parameter
bool SimpleIntInterface(int64_t* v) {
benchmark::DoNotOptimize(bm_cond);
if (bm_cond) {
*v = 42;
return true;
} else {
return false;
}
}
// (b): A boolean return value and a string error message filled in on failure
// and an out int64* parameter filled on success
bool SimpleIntInterfaceWithErrorMessage(int64_t* v, std::string* msg) {
benchmark::DoNotOptimize(bm_cond);
if (bm_cond) {
*v = 42;
return true;
} else {
*msg = "This is an error message";
return false;
}
}
// (c): A Status return value with an out int64* parameter on success
absl::Status SimpleIntInterfaceWithErrorStatus(int64_t* v) {
benchmark::DoNotOptimize(bm_cond);
if (bm_cond) {
*v = 42;
return absl::OkStatus();
} else {
return absl::UnknownError("This is an error message");
}
}
// (d): A StatusOr<int64> return value
absl::StatusOr<int64_t> SimpleIntStatusOrInterface() {
benchmark::DoNotOptimize(bm_cond);
if (bm_cond) {
return 42;
} else {
return absl::StatusOr<int64_t>(
absl::UnknownError("This is an error message"));
}
}
void SetCondition(benchmark::State& state) {
bm_cond = (state.range(0) == 0);
state.SetLabel(bm_cond ? "Success" : "Failure");
}
void BM_SimpleIntInterface(benchmark::State& state) {
SetCondition(state);
int64_t sum = 0;
for (auto s : state) {
int64_t v;
if (SimpleIntInterface(&v)) {
sum += v;
}
benchmark::DoNotOptimize(sum);
}
}
void BM_SimpleIntInterfaceMsg(benchmark::State& state) {
SetCondition(state);
int64_t sum = 0;
std::string msg;
for (auto s : state) {
int64_t v;
if (SimpleIntInterfaceWithErrorMessage(&v, &msg)) {
sum += v;
}
benchmark::DoNotOptimize(sum);
benchmark::DoNotOptimize(msg);
}
}
void BM_SimpleIntInterfaceStatus(benchmark::State& state) {
SetCondition(state);
int64_t sum = 0;
for (auto s : state) {
int64_t v;
auto result = SimpleIntInterfaceWithErrorStatus(&v);
if (result.ok()) {
sum += v;
}
benchmark::DoNotOptimize(sum);
}
}
void BM_SimpleIntStatusOrInterface(benchmark::State& state) {
SetCondition(state);
int64_t sum = 0;
for (auto s : state) {
auto v_s = SimpleIntStatusOrInterface();
if (v_s.ok()) {
sum += *v_s;
}
benchmark::DoNotOptimize(sum);
}
}
// Ordered like this so all the success path benchmarks (Arg(0)) show up,
// then all the failure benchmarks (Arg(1))
BENCHMARK(BM_SimpleIntInterface)->Arg(0);
BENCHMARK(BM_SimpleIntInterfaceMsg)->Arg(0);
BENCHMARK(BM_SimpleIntInterfaceStatus)->Arg(0);
BENCHMARK(BM_SimpleIntStatusOrInterface)->Arg(0);
BENCHMARK(BM_SimpleIntInterface)->Arg(1);
BENCHMARK(BM_SimpleIntInterfaceMsg)->Arg(1);
BENCHMARK(BM_SimpleIntInterfaceStatus)->Arg(1);
BENCHMARK(BM_SimpleIntStatusOrInterface)->Arg(1);
} // namespace