/* * The ManaPlus Client * Copyright (C) 2011-2012 The ManaPlus Developers * * This file is part of The ManaPlus Client. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * any later version. * * This program 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 General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see <http://www.gnu.org/licenses/>. */ #include "utils/process.h" #include <string.h> #include <cstdarg> #include <cstdio> #include <stdlib.h> #include <unistd.h> #include "localconsts.h" #include "debug.h" const int timeOut = 10; #ifdef WIN32 #include <windows.h> int execFileWait(std::string pathName, std::string name A_UNUSED, std::string arg1, std::string arg2, int waitTime) { if (!waitTime) waitTime = timeOut; STARTUPINFO siStartupInfo; PROCESS_INFORMATION piProcessInfo; memset(&siStartupInfo, 0, sizeof(siStartupInfo)); memset(&piProcessInfo, 0, sizeof(piProcessInfo)); siStartupInfo.cb = sizeof(siStartupInfo); DWORD ret = -1; std::string args(pathName + " " + arg1); if (!arg2.empty()) args += " " + arg2; if (CreateProcess(pathName.c_str(), (char*)args.c_str(), nullptr, nullptr, false, CREATE_DEFAULT_ERROR_MODE, nullptr, nullptr, &siStartupInfo, &piProcessInfo) != false) { if (!WaitForSingleObject(piProcessInfo.hProcess, timeOut * 1000)) { if (GetExitCodeProcess(piProcessInfo.hProcess, &ret)) { CloseHandle(piProcessInfo.hProcess); CloseHandle(piProcessInfo.hThread); return ret; } } TerminateProcess(piProcessInfo.hProcess, -1); } CloseHandle(piProcessInfo.hProcess); CloseHandle(piProcessInfo.hThread); return -1; } bool execFile(std::string pathName, std::string name A_UNUSED, std::string arg1, std::string arg2) { STARTUPINFO siStartupInfo; PROCESS_INFORMATION piProcessInfo; memset(&siStartupInfo, 0, sizeof(siStartupInfo)); memset(&piProcessInfo, 0, sizeof(piProcessInfo)); siStartupInfo.cb = sizeof(siStartupInfo); std::string args(pathName + " " + arg1); if (!arg2.empty()) args += " " + arg2; bool res = CreateProcess(pathName.c_str(), const_cast<char*>( args.c_str()), nullptr, nullptr, false, CREATE_DEFAULT_ERROR_MODE, nullptr, nullptr, &siStartupInfo, &piProcessInfo); CloseHandle(piProcessInfo.hProcess); CloseHandle(piProcessInfo.hThread); return res; } bool openBrowser(std::string url) { return (int)ShellExecute(nullptr, "open", url.c_str(), nullptr, nullptr, SW_SHOWNORMAL) > 32; } #elif defined(__APPLE__) int execFileWait(std::string pathName, std::string name, std::string arg1, std::string arg2, int waitTime) { return -1; } bool execFile(std::string pathName, std::string name, std::string arg1, std::string arg2) { return false; } bool openBrowser(std::string url) { return false; } #elif defined __linux__ || defined __linux #include <sys/types.h> #include <sys/wait.h> int execFileWait(std::string pathName, std::string name, std::string arg1, std::string arg2, int waitTime) { pid_t mon_pid; int status; if (!waitTime) waitTime = timeOut; if ((mon_pid = fork()) == -1) { // fork error return -1; } else if (!mon_pid) { // monitoring child pid_t pid; if ((pid = fork()) == -1) { // fork error return -1; } else if (!pid) { // work child if (arg2.empty()) { execl(pathName.c_str(), name.c_str(), arg1.c_str(), static_cast<char *>(nullptr)); } else { execl(pathName.c_str(), name.c_str(), arg1.c_str(), arg2.c_str(), static_cast<char *>(nullptr)); } _exit(-1); } // monitoring process pid_t sleep_pid; if ((sleep_pid = fork()) == -1) { // fork error return -1; } else if (!sleep_pid) { // sleep pid sleep (waitTime); // printf ("time out\n"); execl("/bin/true", "/bin/true", static_cast<char *>(nullptr)); _exit(-1); } // monitoring process const pid_t exited_pid = wait(&status); int ret = -1; if (exited_pid == pid) { kill(sleep_pid, SIGKILL); if (WIFEXITED(status)) ret = WEXITSTATUS(status); } else { kill(pid, SIGKILL); ret = -1; } wait(nullptr); execl("/bin/true", "/bin/true", static_cast<char *>(nullptr)); _exit(ret); } // monitoring parent waitpid(mon_pid, &status, 0); if (WIFEXITED(status)) return WEXITSTATUS(status); return -1; } bool execFile(std::string pathName, std::string name, std::string arg1, std::string arg2) { pid_t pid; if ((pid = fork()) == -1) { // fork error return false; } else if (!pid) { // work child if (arg2.empty()) { execl(pathName.c_str(), name.c_str(), arg1.c_str(), static_cast<char *>(nullptr)); } else { execl(pathName.c_str(), name.c_str(), arg1.c_str(), arg2.c_str(), static_cast<char *>(nullptr)); } return false; } return true; } bool openBrowser(std::string url) { return execFile("/usr/bin/xdg-open", "/usr/bin/xdg-open", url, "") == 0; } #else int execFileWait(std::string pathName, std::string name, std::string arg1, std::string arg2, int waitTime) { return -1; } bool execFile(std::string pathName, std::string name, std::string arg1, std::string arg2) { return false; } bool openBrowser(std::string url) { return false; } #endif