root/tools/perf/tests/workloads/thloop.c
/* SPDX-License-Identifier: GPL-2.0 */
#include <pthread.h>
#include <stdlib.h>
#include <signal.h>
#include <unistd.h>
#include <linux/compiler.h>
#include "../tests.h"

static volatile sig_atomic_t done;

/* We want to check this symbol in perf report */
noinline void test_loop(void);

static void sighandler(int sig __maybe_unused)
{
        done = 1;
}

noinline void test_loop(void)
{
        while (!done);
}

static void *thfunc(void *arg)
{
        void (*loop_fn)(void) = arg;

        loop_fn();
        return NULL;
}

static int thloop(int argc, const char **argv)
{
        int nt = 2, sec = 1, err = 1;
        pthread_t *thread_list = NULL;

        if (argc > 0)
                sec = atoi(argv[0]);

        if (sec <= 0) {
                fprintf(stderr, "Error: seconds (%d) must be >= 1\n", sec);
                return 1;
        }

        if (argc > 1)
                nt = atoi(argv[1]);

        if (nt <= 0) {
                fprintf(stderr, "Error: thread count (%d) must be >= 1\n", nt);
                return 1;
        }

        signal(SIGINT, sighandler);
        signal(SIGALRM, sighandler);

        thread_list = calloc(nt, sizeof(pthread_t));
        if (thread_list == NULL) {
                fprintf(stderr, "Error: malloc failed for %d threads\n", nt);
                goto out;
        }
        for (int i = 1; i < nt; i++) {
                int ret = pthread_create(&thread_list[i], NULL, thfunc, test_loop);

                if (ret) {
                        fprintf(stderr, "Error: failed to create thread %d\n", i);
                        done = 1; // Ensure started threads terminate.
                        goto out;
                }
        }
        alarm(sec);
        test_loop();
        err = 0;
out:
        for (int i = 1; i < nt; i++) {
                if (thread_list && thread_list[i])
                        pthread_join(thread_list[i], /*retval=*/NULL);
        }
        free(thread_list);
        return err;
}

DEFINE_WORKLOAD(thloop);