root/tools/power/x86/intel-speed-select/isst-daemon.c
// SPDX-License-Identifier: GPL-2.0
/*
 * Intel Speed Select -- Allow speed select to daemonize
 * Copyright (c) 2022 Intel Corporation.
 */

#include <stdio.h>
#include <stdlib.h>
#include <stdarg.h>
#include <string.h>
#include <unistd.h>
#include <fcntl.h>
#include <sys/file.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <errno.h>
#include <getopt.h>
#include <signal.h>
#include <time.h>

#include "isst.h"

static int per_package_levels_info[MAX_PACKAGE_COUNT][MAX_DIE_PER_PACKAGE][MAX_PUNIT_PER_DIE];
static time_t per_package_levels_tm[MAX_PACKAGE_COUNT][MAX_DIE_PER_PACKAGE][MAX_PUNIT_PER_DIE];

static void init_levels(void)
{
        int i, j, k;

        for (i = 0; i < MAX_PACKAGE_COUNT; ++i)
                for (j = 0; j < MAX_DIE_PER_PACKAGE; ++j)
                        for (k = 0; k < MAX_PUNIT_PER_DIE; ++k)
                                per_package_levels_info[i][j][k] = -1;
}

void process_level_change(struct isst_id *id)
{
        struct isst_pkg_ctdp_level_info ctdp_level;
        struct isst_pkg_ctdp pkg_dev;
        time_t tm;
        int ret;

        if (id->pkg < 0 || id->die < 0 || id->punit < 0) {
                debug_printf("Invalid package/die info for cpu:%d\n", id->cpu);
                return;
        }

        tm = time(NULL);
        if (tm - per_package_levels_tm[id->pkg][id->die][id->punit] < 2)
                return;

        per_package_levels_tm[id->pkg][id->die][id->punit] = tm;

        ret = isst_get_ctdp_levels(id, &pkg_dev);
        if (ret) {
                debug_printf("Can't get tdp levels for cpu:%d\n", id->cpu);
                return;
        }

        debug_printf("Get Config level %d pkg:%d die:%d current_level:%d\n", id->cpu,
                      id->pkg, id->die, pkg_dev.current_level);

        if (pkg_dev.locked) {
                debug_printf("config TDP s locked \n");
                return;
        }

        if (per_package_levels_info[id->pkg][id->die][id->punit] == pkg_dev.current_level)
                return;

        debug_printf("**Config level change for cpu:%d pkg:%d die:%d from %d to %d\n",
                      id->cpu, id->pkg, id->die, per_package_levels_info[id->pkg][id->die][id->punit],
                      pkg_dev.current_level);

        per_package_levels_info[id->pkg][id->die][id->punit] = pkg_dev.current_level;

        ctdp_level.core_cpumask_size =
                alloc_cpu_set(&ctdp_level.core_cpumask);
        ret = isst_get_coremask_info(id, pkg_dev.current_level, &ctdp_level);
        if (ret) {
                free_cpu_set(ctdp_level.core_cpumask);
                debug_printf("Can't get core_mask:%d\n", id->cpu);
                return;
        }

        if (use_cgroupv2()) {
                int ret;

                ret = enable_cpuset_controller();
                if (ret)
                        goto use_offline;

                isolate_cpus(id, ctdp_level.core_cpumask_size, ctdp_level.core_cpumask,
                             pkg_dev.current_level, 0);

                goto free_mask;
        }

use_offline:
        if (ctdp_level.cpu_count) {
                int i, max_cpus = get_topo_max_cpus();
                for (i = 0; i < max_cpus; ++i) {
                        if (!is_cpu_in_power_domain(i, id))
                                continue;
                        if (CPU_ISSET_S(i, ctdp_level.core_cpumask_size, ctdp_level.core_cpumask)) {
                                fprintf(stderr, "online cpu %d\n", i);
                                set_cpu_online_offline(i, 1);
                        } else {
                                fprintf(stderr, "offline cpu %d\n", i);
                                set_cpu_online_offline(i, 0);
                        }
                }
        }
free_mask:
        free_cpu_set(ctdp_level.core_cpumask);
}

static void _poll_for_config_change(struct isst_id *id, void *arg1, void *arg2,
                                    void *arg3, void *arg4)
{
        process_level_change(id);
}

static void poll_for_config_change(void)
{
        for_each_online_power_domain_in_set(_poll_for_config_change, NULL, NULL,
                                       NULL, NULL);
}

static int done = 0;
static int pid_file_handle;

static void signal_handler(int sig)
{
        switch (sig) {
        case SIGINT:
        case SIGTERM:
                done = 1;
                hfi_exit();
                exit(0);
                break;
        default:
                break;
        }
}

static void daemonize(char *rundir, char *pidfile)
{
        int pid, sid, i;
        char str[10];
        struct sigaction sig_actions;
        sigset_t sig_set;
        int ret;

        if (getppid() == 1)
                return;

        sigemptyset(&sig_set);
        sigaddset(&sig_set, SIGCHLD);
        sigaddset(&sig_set, SIGTSTP);
        sigaddset(&sig_set, SIGTTOU);
        sigaddset(&sig_set, SIGTTIN);
        sigprocmask(SIG_BLOCK, &sig_set, NULL);

        sig_actions.sa_handler = signal_handler;
        sigemptyset(&sig_actions.sa_mask);
        sig_actions.sa_flags = 0;

        sigaction(SIGHUP, &sig_actions, NULL);
        sigaction(SIGTERM, &sig_actions, NULL);
        sigaction(SIGINT, &sig_actions, NULL);

        pid = fork();
        if (pid < 0) {
                /* Could not fork */
                exit(EXIT_FAILURE);
        }
        if (pid > 0)
                exit(EXIT_SUCCESS);

        umask(027);

        sid = setsid();
        if (sid < 0)
                exit(EXIT_FAILURE);

        /* close all descriptors */
        for (i = getdtablesize(); i >= 0; --i)
                close(i);

        i = open("/dev/null", O_RDWR);
        if (i < 0)
                exit(EXIT_FAILURE);

        ret = dup(i);
        if (ret == -1)
                exit(EXIT_FAILURE);

        ret = chdir(rundir);
        if (ret == -1)
                exit(EXIT_FAILURE);

        pid_file_handle = open(pidfile, O_RDWR | O_CREAT, 0600);
        if (pid_file_handle == -1) {
                /* Couldn't open lock file */
                exit(1);
        }
        /* Try to lock file */
#ifdef LOCKF_SUPPORT
        if (lockf(pid_file_handle, F_TLOCK, 0) == -1) {
#else
        if (flock(pid_file_handle, LOCK_EX|LOCK_NB) < 0) {
#endif
                /* Couldn't get lock on lock file */
                fprintf(stderr, "Couldn't get lock file %d\n", getpid());
                exit(1);
        }
        snprintf(str, sizeof(str), "%d\n", getpid());
        ret = write(pid_file_handle, str, strlen(str));
        if (ret == -1)
                exit(EXIT_FAILURE);

        close(i);
}

int isst_daemon(int debug_mode, int poll_interval, int no_daemon)
{
        int ret;

        if (!no_daemon && poll_interval < 0 && !debug_mode) {
                fprintf(stderr, "OOB mode is enabled and will run as daemon\n");
                daemonize((char *) "/tmp/",
                                (char *)"/tmp/hfi-events.pid");
        } else {
                signal(SIGINT, signal_handler);
        }

        init_levels();

        if (poll_interval < 0) {
                ret = hfi_main();
                if (ret) {
                        fprintf(stderr, "HFI initialization failed\n");
                }
                fprintf(stderr, "Must specify poll-interval\n");
                return ret;
        }

        debug_printf("Starting loop\n");
        while (!done) {
                sleep(poll_interval);
                poll_for_config_change();
        }

        return 0;
}