153a5a1b3Sopenharmony_ci/*** 253a5a1b3Sopenharmony_ci This file is part of PulseAudio. 353a5a1b3Sopenharmony_ci 453a5a1b3Sopenharmony_ci Copyright 2004-2006 Lennart Poettering 553a5a1b3Sopenharmony_ci 653a5a1b3Sopenharmony_ci PulseAudio is free software; you can redistribute it and/or modify 753a5a1b3Sopenharmony_ci it under the terms of the GNU Lesser General Public License as 853a5a1b3Sopenharmony_ci published by the Free Software Foundation; either version 2.1 of the 953a5a1b3Sopenharmony_ci License, or (at your option) any later version. 1053a5a1b3Sopenharmony_ci 1153a5a1b3Sopenharmony_ci PulseAudio is distributed in the hope that it will be useful, but 1253a5a1b3Sopenharmony_ci WITHOUT ANY WARRANTY; without even the implied warranty of 1353a5a1b3Sopenharmony_ci MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 1453a5a1b3Sopenharmony_ci General Public License for more details. 1553a5a1b3Sopenharmony_ci 1653a5a1b3Sopenharmony_ci You should have received a copy of the GNU Lesser General Public 1753a5a1b3Sopenharmony_ci License along with PulseAudio; if not, see <http://www.gnu.org/licenses/>. 1853a5a1b3Sopenharmony_ci***/ 1953a5a1b3Sopenharmony_ci 2053a5a1b3Sopenharmony_ci#ifdef HAVE_CONFIG_H 2153a5a1b3Sopenharmony_ci#include <config.h> 2253a5a1b3Sopenharmony_ci#endif 2353a5a1b3Sopenharmony_ci 2453a5a1b3Sopenharmony_ci#include <pulse/rtclock.h> 2553a5a1b3Sopenharmony_ci#include <pulse/timeval.h> 2653a5a1b3Sopenharmony_ci 2753a5a1b3Sopenharmony_ci#include <pulsecore/core-util.h> 2853a5a1b3Sopenharmony_ci#include <pulsecore/core-error.h> 2953a5a1b3Sopenharmony_ci#include <pulsecore/log.h> 3053a5a1b3Sopenharmony_ci#include <pulsecore/macro.h> 3153a5a1b3Sopenharmony_ci 3253a5a1b3Sopenharmony_ci#include "cpulimit.h" 3353a5a1b3Sopenharmony_ci 3453a5a1b3Sopenharmony_ci#ifdef HAVE_SIGXCPU 3553a5a1b3Sopenharmony_ci 3653a5a1b3Sopenharmony_ci#include <errno.h> 3753a5a1b3Sopenharmony_ci#include <stdio.h> 3853a5a1b3Sopenharmony_ci#include <string.h> 3953a5a1b3Sopenharmony_ci#include <unistd.h> 4053a5a1b3Sopenharmony_ci#include <signal.h> 4153a5a1b3Sopenharmony_ci 4253a5a1b3Sopenharmony_ci#ifdef HAVE_SYS_RESOURCE_H 4353a5a1b3Sopenharmony_ci#include <sys/resource.h> 4453a5a1b3Sopenharmony_ci#endif 4553a5a1b3Sopenharmony_ci 4653a5a1b3Sopenharmony_ci/* This module implements a watchdog that makes sure that the current 4753a5a1b3Sopenharmony_ci * process doesn't consume more than 70% CPU time for 10 seconds. This 4853a5a1b3Sopenharmony_ci * is very useful when using SCHED_FIFO scheduling which effectively 4953a5a1b3Sopenharmony_ci * disables multitasking. */ 5053a5a1b3Sopenharmony_ci 5153a5a1b3Sopenharmony_ci/* Method of operation: Using SIGXCPU a signal handler is called every 5253a5a1b3Sopenharmony_ci * 10s process CPU time. That function checks if less than 14s system 5353a5a1b3Sopenharmony_ci * time have passed. In that case, it tries to contact the main event 5453a5a1b3Sopenharmony_ci * loop through a pipe. After two additional seconds it is checked 5553a5a1b3Sopenharmony_ci * whether the main event loop contact was successful. If not, the 5653a5a1b3Sopenharmony_ci * program is terminated forcibly. */ 5753a5a1b3Sopenharmony_ci 5853a5a1b3Sopenharmony_ci/* Utilize this much CPU time at maximum */ 5953a5a1b3Sopenharmony_ci#define CPUTIME_PERCENT 70 6053a5a1b3Sopenharmony_ci 6153a5a1b3Sopenharmony_ci/* Check every 10s */ 6253a5a1b3Sopenharmony_ci#define CPUTIME_INTERVAL_SOFT (10) 6353a5a1b3Sopenharmony_ci 6453a5a1b3Sopenharmony_ci/* Recheck after 5s */ 6553a5a1b3Sopenharmony_ci#define CPUTIME_INTERVAL_HARD (5) 6653a5a1b3Sopenharmony_ci 6753a5a1b3Sopenharmony_ci/* Time of the last CPU load check */ 6853a5a1b3Sopenharmony_cistatic pa_usec_t last_time = 0; 6953a5a1b3Sopenharmony_ci 7053a5a1b3Sopenharmony_ci/* Pipe for communicating with the main loop */ 7153a5a1b3Sopenharmony_cistatic int the_pipe[2] = {-1, -1}; 7253a5a1b3Sopenharmony_ci 7353a5a1b3Sopenharmony_ci/* Main event loop and IO event for the FIFO */ 7453a5a1b3Sopenharmony_cistatic pa_mainloop_api *api = NULL; 7553a5a1b3Sopenharmony_cistatic pa_io_event *io_event = NULL; 7653a5a1b3Sopenharmony_ci 7753a5a1b3Sopenharmony_ci/* Saved sigaction struct for SIGXCPU */ 7853a5a1b3Sopenharmony_cistatic struct sigaction sigaction_prev; 7953a5a1b3Sopenharmony_ci 8053a5a1b3Sopenharmony_ci/* Nonzero after pa_cpu_limit_init() */ 8153a5a1b3Sopenharmony_cistatic bool installed = false; 8253a5a1b3Sopenharmony_ci 8353a5a1b3Sopenharmony_ci/* The current state of operation */ 8453a5a1b3Sopenharmony_cistatic enum { 8553a5a1b3Sopenharmony_ci PHASE_IDLE, /* Normal state */ 8653a5a1b3Sopenharmony_ci PHASE_SOFT /* After CPU overload has been detected */ 8753a5a1b3Sopenharmony_ci} phase = PHASE_IDLE; 8853a5a1b3Sopenharmony_ci 8953a5a1b3Sopenharmony_ci/* Reset the SIGXCPU timer to the next t seconds */ 9053a5a1b3Sopenharmony_cistatic void reset_cpu_time(int t) { 9153a5a1b3Sopenharmony_ci long n; 9253a5a1b3Sopenharmony_ci struct rlimit rl; 9353a5a1b3Sopenharmony_ci struct rusage ru; 9453a5a1b3Sopenharmony_ci 9553a5a1b3Sopenharmony_ci /* Get the current CPU time of the current process */ 9653a5a1b3Sopenharmony_ci pa_assert_se(getrusage(RUSAGE_SELF, &ru) >= 0); 9753a5a1b3Sopenharmony_ci 9853a5a1b3Sopenharmony_ci n = ru.ru_utime.tv_sec + ru.ru_stime.tv_sec + t; 9953a5a1b3Sopenharmony_ci pa_assert_se(getrlimit(RLIMIT_CPU, &rl) >= 0); 10053a5a1b3Sopenharmony_ci 10153a5a1b3Sopenharmony_ci rl.rlim_cur = (rlim_t) n; 10253a5a1b3Sopenharmony_ci pa_assert_se(setrlimit(RLIMIT_CPU, &rl) >= 0); 10353a5a1b3Sopenharmony_ci} 10453a5a1b3Sopenharmony_ci 10553a5a1b3Sopenharmony_ci/* A simple, thread-safe puts() work-alike */ 10653a5a1b3Sopenharmony_cistatic void write_err(const char *p) { 10753a5a1b3Sopenharmony_ci pa_loop_write(2, p, strlen(p), NULL); 10853a5a1b3Sopenharmony_ci} 10953a5a1b3Sopenharmony_ci 11053a5a1b3Sopenharmony_ci/* The signal handler, called on every SIGXCPU */ 11153a5a1b3Sopenharmony_cistatic void signal_handler(int sig) { 11253a5a1b3Sopenharmony_ci int saved_errno; 11353a5a1b3Sopenharmony_ci 11453a5a1b3Sopenharmony_ci saved_errno = errno; 11553a5a1b3Sopenharmony_ci pa_assert(sig == SIGXCPU); 11653a5a1b3Sopenharmony_ci 11753a5a1b3Sopenharmony_ci if (phase == PHASE_IDLE) { 11853a5a1b3Sopenharmony_ci pa_usec_t now, elapsed; 11953a5a1b3Sopenharmony_ci 12053a5a1b3Sopenharmony_ci#ifdef PRINT_CPU_LOAD 12153a5a1b3Sopenharmony_ci char t[256]; 12253a5a1b3Sopenharmony_ci#endif 12353a5a1b3Sopenharmony_ci 12453a5a1b3Sopenharmony_ci now = pa_rtclock_now(); 12553a5a1b3Sopenharmony_ci elapsed = now - last_time; 12653a5a1b3Sopenharmony_ci 12753a5a1b3Sopenharmony_ci#ifdef PRINT_CPU_LOAD 12853a5a1b3Sopenharmony_ci pa_snprintf(t, sizeof(t), "Using %0.1f%% CPU\n", ((double) CPUTIME_INTERVAL_SOFT * (double) PA_USEC_PER_SEC) / (double) elapsed * 100.0); 12953a5a1b3Sopenharmony_ci write_err(t); 13053a5a1b3Sopenharmony_ci#endif 13153a5a1b3Sopenharmony_ci 13253a5a1b3Sopenharmony_ci if (((double) CPUTIME_INTERVAL_SOFT * (double) PA_USEC_PER_SEC) >= ((double) elapsed * (double) CPUTIME_PERCENT / 100.0)) { 13353a5a1b3Sopenharmony_ci static const char c = 'X'; 13453a5a1b3Sopenharmony_ci 13553a5a1b3Sopenharmony_ci write_err("Soft CPU time limit exhausted, terminating.\n"); 13653a5a1b3Sopenharmony_ci 13753a5a1b3Sopenharmony_ci /* Try a soft cleanup */ 13853a5a1b3Sopenharmony_ci (void) pa_write(the_pipe[1], &c, sizeof(c), NULL); 13953a5a1b3Sopenharmony_ci phase = PHASE_SOFT; 14053a5a1b3Sopenharmony_ci reset_cpu_time(CPUTIME_INTERVAL_HARD); 14153a5a1b3Sopenharmony_ci 14253a5a1b3Sopenharmony_ci } else { 14353a5a1b3Sopenharmony_ci 14453a5a1b3Sopenharmony_ci /* Everything's fine */ 14553a5a1b3Sopenharmony_ci reset_cpu_time(CPUTIME_INTERVAL_SOFT); 14653a5a1b3Sopenharmony_ci last_time = now; 14753a5a1b3Sopenharmony_ci } 14853a5a1b3Sopenharmony_ci 14953a5a1b3Sopenharmony_ci } else if (phase == PHASE_SOFT) { 15053a5a1b3Sopenharmony_ci write_err("Hard CPU time limit exhausted, terminating forcibly.\n"); 15153a5a1b3Sopenharmony_ci abort(); /* Forced exit */ 15253a5a1b3Sopenharmony_ci } 15353a5a1b3Sopenharmony_ci 15453a5a1b3Sopenharmony_ci errno = saved_errno; 15553a5a1b3Sopenharmony_ci} 15653a5a1b3Sopenharmony_ci 15753a5a1b3Sopenharmony_ci/* Callback for IO events on the FIFO */ 15853a5a1b3Sopenharmony_cistatic void callback(pa_mainloop_api*m, pa_io_event*e, int fd, pa_io_event_flags_t f, void *userdata) { 15953a5a1b3Sopenharmony_ci char c; 16053a5a1b3Sopenharmony_ci pa_assert(m); 16153a5a1b3Sopenharmony_ci pa_assert(e); 16253a5a1b3Sopenharmony_ci pa_assert(f == PA_IO_EVENT_INPUT); 16353a5a1b3Sopenharmony_ci pa_assert(e == io_event); 16453a5a1b3Sopenharmony_ci pa_assert(fd == the_pipe[0]); 16553a5a1b3Sopenharmony_ci 16653a5a1b3Sopenharmony_ci pa_log("Received request to terminate due to CPU overload."); 16753a5a1b3Sopenharmony_ci 16853a5a1b3Sopenharmony_ci (void) pa_read(the_pipe[0], &c, sizeof(c), NULL); 16953a5a1b3Sopenharmony_ci m->quit(m, 1); /* Quit the main loop */ 17053a5a1b3Sopenharmony_ci} 17153a5a1b3Sopenharmony_ci 17253a5a1b3Sopenharmony_ci/* Initializes CPU load limiter */ 17353a5a1b3Sopenharmony_ciint pa_cpu_limit_init(pa_mainloop_api *m) { 17453a5a1b3Sopenharmony_ci struct sigaction sa; 17553a5a1b3Sopenharmony_ci 17653a5a1b3Sopenharmony_ci pa_assert(m); 17753a5a1b3Sopenharmony_ci pa_assert(!api); 17853a5a1b3Sopenharmony_ci pa_assert(!io_event); 17953a5a1b3Sopenharmony_ci pa_assert(the_pipe[0] == -1); 18053a5a1b3Sopenharmony_ci pa_assert(the_pipe[1] == -1); 18153a5a1b3Sopenharmony_ci pa_assert(!installed); 18253a5a1b3Sopenharmony_ci 18353a5a1b3Sopenharmony_ci last_time = pa_rtclock_now(); 18453a5a1b3Sopenharmony_ci 18553a5a1b3Sopenharmony_ci /* Prepare the main loop pipe */ 18653a5a1b3Sopenharmony_ci if (pa_pipe_cloexec(the_pipe) < 0) { 18753a5a1b3Sopenharmony_ci pa_log("pipe() failed: %s", pa_cstrerror(errno)); 18853a5a1b3Sopenharmony_ci return -1; 18953a5a1b3Sopenharmony_ci } 19053a5a1b3Sopenharmony_ci 19153a5a1b3Sopenharmony_ci pa_make_fd_nonblock(the_pipe[0]); 19253a5a1b3Sopenharmony_ci pa_make_fd_nonblock(the_pipe[1]); 19353a5a1b3Sopenharmony_ci 19453a5a1b3Sopenharmony_ci api = m; 19553a5a1b3Sopenharmony_ci io_event = api->io_new(m, the_pipe[0], PA_IO_EVENT_INPUT, callback, NULL); 19653a5a1b3Sopenharmony_ci 19753a5a1b3Sopenharmony_ci phase = PHASE_IDLE; 19853a5a1b3Sopenharmony_ci 19953a5a1b3Sopenharmony_ci /* Install signal handler for SIGXCPU */ 20053a5a1b3Sopenharmony_ci memset(&sa, 0, sizeof(sa)); 20153a5a1b3Sopenharmony_ci sa.sa_handler = signal_handler; 20253a5a1b3Sopenharmony_ci sigemptyset(&sa.sa_mask); 20353a5a1b3Sopenharmony_ci sa.sa_flags = SA_RESTART; 20453a5a1b3Sopenharmony_ci 20553a5a1b3Sopenharmony_ci if (sigaction(SIGXCPU, &sa, &sigaction_prev) < 0) { 20653a5a1b3Sopenharmony_ci pa_cpu_limit_done(); 20753a5a1b3Sopenharmony_ci return -1; 20853a5a1b3Sopenharmony_ci } 20953a5a1b3Sopenharmony_ci 21053a5a1b3Sopenharmony_ci installed = true; 21153a5a1b3Sopenharmony_ci 21253a5a1b3Sopenharmony_ci reset_cpu_time(CPUTIME_INTERVAL_SOFT); 21353a5a1b3Sopenharmony_ci 21453a5a1b3Sopenharmony_ci return 0; 21553a5a1b3Sopenharmony_ci} 21653a5a1b3Sopenharmony_ci 21753a5a1b3Sopenharmony_ci/* Shutdown CPU load limiter */ 21853a5a1b3Sopenharmony_civoid pa_cpu_limit_done(void) { 21953a5a1b3Sopenharmony_ci 22053a5a1b3Sopenharmony_ci if (io_event) { 22153a5a1b3Sopenharmony_ci pa_assert(api); 22253a5a1b3Sopenharmony_ci api->io_free(io_event); 22353a5a1b3Sopenharmony_ci io_event = NULL; 22453a5a1b3Sopenharmony_ci api = NULL; 22553a5a1b3Sopenharmony_ci } 22653a5a1b3Sopenharmony_ci 22753a5a1b3Sopenharmony_ci pa_close_pipe(the_pipe); 22853a5a1b3Sopenharmony_ci 22953a5a1b3Sopenharmony_ci if (installed) { 23053a5a1b3Sopenharmony_ci pa_assert_se(sigaction(SIGXCPU, &sigaction_prev, NULL) >= 0); 23153a5a1b3Sopenharmony_ci installed = false; 23253a5a1b3Sopenharmony_ci } 23353a5a1b3Sopenharmony_ci} 23453a5a1b3Sopenharmony_ci 23553a5a1b3Sopenharmony_ci#else /* HAVE_SIGXCPU */ 23653a5a1b3Sopenharmony_ci 23753a5a1b3Sopenharmony_ciint pa_cpu_limit_init(pa_mainloop_api *m) { 23853a5a1b3Sopenharmony_ci return 0; 23953a5a1b3Sopenharmony_ci} 24053a5a1b3Sopenharmony_ci 24153a5a1b3Sopenharmony_civoid pa_cpu_limit_done(void) { 24253a5a1b3Sopenharmony_ci} 24353a5a1b3Sopenharmony_ci 24453a5a1b3Sopenharmony_ci#endif 245