162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * Intel Speed Select -- Allow speed select to daemonize 462306a36Sopenharmony_ci * Copyright (c) 2022 Intel Corporation. 562306a36Sopenharmony_ci */ 662306a36Sopenharmony_ci 762306a36Sopenharmony_ci#include <stdio.h> 862306a36Sopenharmony_ci#include <stdlib.h> 962306a36Sopenharmony_ci#include <stdarg.h> 1062306a36Sopenharmony_ci#include <string.h> 1162306a36Sopenharmony_ci#include <unistd.h> 1262306a36Sopenharmony_ci#include <fcntl.h> 1362306a36Sopenharmony_ci#include <sys/file.h> 1462306a36Sopenharmony_ci#include <sys/types.h> 1562306a36Sopenharmony_ci#include <sys/stat.h> 1662306a36Sopenharmony_ci#include <errno.h> 1762306a36Sopenharmony_ci#include <getopt.h> 1862306a36Sopenharmony_ci#include <signal.h> 1962306a36Sopenharmony_ci#include <time.h> 2062306a36Sopenharmony_ci 2162306a36Sopenharmony_ci#include "isst.h" 2262306a36Sopenharmony_ci 2362306a36Sopenharmony_cistatic int per_package_levels_info[MAX_PACKAGE_COUNT][MAX_DIE_PER_PACKAGE][MAX_PUNIT_PER_DIE]; 2462306a36Sopenharmony_cistatic time_t per_package_levels_tm[MAX_PACKAGE_COUNT][MAX_DIE_PER_PACKAGE][MAX_PUNIT_PER_DIE]; 2562306a36Sopenharmony_ci 2662306a36Sopenharmony_cistatic void init_levels(void) 2762306a36Sopenharmony_ci{ 2862306a36Sopenharmony_ci int i, j, k; 2962306a36Sopenharmony_ci 3062306a36Sopenharmony_ci for (i = 0; i < MAX_PACKAGE_COUNT; ++i) 3162306a36Sopenharmony_ci for (j = 0; j < MAX_DIE_PER_PACKAGE; ++j) 3262306a36Sopenharmony_ci for (k = 0; k < MAX_PUNIT_PER_DIE; ++k) 3362306a36Sopenharmony_ci per_package_levels_info[i][j][k] = -1; 3462306a36Sopenharmony_ci} 3562306a36Sopenharmony_ci 3662306a36Sopenharmony_civoid process_level_change(struct isst_id *id) 3762306a36Sopenharmony_ci{ 3862306a36Sopenharmony_ci struct isst_pkg_ctdp_level_info ctdp_level; 3962306a36Sopenharmony_ci struct isst_pkg_ctdp pkg_dev; 4062306a36Sopenharmony_ci time_t tm; 4162306a36Sopenharmony_ci int ret; 4262306a36Sopenharmony_ci 4362306a36Sopenharmony_ci if (id->pkg < 0 || id->die < 0 || id->punit < 0) { 4462306a36Sopenharmony_ci debug_printf("Invalid package/die info for cpu:%d\n", id->cpu); 4562306a36Sopenharmony_ci return; 4662306a36Sopenharmony_ci } 4762306a36Sopenharmony_ci 4862306a36Sopenharmony_ci tm = time(NULL); 4962306a36Sopenharmony_ci if (tm - per_package_levels_tm[id->pkg][id->die][id->punit] < 2) 5062306a36Sopenharmony_ci return; 5162306a36Sopenharmony_ci 5262306a36Sopenharmony_ci per_package_levels_tm[id->pkg][id->die][id->punit] = tm; 5362306a36Sopenharmony_ci 5462306a36Sopenharmony_ci ret = isst_get_ctdp_levels(id, &pkg_dev); 5562306a36Sopenharmony_ci if (ret) { 5662306a36Sopenharmony_ci debug_printf("Can't get tdp levels for cpu:%d\n", id->cpu); 5762306a36Sopenharmony_ci return; 5862306a36Sopenharmony_ci } 5962306a36Sopenharmony_ci 6062306a36Sopenharmony_ci debug_printf("Get Config level %d pkg:%d die:%d current_level:%d\n", id->cpu, 6162306a36Sopenharmony_ci id->pkg, id->die, pkg_dev.current_level); 6262306a36Sopenharmony_ci 6362306a36Sopenharmony_ci if (pkg_dev.locked) { 6462306a36Sopenharmony_ci debug_printf("config TDP s locked \n"); 6562306a36Sopenharmony_ci return; 6662306a36Sopenharmony_ci } 6762306a36Sopenharmony_ci 6862306a36Sopenharmony_ci if (per_package_levels_info[id->pkg][id->die][id->punit] == pkg_dev.current_level) 6962306a36Sopenharmony_ci return; 7062306a36Sopenharmony_ci 7162306a36Sopenharmony_ci debug_printf("**Config level change for cpu:%d pkg:%d die:%d from %d to %d\n", 7262306a36Sopenharmony_ci id->cpu, id->pkg, id->die, per_package_levels_info[id->pkg][id->die][id->punit], 7362306a36Sopenharmony_ci pkg_dev.current_level); 7462306a36Sopenharmony_ci 7562306a36Sopenharmony_ci per_package_levels_info[id->pkg][id->die][id->punit] = pkg_dev.current_level; 7662306a36Sopenharmony_ci 7762306a36Sopenharmony_ci ctdp_level.core_cpumask_size = 7862306a36Sopenharmony_ci alloc_cpu_set(&ctdp_level.core_cpumask); 7962306a36Sopenharmony_ci ret = isst_get_coremask_info(id, pkg_dev.current_level, &ctdp_level); 8062306a36Sopenharmony_ci if (ret) { 8162306a36Sopenharmony_ci free_cpu_set(ctdp_level.core_cpumask); 8262306a36Sopenharmony_ci debug_printf("Can't get core_mask:%d\n", id->cpu); 8362306a36Sopenharmony_ci return; 8462306a36Sopenharmony_ci } 8562306a36Sopenharmony_ci 8662306a36Sopenharmony_ci if (use_cgroupv2()) { 8762306a36Sopenharmony_ci int ret; 8862306a36Sopenharmony_ci 8962306a36Sopenharmony_ci ret = enable_cpuset_controller(); 9062306a36Sopenharmony_ci if (ret) 9162306a36Sopenharmony_ci goto use_offline; 9262306a36Sopenharmony_ci 9362306a36Sopenharmony_ci isolate_cpus(id, ctdp_level.core_cpumask_size, ctdp_level.core_cpumask, pkg_dev.current_level); 9462306a36Sopenharmony_ci 9562306a36Sopenharmony_ci goto free_mask; 9662306a36Sopenharmony_ci } 9762306a36Sopenharmony_ci 9862306a36Sopenharmony_ciuse_offline: 9962306a36Sopenharmony_ci if (ctdp_level.cpu_count) { 10062306a36Sopenharmony_ci int i, max_cpus = get_topo_max_cpus(); 10162306a36Sopenharmony_ci for (i = 0; i < max_cpus; ++i) { 10262306a36Sopenharmony_ci if (!is_cpu_in_power_domain(i, id)) 10362306a36Sopenharmony_ci continue; 10462306a36Sopenharmony_ci if (CPU_ISSET_S(i, ctdp_level.core_cpumask_size, ctdp_level.core_cpumask)) { 10562306a36Sopenharmony_ci fprintf(stderr, "online cpu %d\n", i); 10662306a36Sopenharmony_ci set_cpu_online_offline(i, 1); 10762306a36Sopenharmony_ci } else { 10862306a36Sopenharmony_ci fprintf(stderr, "offline cpu %d\n", i); 10962306a36Sopenharmony_ci set_cpu_online_offline(i, 0); 11062306a36Sopenharmony_ci } 11162306a36Sopenharmony_ci } 11262306a36Sopenharmony_ci } 11362306a36Sopenharmony_cifree_mask: 11462306a36Sopenharmony_ci free_cpu_set(ctdp_level.core_cpumask); 11562306a36Sopenharmony_ci} 11662306a36Sopenharmony_ci 11762306a36Sopenharmony_cistatic void _poll_for_config_change(struct isst_id *id, void *arg1, void *arg2, 11862306a36Sopenharmony_ci void *arg3, void *arg4) 11962306a36Sopenharmony_ci{ 12062306a36Sopenharmony_ci process_level_change(id); 12162306a36Sopenharmony_ci} 12262306a36Sopenharmony_ci 12362306a36Sopenharmony_cistatic void poll_for_config_change(void) 12462306a36Sopenharmony_ci{ 12562306a36Sopenharmony_ci for_each_online_power_domain_in_set(_poll_for_config_change, NULL, NULL, 12662306a36Sopenharmony_ci NULL, NULL); 12762306a36Sopenharmony_ci} 12862306a36Sopenharmony_ci 12962306a36Sopenharmony_cistatic int done = 0; 13062306a36Sopenharmony_cistatic int pid_file_handle; 13162306a36Sopenharmony_ci 13262306a36Sopenharmony_cistatic void signal_handler(int sig) 13362306a36Sopenharmony_ci{ 13462306a36Sopenharmony_ci switch (sig) { 13562306a36Sopenharmony_ci case SIGINT: 13662306a36Sopenharmony_ci case SIGTERM: 13762306a36Sopenharmony_ci done = 1; 13862306a36Sopenharmony_ci hfi_exit(); 13962306a36Sopenharmony_ci exit(0); 14062306a36Sopenharmony_ci break; 14162306a36Sopenharmony_ci default: 14262306a36Sopenharmony_ci break; 14362306a36Sopenharmony_ci } 14462306a36Sopenharmony_ci} 14562306a36Sopenharmony_ci 14662306a36Sopenharmony_cistatic void daemonize(char *rundir, char *pidfile) 14762306a36Sopenharmony_ci{ 14862306a36Sopenharmony_ci int pid, sid, i; 14962306a36Sopenharmony_ci char str[10]; 15062306a36Sopenharmony_ci struct sigaction sig_actions; 15162306a36Sopenharmony_ci sigset_t sig_set; 15262306a36Sopenharmony_ci int ret; 15362306a36Sopenharmony_ci 15462306a36Sopenharmony_ci if (getppid() == 1) 15562306a36Sopenharmony_ci return; 15662306a36Sopenharmony_ci 15762306a36Sopenharmony_ci sigemptyset(&sig_set); 15862306a36Sopenharmony_ci sigaddset(&sig_set, SIGCHLD); 15962306a36Sopenharmony_ci sigaddset(&sig_set, SIGTSTP); 16062306a36Sopenharmony_ci sigaddset(&sig_set, SIGTTOU); 16162306a36Sopenharmony_ci sigaddset(&sig_set, SIGTTIN); 16262306a36Sopenharmony_ci sigprocmask(SIG_BLOCK, &sig_set, NULL); 16362306a36Sopenharmony_ci 16462306a36Sopenharmony_ci sig_actions.sa_handler = signal_handler; 16562306a36Sopenharmony_ci sigemptyset(&sig_actions.sa_mask); 16662306a36Sopenharmony_ci sig_actions.sa_flags = 0; 16762306a36Sopenharmony_ci 16862306a36Sopenharmony_ci sigaction(SIGHUP, &sig_actions, NULL); 16962306a36Sopenharmony_ci sigaction(SIGTERM, &sig_actions, NULL); 17062306a36Sopenharmony_ci sigaction(SIGINT, &sig_actions, NULL); 17162306a36Sopenharmony_ci 17262306a36Sopenharmony_ci pid = fork(); 17362306a36Sopenharmony_ci if (pid < 0) { 17462306a36Sopenharmony_ci /* Could not fork */ 17562306a36Sopenharmony_ci exit(EXIT_FAILURE); 17662306a36Sopenharmony_ci } 17762306a36Sopenharmony_ci if (pid > 0) 17862306a36Sopenharmony_ci exit(EXIT_SUCCESS); 17962306a36Sopenharmony_ci 18062306a36Sopenharmony_ci umask(027); 18162306a36Sopenharmony_ci 18262306a36Sopenharmony_ci sid = setsid(); 18362306a36Sopenharmony_ci if (sid < 0) 18462306a36Sopenharmony_ci exit(EXIT_FAILURE); 18562306a36Sopenharmony_ci 18662306a36Sopenharmony_ci /* close all descriptors */ 18762306a36Sopenharmony_ci for (i = getdtablesize(); i >= 0; --i) 18862306a36Sopenharmony_ci close(i); 18962306a36Sopenharmony_ci 19062306a36Sopenharmony_ci i = open("/dev/null", O_RDWR); 19162306a36Sopenharmony_ci if (i < 0) 19262306a36Sopenharmony_ci exit(EXIT_FAILURE); 19362306a36Sopenharmony_ci 19462306a36Sopenharmony_ci ret = dup(i); 19562306a36Sopenharmony_ci if (ret == -1) 19662306a36Sopenharmony_ci exit(EXIT_FAILURE); 19762306a36Sopenharmony_ci 19862306a36Sopenharmony_ci ret = chdir(rundir); 19962306a36Sopenharmony_ci if (ret == -1) 20062306a36Sopenharmony_ci exit(EXIT_FAILURE); 20162306a36Sopenharmony_ci 20262306a36Sopenharmony_ci pid_file_handle = open(pidfile, O_RDWR | O_CREAT, 0600); 20362306a36Sopenharmony_ci if (pid_file_handle == -1) { 20462306a36Sopenharmony_ci /* Couldn't open lock file */ 20562306a36Sopenharmony_ci exit(1); 20662306a36Sopenharmony_ci } 20762306a36Sopenharmony_ci /* Try to lock file */ 20862306a36Sopenharmony_ci#ifdef LOCKF_SUPPORT 20962306a36Sopenharmony_ci if (lockf(pid_file_handle, F_TLOCK, 0) == -1) { 21062306a36Sopenharmony_ci#else 21162306a36Sopenharmony_ci if (flock(pid_file_handle, LOCK_EX|LOCK_NB) < 0) { 21262306a36Sopenharmony_ci#endif 21362306a36Sopenharmony_ci /* Couldn't get lock on lock file */ 21462306a36Sopenharmony_ci fprintf(stderr, "Couldn't get lock file %d\n", getpid()); 21562306a36Sopenharmony_ci exit(1); 21662306a36Sopenharmony_ci } 21762306a36Sopenharmony_ci snprintf(str, sizeof(str), "%d\n", getpid()); 21862306a36Sopenharmony_ci ret = write(pid_file_handle, str, strlen(str)); 21962306a36Sopenharmony_ci if (ret == -1) 22062306a36Sopenharmony_ci exit(EXIT_FAILURE); 22162306a36Sopenharmony_ci 22262306a36Sopenharmony_ci close(i); 22362306a36Sopenharmony_ci} 22462306a36Sopenharmony_ci 22562306a36Sopenharmony_ciint isst_daemon(int debug_mode, int poll_interval, int no_daemon) 22662306a36Sopenharmony_ci{ 22762306a36Sopenharmony_ci int ret; 22862306a36Sopenharmony_ci 22962306a36Sopenharmony_ci if (!no_daemon && poll_interval < 0 && !debug_mode) { 23062306a36Sopenharmony_ci fprintf(stderr, "OOB mode is enabled and will run as daemon\n"); 23162306a36Sopenharmony_ci daemonize((char *) "/tmp/", 23262306a36Sopenharmony_ci (char *)"/tmp/hfi-events.pid"); 23362306a36Sopenharmony_ci } else { 23462306a36Sopenharmony_ci signal(SIGINT, signal_handler); 23562306a36Sopenharmony_ci } 23662306a36Sopenharmony_ci 23762306a36Sopenharmony_ci init_levels(); 23862306a36Sopenharmony_ci 23962306a36Sopenharmony_ci if (poll_interval < 0) { 24062306a36Sopenharmony_ci ret = hfi_main(); 24162306a36Sopenharmony_ci if (ret) { 24262306a36Sopenharmony_ci fprintf(stderr, "HFI initialization failed\n"); 24362306a36Sopenharmony_ci } 24462306a36Sopenharmony_ci fprintf(stderr, "Must specify poll-interval\n"); 24562306a36Sopenharmony_ci return ret; 24662306a36Sopenharmony_ci } 24762306a36Sopenharmony_ci 24862306a36Sopenharmony_ci debug_printf("Starting loop\n"); 24962306a36Sopenharmony_ci while (!done) { 25062306a36Sopenharmony_ci sleep(poll_interval); 25162306a36Sopenharmony_ci poll_for_config_change(); 25262306a36Sopenharmony_ci } 25362306a36Sopenharmony_ci 25462306a36Sopenharmony_ci return 0; 25562306a36Sopenharmony_ci} 256