/*
* Copyright 2017, 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.
*/
#include <err.h>
#include <errno.h>
#include <sched.h>
#include <string.h>
#include <sys/wait.h>
#include <unistd.h>
#include <chrono>
#include <thread>
#include <benchmark/benchmark.h>
#include <debuggerd/client.h>
using namespace std::chrono_literals;
static_assert(std::chrono::high_resolution_clock::is_steady);
enum class ThreadState { Starting, Started, Stopping };
static void SetScheduler() {
struct sched_param param {
.sched_priority = 1,
};
if (sched_setscheduler(getpid(), SCHED_FIFO, ¶m) != 0) {
fprintf(stderr, "failed to set scheduler to SCHED_FIFO: %s", strerror(errno));
}
}
static std::chrono::duration<double> GetMaximumPause(std::atomic<ThreadState>& state) {
std::chrono::duration<double> max_diff(0);
const auto begin = std::chrono::high_resolution_clock::now();
auto last = begin;
state.store(ThreadState::Started);
while (state.load() != ThreadState::Stopping) {
auto now = std::chrono::high_resolution_clock::now();
auto diff = now - last;
if (diff > max_diff) {
max_diff = diff;
}
last = now;
}
return max_diff;
}
static void PerformDump() {
pid_t target = getpid();
pid_t forkpid = fork();
if (forkpid == -1) {
err(1, "fork failed");
} else if (forkpid != 0) {
int status;
pid_t pid = waitpid(forkpid, &status, 0);
if (pid == -1) {
err(1, "waitpid failed");
} else if (!WIFEXITED(status)) {
err(1, "child didn't exit");
} else if (WEXITSTATUS(status) != 0) {
errx(1, "child exited with non-zero status %d", WEXITSTATUS(status));
}
} else {
android::base::unique_fd output_fd(open("/dev/null", O_WRONLY | O_CLOEXEC));
if (output_fd == -1) {
err(1, "failed to open /dev/null");
}
if (!debuggerd_trigger_dump(target, kDebuggerdNativeBacktrace, 1000, std::move(output_fd))) {
errx(1, "failed to trigger dump");
}
_exit(0);
}
}
template <typename Fn>
static void BM_maximum_pause_impl(benchmark::State& state, const Fn& function) {
SetScheduler();
for (auto _ : state) {
std::chrono::duration<double> max_pause;
std::atomic<ThreadState> thread_state(ThreadState::Starting);
auto thread = std::thread([&]() { max_pause = GetMaximumPause(thread_state); });
while (thread_state != ThreadState::Started) {
std::this_thread::sleep_for(1ms);
}
function();
thread_state = ThreadState::Stopping;
thread.join();
state.SetIterationTime(max_pause.count());
}
}
static void BM_maximum_pause_noop(benchmark::State& state) {
BM_maximum_pause_impl(state, []() {});
}
static void BM_maximum_pause_debuggerd(benchmark::State& state) {
BM_maximum_pause_impl(state, []() { PerformDump(); });
}
BENCHMARK(BM_maximum_pause_noop)->Iterations(128)->UseManualTime();
BENCHMARK(BM_maximum_pause_debuggerd)->Iterations(128)->UseManualTime();
BENCHMARK_MAIN();