Path: blob/master/tools/power/x86/intel-speed-select/isst-daemon.c
26292 views
// SPDX-License-Identifier: GPL-2.01/*2* Intel Speed Select -- Allow speed select to daemonize3* Copyright (c) 2022 Intel Corporation.4*/56#include <stdio.h>7#include <stdlib.h>8#include <stdarg.h>9#include <string.h>10#include <unistd.h>11#include <fcntl.h>12#include <sys/file.h>13#include <sys/types.h>14#include <sys/stat.h>15#include <errno.h>16#include <getopt.h>17#include <signal.h>18#include <time.h>1920#include "isst.h"2122static int per_package_levels_info[MAX_PACKAGE_COUNT][MAX_DIE_PER_PACKAGE][MAX_PUNIT_PER_DIE];23static time_t per_package_levels_tm[MAX_PACKAGE_COUNT][MAX_DIE_PER_PACKAGE][MAX_PUNIT_PER_DIE];2425static void init_levels(void)26{27int i, j, k;2829for (i = 0; i < MAX_PACKAGE_COUNT; ++i)30for (j = 0; j < MAX_DIE_PER_PACKAGE; ++j)31for (k = 0; k < MAX_PUNIT_PER_DIE; ++k)32per_package_levels_info[i][j][k] = -1;33}3435void process_level_change(struct isst_id *id)36{37struct isst_pkg_ctdp_level_info ctdp_level;38struct isst_pkg_ctdp pkg_dev;39time_t tm;40int ret;4142if (id->pkg < 0 || id->die < 0 || id->punit < 0) {43debug_printf("Invalid package/die info for cpu:%d\n", id->cpu);44return;45}4647tm = time(NULL);48if (tm - per_package_levels_tm[id->pkg][id->die][id->punit] < 2)49return;5051per_package_levels_tm[id->pkg][id->die][id->punit] = tm;5253ret = isst_get_ctdp_levels(id, &pkg_dev);54if (ret) {55debug_printf("Can't get tdp levels for cpu:%d\n", id->cpu);56return;57}5859debug_printf("Get Config level %d pkg:%d die:%d current_level:%d\n", id->cpu,60id->pkg, id->die, pkg_dev.current_level);6162if (pkg_dev.locked) {63debug_printf("config TDP s locked \n");64return;65}6667if (per_package_levels_info[id->pkg][id->die][id->punit] == pkg_dev.current_level)68return;6970debug_printf("**Config level change for cpu:%d pkg:%d die:%d from %d to %d\n",71id->cpu, id->pkg, id->die, per_package_levels_info[id->pkg][id->die][id->punit],72pkg_dev.current_level);7374per_package_levels_info[id->pkg][id->die][id->punit] = pkg_dev.current_level;7576ctdp_level.core_cpumask_size =77alloc_cpu_set(&ctdp_level.core_cpumask);78ret = isst_get_coremask_info(id, pkg_dev.current_level, &ctdp_level);79if (ret) {80free_cpu_set(ctdp_level.core_cpumask);81debug_printf("Can't get core_mask:%d\n", id->cpu);82return;83}8485if (use_cgroupv2()) {86int ret;8788ret = enable_cpuset_controller();89if (ret)90goto use_offline;9192isolate_cpus(id, ctdp_level.core_cpumask_size, ctdp_level.core_cpumask,93pkg_dev.current_level, 0);9495goto free_mask;96}9798use_offline:99if (ctdp_level.cpu_count) {100int i, max_cpus = get_topo_max_cpus();101for (i = 0; i < max_cpus; ++i) {102if (!is_cpu_in_power_domain(i, id))103continue;104if (CPU_ISSET_S(i, ctdp_level.core_cpumask_size, ctdp_level.core_cpumask)) {105fprintf(stderr, "online cpu %d\n", i);106set_cpu_online_offline(i, 1);107} else {108fprintf(stderr, "offline cpu %d\n", i);109set_cpu_online_offline(i, 0);110}111}112}113free_mask:114free_cpu_set(ctdp_level.core_cpumask);115}116117static void _poll_for_config_change(struct isst_id *id, void *arg1, void *arg2,118void *arg3, void *arg4)119{120process_level_change(id);121}122123static void poll_for_config_change(void)124{125for_each_online_power_domain_in_set(_poll_for_config_change, NULL, NULL,126NULL, NULL);127}128129static int done = 0;130static int pid_file_handle;131132static void signal_handler(int sig)133{134switch (sig) {135case SIGINT:136case SIGTERM:137done = 1;138hfi_exit();139exit(0);140break;141default:142break;143}144}145146static void daemonize(char *rundir, char *pidfile)147{148int pid, sid, i;149char str[10];150struct sigaction sig_actions;151sigset_t sig_set;152int ret;153154if (getppid() == 1)155return;156157sigemptyset(&sig_set);158sigaddset(&sig_set, SIGCHLD);159sigaddset(&sig_set, SIGTSTP);160sigaddset(&sig_set, SIGTTOU);161sigaddset(&sig_set, SIGTTIN);162sigprocmask(SIG_BLOCK, &sig_set, NULL);163164sig_actions.sa_handler = signal_handler;165sigemptyset(&sig_actions.sa_mask);166sig_actions.sa_flags = 0;167168sigaction(SIGHUP, &sig_actions, NULL);169sigaction(SIGTERM, &sig_actions, NULL);170sigaction(SIGINT, &sig_actions, NULL);171172pid = fork();173if (pid < 0) {174/* Could not fork */175exit(EXIT_FAILURE);176}177if (pid > 0)178exit(EXIT_SUCCESS);179180umask(027);181182sid = setsid();183if (sid < 0)184exit(EXIT_FAILURE);185186/* close all descriptors */187for (i = getdtablesize(); i >= 0; --i)188close(i);189190i = open("/dev/null", O_RDWR);191if (i < 0)192exit(EXIT_FAILURE);193194ret = dup(i);195if (ret == -1)196exit(EXIT_FAILURE);197198ret = chdir(rundir);199if (ret == -1)200exit(EXIT_FAILURE);201202pid_file_handle = open(pidfile, O_RDWR | O_CREAT, 0600);203if (pid_file_handle == -1) {204/* Couldn't open lock file */205exit(1);206}207/* Try to lock file */208#ifdef LOCKF_SUPPORT209if (lockf(pid_file_handle, F_TLOCK, 0) == -1) {210#else211if (flock(pid_file_handle, LOCK_EX|LOCK_NB) < 0) {212#endif213/* Couldn't get lock on lock file */214fprintf(stderr, "Couldn't get lock file %d\n", getpid());215exit(1);216}217snprintf(str, sizeof(str), "%d\n", getpid());218ret = write(pid_file_handle, str, strlen(str));219if (ret == -1)220exit(EXIT_FAILURE);221222close(i);223}224225int isst_daemon(int debug_mode, int poll_interval, int no_daemon)226{227int ret;228229if (!no_daemon && poll_interval < 0 && !debug_mode) {230fprintf(stderr, "OOB mode is enabled and will run as daemon\n");231daemonize((char *) "/tmp/",232(char *)"/tmp/hfi-events.pid");233} else {234signal(SIGINT, signal_handler);235}236237init_levels();238239if (poll_interval < 0) {240ret = hfi_main();241if (ret) {242fprintf(stderr, "HFI initialization failed\n");243}244fprintf(stderr, "Must specify poll-interval\n");245return ret;246}247248debug_printf("Starting loop\n");249while (!done) {250sleep(poll_interval);251poll_for_config_change();252}253254return 0;255}256257258