// This file is part of BOINC. // http://boinc.berkeley.edu // Copyright (C) 2020 University of California // // BOINC is free software; you can redistribute it and/or modify it // under the terms of the GNU Lesser General Public License // as published by the Free Software Foundation, // either version 3 of the License, or (at your option) any later version. // // BOINC is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. // See the GNU Lesser General Public License for more details. // // You should have received a copy of the GNU Lesser General Public License // along with BOINC. If not, see . #include #ifdef _WIN32 #include "diagnostics.h" #include "boinc_win.h" #include "win_util.h" #else #include "config.h" #include #include #include #include #include #if HAVE_CSIGNAL #include #elif HAVE_SYS_SIGNAL_H #include #elif HAVE_SIGNAL_H #include #endif #endif #include "common_defs.h" #include "procinfo.h" #include "str_util.h" #include "util.h" #include "proc_control.h" using std::vector; //#define DEBUG_PROC_CONTROL #ifdef DEBUG_PROC_CONTROL #include #endif static void get_descendants_aux(PROC_MAP& pm, int pid, vector& pids) { PROC_MAP::iterator i = pm.find(pid); if (i == pm.end()) return; PROCINFO& p = i->second; if (p.scanned) return; // avoid infinite recursion p.scanned = true; for (unsigned int j=0; j& pids) { int retval; PROC_MAP pm; pids.clear(); retval = procinfo_setup(pm); if (retval) return; get_descendants_aux(pm, pid, pids); #ifdef DEBUG_PROC_CONTROL fprintf(stderr, "descendants of %d:\n", pid); for (unsigned int i=0; ipids, DWORD calling_thread_id, bool resume, bool check_exempt ) { HANDLE threads, thread; THREADENTRY32 te = {0}; int retval = 0; DWORD n; static vector suspended_threads; #ifdef DEBUG_PROC_CONTROL fprintf(stderr, "start: check_exempt %d %s\n", check_exempt, precision_time_to_string(dtime())); fprintf(stderr, "%s processes", resume?"resume":"suspend"); for (unsigned int i=0; i& pids) { int status; for (unsigned int i=0; i= 0) { return true; } } return false; } #endif void kill_all(vector& pids) { for (unsigned int i=0; i descendants; // on Win, kill descendants directly // get_descendants(GetCurrentProcessId(), descendants); kill_all(descendants); } #else // Same, but if child_pid is nonzero, // give it a chance to exit gracefully on Unix // void kill_descendants(int child_pid) { vector descendants; // on Unix, ask main process nicely. // it descendants still exist after 10 sec, use the nuclear option // get_descendants(getpid(), descendants); if (child_pid) { ::kill(child_pid, SIGTERM); for (int i=0; i<10; i++) { if (!any_process_exists(descendants)) { return; } boinc_sleep(1); } kill_all(descendants); // kill any processes that might have been created // in the last 10 secs get_descendants(getpid(), descendants); } kill_all(descendants); } #endif // suspend/resume the descendants of the calling process. // Unix version lets you choose the stop signal: // SIGSTOP (the default) can't be caught. // SIGTSTP can be caught, but it has no effect for processes without a TTY. // So it's useful only for programs that are wrappers of some sort; // they must catch and handle it. // #ifdef _WIN32 void suspend_or_resume_descendants(bool resume) { vector descendants; int pid = GetCurrentProcessId(); get_descendants(pid, descendants); suspend_or_resume_threads(descendants, 0, resume, false); } #else void suspend_or_resume_descendants(bool resume, bool use_tstp) { vector descendants; int pid = getpid(); get_descendants(pid, descendants); for (unsigned int i=0; i pids; pids.push_back(pid); suspend_or_resume_threads(pids, 0, resume, false); } #else void suspend_or_resume_process(int pid, bool resume, bool use_tstp) { ::kill(pid, resume?SIGCONT:(use_tstp?SIGTSTP:SIGSTOP)); } #endif // return OS-specific value associated with priority code // int process_priority_value(int priority) { #ifdef _WIN32 switch (priority) { case PROCESS_PRIORITY_LOWEST: return IDLE_PRIORITY_CLASS; case PROCESS_PRIORITY_LOW: return BELOW_NORMAL_PRIORITY_CLASS; case PROCESS_PRIORITY_NORMAL: return NORMAL_PRIORITY_CLASS; case PROCESS_PRIORITY_HIGH: return ABOVE_NORMAL_PRIORITY_CLASS; case PROCESS_PRIORITY_HIGHEST: return HIGH_PRIORITY_CLASS; } return 0; #else switch (priority) { case PROCESS_PRIORITY_LOWEST: return PROCESS_IDLE_PRIORITY; case PROCESS_PRIORITY_LOW: return PROCESS_MEDIUM_PRIORITY; case PROCESS_PRIORITY_NORMAL: return PROCESS_NORMAL_PRIORITY; case PROCESS_PRIORITY_HIGH: return PROCESS_ABOVE_NORMAL_PRIORITY; case PROCESS_PRIORITY_HIGHEST: return PROCESS_HIGH_PRIORITY; } return 0; #endif }