| // 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); |
| |
| 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; |
| } |