| /* |
| * Copyright (C) 2019 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 <inttypes.h> |
| #include <stdint.h> |
| #include <unistd.h> |
| |
| #include <thread> |
| |
| #include "perfetto/base/logging.h" |
| #include "perfetto/base/time.h" |
| #include "perfetto/ext/base/file_utils.h" |
| #include "perfetto/ext/base/getopt.h" |
| #include "perfetto/ext/base/scoped_file.h" |
| |
| #define PERFETTO_HAVE_PTHREADS \ |
| (PERFETTO_BUILDFLAG(PERFETTO_OS_LINUX) || \ |
| PERFETTO_BUILDFLAG(PERFETTO_OS_ANDROID) || \ |
| PERFETTO_BUILDFLAG(PERFETTO_OS_APPLE)) |
| |
| #if PERFETTO_HAVE_PTHREADS |
| #include <pthread.h> |
| #endif |
| |
| // Spawns the requested number threads that alternate between busy-waiting and |
| // sleeping. |
| |
| namespace perfetto { |
| namespace { |
| |
| void SetRandomThreadName(uint32_t thread_name_count) { |
| #if PERFETTO_HAVE_PTHREADS |
| char name[16] = {}; |
| snprintf(name, sizeof(name), "busy-%" PRIu32, |
| static_cast<uint32_t>(rand()) % thread_name_count); |
| pthread_setname_np(pthread_self(), name); |
| #endif |
| } |
| |
| void PrintUsage(const char* bin_name) { |
| #if PERFETTO_HAVE_PTHREADS |
| PERFETTO_ELOG( |
| "Usage: %s [--background] --threads=N --period_us=N --duty_cycle=[1-100] " |
| "[--thread_names=N]", |
| bin_name); |
| #else |
| PERFETTO_ELOG( |
| "Usage: %s [--background] --threads=N --period_us=N --duty_cycle=[1-100]", |
| bin_name); |
| #endif |
| } |
| |
| __attribute__((noreturn)) void BusyWait(int64_t tstart, |
| int64_t period_us, |
| int64_t busy_us, |
| uint32_t thread_name_count) { |
| int64_t tbusy = tstart; |
| int64_t tnext = tstart; |
| for (;;) { |
| if (thread_name_count) |
| SetRandomThreadName(thread_name_count); |
| |
| tbusy = tnext + busy_us * 1000; |
| tnext += period_us * 1000; |
| while (base::GetWallTimeNs().count() < tbusy) { |
| for (int i = 0; i < 10000; i++) { |
| asm volatile("" ::: "memory"); |
| } |
| } |
| auto tnow = base::GetWallTimeNs().count(); |
| if (tnow >= tnext) { |
| std::this_thread::yield(); |
| continue; |
| } |
| |
| while (tnow < tnext) { |
| // +1 to prevent sleeping twice when there is truncation. |
| base::SleepMicroseconds(static_cast<uint32_t>((tnext - tnow) / 1000) + 1); |
| tnow = base::GetWallTimeNs().count(); |
| } |
| } |
| } |
| |
| int BusyThreadsMain(int argc, char** argv) { |
| bool background = false; |
| int64_t num_threads = -1; |
| int64_t period_us = -1; |
| int64_t duty_cycle = -1; |
| uint32_t thread_name_count = 0; |
| |
| static option long_options[] = { |
| {"background", no_argument, nullptr, 'd'}, |
| {"threads", required_argument, nullptr, 't'}, |
| {"period_us", required_argument, nullptr, 'p'}, |
| {"duty_cycle", required_argument, nullptr, 'c'}, |
| #if PERFETTO_HAVE_PTHREADS |
| {"thread_names", required_argument, nullptr, 'r'}, |
| #endif |
| {nullptr, 0, nullptr, 0} |
| }; |
| int c; |
| while ((c = getopt_long(argc, argv, "", long_options, nullptr)) != -1) { |
| switch (c) { |
| case 'd': |
| background = true; |
| break; |
| case 't': |
| num_threads = atol(optarg); |
| break; |
| case 'p': |
| period_us = atol(optarg); |
| break; |
| case 'c': |
| duty_cycle = atol(optarg); |
| break; |
| #if PERFETTO_HAVE_PTHREADS |
| case 'r': |
| thread_name_count = static_cast<uint32_t>(atoi(optarg)); |
| break; |
| #endif |
| default: |
| break; |
| } |
| } |
| if (num_threads < 1 || period_us < 0 || duty_cycle < 1 || duty_cycle > 100 || |
| thread_name_count > (1 << 20)) { |
| PrintUsage(argv[0]); |
| return 1; |
| } |
| |
| if (background) { |
| pid_t pid; |
| switch (pid = fork()) { |
| case -1: |
| PERFETTO_FATAL("fork"); |
| case 0: { |
| PERFETTO_CHECK(setsid() != -1); |
| base::ignore_result(chdir("/")); |
| base::ScopedFile null = base::OpenFile("/dev/null", O_RDONLY); |
| PERFETTO_CHECK(null); |
| PERFETTO_CHECK(dup2(*null, STDIN_FILENO) != -1); |
| PERFETTO_CHECK(dup2(*null, STDOUT_FILENO) != -1); |
| PERFETTO_CHECK(dup2(*null, STDERR_FILENO) != -1); |
| // Do not accidentally close stdin/stdout/stderr. |
| if (*null <= 2) |
| null.release(); |
| break; |
| } |
| default: |
| printf("%d\n", pid); |
| exit(0); |
| } |
| } |
| |
| int64_t busy_us = |
| static_cast<int64_t>(static_cast<double>(period_us) * |
| (static_cast<double>(duty_cycle) / 100.0)); |
| |
| PERFETTO_LOG("Spawning %" PRId64 " threads; period duration: %" PRId64 |
| "us; busy duration: %" PRId64 "us.", |
| num_threads, period_us, busy_us); |
| |
| int64_t tstart = base::GetWallTimeNs().count(); |
| for (int i = 0; i < num_threads; i++) { |
| std::thread th(BusyWait, tstart, period_us, busy_us, thread_name_count); |
| th.detach(); |
| } |
| PERFETTO_LOG("Threads spawned, Ctrl-C to stop."); |
| while (sleep(600)) |
| ; |
| |
| return 0; |
| } |
| |
| } // namespace |
| } // namespace perfetto |
| |
| int main(int argc, char** argv) { |
| return perfetto::BusyThreadsMain(argc, argv); |
| } |