123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344 |
- /*
- Copyright (c) 2017-2019, Feral Interactive
- All rights reserved.
- Redistribution and use in source and binary forms, with or without
- modification, are permitted provided that the following conditions are met:
- * Redistributions of source code must retain the above copyright notice,
- this list of conditions and the following disclaimer.
- * Redistributions in binary form must reproduce the above copyright
- notice, this list of conditions and the following disclaimer in the
- documentation and/or other materials provided with the distribution.
- * Neither the name of Feral Interactive nor the names of its contributors
- may be used to endorse or promote products derived from this software
- without specific prior written permission.
- THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
- AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
- IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
- ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
- LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
- CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
- SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
- INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
- CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
- ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
- POSSIBILITY OF SUCH DAMAGE.
- */
- /**
- * Simple daemon to allow user space programs to control the CPU governors
- *
- * The main process is responsible for bootstrapping the D-BUS daemon, caching
- * the initial governor settings, and then responding to requests over D-BUS.
- *
- * Clients register their pid(s) with the service, which are routinely checked
- * to see if they've expired. Once we reach our first actively registered client
- * we put the system into "game mode", i.e. move the CPU governor into a performance
- * mode.
- *
- * Upon exit, or when all clients have stopped running, we put the system back
- * into the default governor policy, which is invariably powersave or similar
- * on laptops. This ensures that the system is obtaining the maximum performance
- * whilst gaming, and allowed to sanely return to idle once the workload is
- * complete.
- */
- #define _GNU_SOURCE
- #include "gamemode.h"
- #include "common-logging.h"
- #include "gamemode-config.h"
- #include "gamemode_client.h"
- #include "build-config.h"
- #include <fcntl.h>
- #include <getopt.h>
- #include <signal.h>
- #include <sys/stat.h>
- #include <systemd/sd-daemon.h> /* TODO: Move usage to gamemode-dbus.c */
- #include <unistd.h>
- #define USAGE_TEXT \
- "Usage: %s [-d] [-l] [-r] [-t] [-h] [-v]\n\n" \
- " -r[PID], --request=[PID] Toggle gamemode for process\n" \
- " When no PID given, requests gamemode and pauses\n" \
- " -s[PID], --status=[PID] Query the status of gamemode for process\n" \
- " When no PID given, queries the status globally\n" \
- " -d, --daemonize Daemonize self after launch\n" \
- " -l, --log-to-syslog Log to syslog\n" \
- " -t, --test Run tests\n" \
- " -h, --help Print this help\n" \
- " -v, --version Print version\n" \
- "\n" \
- "See man page for more information.\n"
- #define VERSION_TEXT "gamemode version: v" GAMEMODE_VERSION "\n"
- static void sigint_handler(__attribute__((unused)) int signo)
- {
- LOG_MSG("Quitting by request...\n");
- sd_notify(0, "STATUS=GameMode is quitting by request...\n");
- /* Clean up nicely */
- game_mode_context_destroy(game_mode_context_instance());
- _Exit(EXIT_SUCCESS);
- }
- static void sigint_handler_noexit(__attribute__((unused)) int signo)
- {
- LOG_MSG("Quitting by request...\n");
- }
- /**
- * Helper to perform standard UNIX daemonization
- */
- static void daemonize(const char *name)
- {
- /* Initial fork */
- pid_t pid = fork();
- if (pid < 0) {
- FATAL_ERRORNO("Failed to fork");
- }
- if (pid != 0) {
- LOG_MSG("Daemon launched as %s...\n", name);
- exit(EXIT_SUCCESS);
- }
- /* Fork a second time */
- pid = fork();
- if (pid < 0) {
- FATAL_ERRORNO("Failed to fork");
- } else if (pid > 0) {
- exit(EXIT_SUCCESS);
- }
- /* Now continue execution */
- umask(0022);
- if (setsid() < 0) {
- FATAL_ERRORNO("Failed to create process group\n");
- }
- if (chdir("/") < 0) {
- FATAL_ERRORNO("Failed to change to root directory\n");
- }
- /* replace standard file descriptors by /dev/null */
- int devnull_r = open("/dev/null", O_RDONLY);
- int devnull_w = open("/dev/null", O_WRONLY);
- if (devnull_r == -1 || devnull_w == -1)
- {
- LOG_ERROR("Failed to redirect standard input and output to /dev/null\n");
- }
- else
- {
- dup2(devnull_r, STDIN_FILENO);
- dup2(devnull_w, STDOUT_FILENO);
- dup2(devnull_w, STDERR_FILENO);
- close(devnull_r);
- close(devnull_w);
- }
- }
- /**
- * Main bootstrap entry into gamemoded
- */
- int main(int argc, char *argv[])
- {
- GameModeContext *context = NULL;
- /* Gather command line options */
- bool daemon = false;
- bool use_syslog = false;
- int opt = 0;
- /* Options struct for getopt_long */
- static struct option long_options[] = {
- { "daemonize", no_argument, 0, 'd' }, { "log-to-syslog", no_argument, 0, 'l' },
- { "request", optional_argument, 0, 'r' }, { "test", no_argument, 0, 't' },
- { "status", optional_argument, 0, 's' }, { "help", no_argument, 0, 'h' },
- { "version", no_argument, 0, 'v' }, { NULL, 0, NULL, 0 },
- };
- static const char *short_options = "dls::r::tvh";
- while ((opt = getopt_long(argc, argv, short_options, long_options, 0)) != -1) {
- switch (opt) {
- case 'd':
- daemon = true;
- break;
- case 'l':
- use_syslog = true;
- break;
- case 's':
- if (optarg != NULL) {
- pid_t pid = atoi(optarg);
- switch (gamemode_query_status_for(pid)) {
- case 0: /* inactive */
- LOG_MSG("gamemode is inactive\n");
- break;
- case 1: /* active not not registered */
- LOG_MSG("gamemode is active but [%d] not registered\n", pid);
- break;
- case 2: /* active for client */
- LOG_MSG("gamemode is active and [%d] registered\n", pid);
- break;
- case -1:
- LOG_ERROR("gamemode_query_status_for(%d) failed: %s\n",
- pid,
- gamemode_error_string());
- exit(EXIT_FAILURE);
- default:
- LOG_ERROR("gamemode_query_status returned unexpected value 2\n");
- exit(EXIT_FAILURE);
- }
- } else {
- int ret = 0;
- switch ((ret = gamemode_query_status())) {
- case 0: /* inactive */
- LOG_MSG("gamemode is inactive\n");
- break;
- case 1: /* active */
- LOG_MSG("gamemode is active\n");
- break;
- case -1: /* error */
- LOG_ERROR("gamemode status request failed: %s\n", gamemode_error_string());
- exit(EXIT_FAILURE);
- default: /* unexpected value eg. 2 */
- LOG_ERROR("gamemode_query_status returned unexpected value %d\n", ret);
- exit(EXIT_FAILURE);
- }
- }
- exit(EXIT_SUCCESS);
- case 'r':
- if (optarg != NULL) {
- pid_t pid = atoi(optarg);
- /* toggle gamemode for the process */
- switch (gamemode_query_status_for(pid)) {
- case 0: /* inactive */
- case 1: /* active not not registered */
- LOG_MSG("gamemode not active for client, requesting start for %d...\n", pid);
- if (gamemode_request_start_for(pid) < 0) {
- LOG_ERROR("gamemode_request_start_for(%d) failed: %s\n",
- pid,
- gamemode_error_string());
- exit(EXIT_FAILURE);
- }
- LOG_MSG("request succeeded\n");
- break;
- case 2: /* active for client */
- LOG_MSG("gamemode active for client, requesting end for %d...\n", pid);
- if (gamemode_request_end_for(pid) < 0) {
- LOG_ERROR("gamemode_request_end_for(%d) failed: %s\n",
- pid,
- gamemode_error_string());
- exit(EXIT_FAILURE);
- }
- LOG_MSG("request succeeded\n");
- break;
- case -1: /* error */
- LOG_ERROR("gamemode_query_status_for(%d) failed: %s\n",
- pid,
- gamemode_error_string());
- exit(EXIT_FAILURE);
- }
- } else {
- /* Request gamemode for this process */
- if (gamemode_request_start() < 0) {
- LOG_ERROR("gamemode request failed: %s\n", gamemode_error_string());
- exit(EXIT_FAILURE);
- }
- /* Request and report on the status */
- switch (gamemode_query_status()) {
- case 2: /* active for this client */
- LOG_MSG("gamemode request succeeded and is active\n");
- break;
- case 1: /* active */
- LOG_ERROR("gamemode request succeeded and is active but registration failed\n");
- exit(EXIT_FAILURE);
- case 0: /* inactive */
- LOG_ERROR("gamemode request succeeded but is not active\n");
- exit(EXIT_FAILURE);
- case -1: /* error */
- LOG_ERROR("gamemode_query_status failed: %s\n", gamemode_error_string());
- exit(EXIT_FAILURE);
- }
- /* Simply pause and wait a SIGINT */
- if (signal(SIGINT, sigint_handler_noexit) == SIG_ERR) {
- FATAL_ERRORNO("Could not catch SIGINT");
- }
- pause();
- /* Explicitly clean up */
- if (gamemode_request_end() < 0) {
- LOG_ERROR("gamemode request failed: %s\n", gamemode_error_string());
- exit(EXIT_FAILURE);
- }
- }
- exit(EXIT_SUCCESS);
- case 't': {
- int status = game_mode_run_client_tests();
- exit(status);
- }
- case 'v':
- LOG_MSG(VERSION_TEXT);
- exit(EXIT_SUCCESS);
- case 'h':
- LOG_MSG(USAGE_TEXT, argv[0]);
- exit(EXIT_SUCCESS);
- default:
- fprintf(stderr, USAGE_TEXT, argv[0]);
- exit(EXIT_FAILURE);
- }
- }
- /* If syslog is requested, set it up with our process name */
- if (use_syslog) {
- set_use_syslog(argv[0]);
- }
- /* Daemonize ourselves first if asked */
- if (daemon) {
- daemonize(argv[0]);
- }
- /* Log a version message on startup */
- LOG_MSG("v%s\n", GAMEMODE_VERSION);
- /* Set up the game mode context */
- context = game_mode_context_instance();
- game_mode_context_init(context);
- /* Handle quits cleanly */
- if (signal(SIGINT, sigint_handler) == SIG_ERR) {
- FATAL_ERRORNO("Could not catch SIGINT");
- }
- if (signal(SIGTERM, sigint_handler) == SIG_ERR) {
- FATAL_ERRORNO("Could not catch SIGTERM");
- }
- /* Run the main dbus message loop */
- game_mode_context_loop(context);
- game_mode_context_destroy(context);
- /* Log we're finished */
- LOG_MSG("Quitting naturally...\n");
- sd_notify(0, "STATUS=GameMode is quitting naturally...\n");
- }
|