123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209 |
- /*
- 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 "config.h"
- #include "daemonize.h"
- #include "dbus_messaging.h"
- #include "gamemode.h"
- #include "gamemode_client.h"
- #include "logging.h"
- #include <signal.h>
- #include <stdlib.h>
- #include <string.h>
- #include <systemd/sd-daemon.h>
- #include <unistd.h>
- #define USAGE_TEXT \
- "Usage: %s [-d] [-l] [-r] [-t] [-h] [-v]\n\n" \
- " -d daemonize self after launch\n" \
- " -l log to syslog\n" \
- " -r request gamemode and pause\n" \
- " -t run tests\n" \
- " -h print this help\n" \
- " -v 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");
- }
- /**
- * 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;
- int status;
- while ((opt = getopt(argc, argv, "dlsrtvh")) != -1) {
- switch (opt) {
- case 'd':
- daemon = true;
- break;
- case 'l':
- use_syslog = true;
- break;
- case 's': {
- if ((status = gamemode_query_status()) < 0) {
- LOG_ERROR("gamemode status request failed: %s\n", gamemode_error_string());
- exit(EXIT_FAILURE);
- } else if (status > 0) {
- LOG_MSG("gamemode is active\n");
- } else {
- LOG_MSG("gamemode is inactive\n");
- }
- exit(EXIT_SUCCESS);
- break;
- }
- case 'r':
- if (gamemode_request_start() < 0) {
- LOG_ERROR("gamemode request failed: %s\n", gamemode_error_string());
- exit(EXIT_FAILURE);
- }
- if ((status = gamemode_query_status()) == 2) {
- LOG_MSG("gamemode request succeeded and is active\n");
- } else if (status == 1) {
- LOG_ERROR("gamemode request succeeded and is active but registration failed\n");
- exit(EXIT_FAILURE);
- } else {
- LOG_ERROR("gamemode request succeeded but is not active\n");
- 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);
- break;
- case 't':
- status = game_mode_run_client_tests();
- exit(status);
- break;
- case 'v':
- LOG_MSG(VERSION_TEXT);
- exit(EXIT_SUCCESS);
- break;
- case 'h':
- LOG_MSG(USAGE_TEXT, argv[0]);
- exit(EXIT_SUCCESS);
- break;
- default:
- fprintf(stderr, USAGE_TEXT, argv[0]);
- exit(EXIT_FAILURE);
- break;
- }
- }
- /* 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");
- }
|