blob: c6eae0a7cb4da8e1e96798362d5a743f2bc6c1e3 [file] [log] [blame]
// Copyright 2017 The Bazel Authors. All rights reserved.
//
// 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 <sys/param.h>
#include <sys/time.h>
#include <sys/types.h>
#include <sys/resource.h>
#include <err.h>
#include <inttypes.h>
#include <time.h>
#include <unistd.h>
#include <cerrno>
#include <cstdint>
#include <cstdio>
#include <cstdlib>
#include <cstring>
// Computes the time that passed, in millis, since the previous timestamp.
static uint64_t ElapsedCpuMillisSince(const clock_t before) {
const clock_t now = clock();
return 1000 * (now - before) / CLOCKS_PER_SEC;
}
// Computes the time that passed, in millis, since the previous timestamp.
static uint64_t ElapsedWallTimeMillisSince(const struct timeval* before) {
struct timeval now;
gettimeofday(&now, NULL);
return (now.tv_sec * 1000 + now.tv_usec / 1000) -
(before->tv_sec * 1000 + before->tv_usec / 1000);
}
// Spends CPU time for about the requested number of milliseconds.
//
// This function should not invoke any system calls, but as this is very hard to
// do in a portable way, the number of such invocations should be kept to a
// minimum so that their cost is not noticeable.
//
// This function does not guarantee that the used CPU time is above the given
// millis. The caller needs to check this and, if not yet achieved, call this
// function again with the remainder.
static void WasteUserTime(const uint64_t millis) {
const clock_t before = clock();
while (ElapsedCpuMillisSince(before) < millis) {
// The body of this loop is supposed to consume enough CPU time to make the
// actual calls to clock() insignificant. This means that if this loop gets
// optimized, or if the CPU becomes fast enough to run this "too fast", this
// function may consume more system time than user time and cause tests to
// fail.
volatile uint64_t counter = 0;
while (counter < 1000000) {
counter++;
}
}
}
// Spends system time for about the requested number of milliseconds.
//
// This function does not guarantee that the used system time is above the given
// millis. The caller needs to check this and, if not yet achieved, call this
// function again with the remainder.
static void WasteSystemTime(const uint64_t millis) {
char current_dir_path[MAXPATHLEN];
if (getcwd(current_dir_path, sizeof(current_dir_path)) == NULL) {
err(EXIT_FAILURE, "getcwd() failed");
}
struct timeval before;
gettimeofday(&before, NULL);
while (ElapsedWallTimeMillisSince(&before) < millis) {
// Arbitrary syscall to waste system time.
if (chdir(current_dir_path) != 0) {
err(EXIT_FAILURE, "chdir() failed");
}
}
}
static void GetResourceUsage(struct rusage *rusage) {
if (getrusage(RUSAGE_SELF, rusage) != 0) {
err(EXIT_FAILURE, "getrusage() failed");
}
}
static uint64_t GetUsedUserTimeMillis() {
struct rusage my_rusage;
GetResourceUsage(&my_rusage);
return my_rusage.ru_utime.tv_sec * 1000 + my_rusage.ru_utime.tv_usec / 1000;
}
static uint64_t GetUsedSystemTimeMillis() {
struct rusage my_rusage;
GetResourceUsage(&my_rusage);
return my_rusage.ru_stime.tv_sec * 1000 + my_rusage.ru_stime.tv_usec / 1000;
}
// Subtracts subtrahend from minuend, or returns zero if the subtrahend is
// larger than the minuend.
static uint64_t SubtractOrZero(const uint64_t minuend,
const uint64_t subtrahend) {
if (subtrahend > minuend) {
return 0;
} else {
return minuend - subtrahend;
}
}
// This program just wastes (at least) the desired amount of CPU time, by
// checking its own resource usage (rusage) while running.
int main(int argc, char **argv) {
// Parse command-line arguments.
const char *progname = argv[0] ? argv[0] : "spend_cpu_time";
if (argc != 3) {
fprintf(stderr, "Usage: %s <user_time_seconds> <system_time_seconds>\n",
progname);
exit(EXIT_FAILURE);
}
int requested_user_time_seconds \
= atoi(argv[1]); // NOLINT(runtime/deprecated_fn)
int requested_system_time_seconds \
= atoi(argv[2]); // NOLINT(runtime/deprecated_fn)
// Waste system time first, because this also wastes some user time.
if (requested_system_time_seconds > 0) {
const uint64_t requested_millis = requested_system_time_seconds * 1000;
for (;;) {
const uint64_t remaining_millis =
SubtractOrZero(requested_millis, GetUsedSystemTimeMillis());
if (remaining_millis == 0) {
break;
}
WasteSystemTime(remaining_millis);
}
}
// Waste user time if we haven't already wasted enough.
if (requested_user_time_seconds > 0) {
const uint64_t requested_millis = requested_user_time_seconds * 1000;
for (;;) {
const uint64_t remaining_millis =
SubtractOrZero(requested_millis, GetUsedUserTimeMillis());
if (remaining_millis == 0) {
break;
}
WasteUserTime(remaining_millis);
}
}
printf("Total user time wasted: %" PRIu64 " ms\n",
GetUsedUserTimeMillis());
printf("Total system time wasted: %" PRIu64 " ms\n",
GetUsedSystemTimeMillis());
exit(EXIT_SUCCESS);
}