/*
* The ManaPlus Client
* Copyright (C) 2011-2015 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 <unistd.h>
#include "localconsts.h"
#ifdef USE_SDL2
#ifdef ANDROID
#include <SDL_system.h>
#endif
#endif
#include "debug.h"
const int timeOut = 10;
#ifdef WIN32
#include "utils/stringutils.h"
#include <windows.h>
int execFileWait(const std::string &pathName, const std::string &name A_UNUSED,
const std::string &arg1, const std::string &arg2,
const int waitTime A_UNUSED)
{
// 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(std::string(pathName).append(" ").append(arg1));
if (!arg2.empty())
args.append(" ").append(arg2);
if (CreateProcess(pathName.c_str(), const_cast<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(const std::string &pathName, const std::string &name A_UNUSED,
const std::string &arg1, const 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(std::string(pathName).append(" ").append(arg1));
if (!arg2.empty())
args.append(" ").append(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;
}
#elif defined __linux__ || defined __linux || defined __APPLE__
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/wait.h>
#include <signal.h>
int execFileWait(const std::string &pathName, const std::string &name,
const std::string &arg1, const 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);
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(const std::string &pathName, const std::string &name,
const std::string &arg1, const std::string &arg2)
{
struct stat statbuf;
// file not exists
if (stat(pathName.c_str(), &statbuf))
return false;
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));
}
_exit(-1);
return false;
}
return true;
}
#else
int execFileWait(const std::string &pathName, const std::string &name,
const std::string &arg1, const std::string &arg2,
int waitTime)
{
return -1;
}
bool execFile(const std::string &pathName, const std::string &name,
const std::string &arg1, const std::string &arg2)
{
return false;
}
#endif
#ifdef WIN32
bool openBrowser(std::string url)
{
return reinterpret_cast<int>(ShellExecute(nullptr, "open",
replaceAll(url, " ", "").c_str(),
nullptr, nullptr, SW_SHOWNORMAL)) > 32;
}
#elif defined ANDROID
#include "utils/stringutils.h"
#ifndef USE_SDL2
#include <SDL_screenkeyboard.h>
#endif
bool openBrowser(std::string url)
{
#ifdef USE_SDL2
SDL_OpenBrowser(replaceAll(url, " ", "").c_str());
#else
SDL_ANDROID_OpenBrowser(replaceAll(url, " ", "").c_str());
#endif
return true;
}
#elif defined __APPLE__
bool openBrowser(std::string url)
{
return execFile("/usr/bin/open", "/usr/bin/open", url, "");
}
#elif defined __OpenBSD__ || defined __FreeBSD__ || defined __DragonFly__
bool openBrowser(std::string url)
{
return execFile("/usr/local/bin/xdg-open",
"/usr/local/bin/xdg-open", url, "");
}
#elif defined __linux__ || defined __linux
bool openBrowser(std::string url)
{
return execFile("/usr/bin/xdg-open", "/usr/bin/xdg-open", url, "");
}
#elif defined __native_client__
bool openBrowser(std::string url)
{
naclPostMessage("open-browser", url);
return true;
}
#else
bool openBrowser(std::string url)
{
return false;
}
#endif
#ifdef WIN32
void setPriority(const bool big)
{
HANDLE hCurrentProcess = GetCurrentProcess();
if (big)
SetPriorityClass(hCurrentProcess, ABOVE_NORMAL_PRIORITY_CLASS);
else
SetPriorityClass(hCurrentProcess, BELOW_NORMAL_PRIORITY_CLASS);
}
#else
void setPriority(const bool big A_UNUSED)
{
}
#endif
#ifdef __native_client__
#include <ppapi_simple/ps.h>
#include <ppapi_simple/ps_event.h>
#include <ppapi/cpp/instance.h>
#include <ppapi/cpp/var.h>
#include <mutex>
#include <condition_variable>
struct _NaclMessageHandle final
{
bool handled;
std::string type;
std::string message;
std::condition_variable condv;
};
void naclPostMessage(const std::string &type, const std::string &message)
{
pp::Var msgVar = pp::Var(std::string(type).append(":").append(message));
pp::Instance(PSGetInstanceId()).PostMessage(msgVar);
}
void naclMessageHandlerFunc(struct PP_Var key,
struct PP_Var value,
void* user_data)
{
NaclMessageHandle *handle = (NaclMessageHandle *)user_data;
if (key.type != PP_VARTYPE_STRING || value.type != PP_VARTYPE_STRING)
return;
if (pp::Var(key).AsString() != handle->type)
return;
handle->handled = true;
handle->message = pp::Var(value).AsString();
handle->condv.notify_one();
}
NaclMessageHandle *naclRegisterMessageHandler(const std::string &type)
{
NaclMessageHandle *handle = new NaclMessageHandle;
handle->handled = false;
handle->type = type;
PSEventRegisterMessageHandler(type.c_str(),
naclMessageHandlerFunc,
(void *)handle);
return handle;
}
void naclUnregisterMessageHandler(NaclMessageHandle *handle)
{
PSEventRegisterMessageHandler(handle->type.c_str(),
nullptr,
nullptr);
delete handle;
}
std::string naclWaitForMessage(NaclMessageHandle *handle)
{
std::mutex mtx;
std::unique_lock <std::mutex> lck(mtx);
while (!handle->handled)
handle->condv.wait(lck);
handle->handled = false;
return handle->message;
}
#endif