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