You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
183 lines
4.1 KiB
183 lines
4.1 KiB
/* |
|
* Copyright (C) 2013-2015 Daniel Scharrer |
|
* |
|
* This software is provided 'as-is', without any express or implied |
|
* warranty. In no event will the author(s) be held liable for any damages |
|
* arising from the use of this software. |
|
* |
|
* Permission is granted to anyone to use this software for any purpose, |
|
* including commercial applications, and to alter it and redistribute it |
|
* freely, subject to the following restrictions: |
|
* |
|
* 1. The origin of this software must not be misrepresented; you must not |
|
* claim that you wrote the original software. If you use this software |
|
* in a product, an acknowledgment in the product documentation would be |
|
* appreciated but is not required. |
|
* 2. Altered source versions must be plainly marked as such, and must not be |
|
* misrepresented as being the original software. |
|
* 3. This notice may not be removed or altered from any source distribution. |
|
*/ |
|
|
|
#include "util/process.hpp" |
|
|
|
#include <sstream> |
|
#include <iostream> |
|
|
|
#include "configure.hpp" |
|
|
|
#if defined(_WIN32) |
|
|
|
#include <string.h> |
|
#include <windows.h> |
|
|
|
#include <boost/format.hpp> |
|
#include <boost/algorithm/string.hpp> |
|
|
|
#elif INNOEXTRACT_HAVE_POSIX_SPAWNP || (INNOEXTRACT_HAVE_FORK && INNOEXTRACT_HAVE_EXECVP) |
|
|
|
#if INNOEXTRACT_HAVE_POSIX_SPAWNP |
|
#include <spawn.h> |
|
#if defined(__FreeBSD__) && defined(__GNUC__) && __GNUC__ >= 4 |
|
/* |
|
* When combining -flto and -fvisibility=hidden we and up with a hidden |
|
* 'environ' symbol in crt1.o on FreeBSD 9, which causes the link to fail. |
|
*/ |
|
extern char ** environ __attribute__((visibility("default"))); |
|
#else |
|
extern char ** environ; |
|
#endif |
|
#else |
|
#include <unistd.h> |
|
#endif |
|
|
|
#if INNOEXTRACT_HAVE_WAITPID |
|
#include <sys/wait.h> |
|
#endif |
|
|
|
#else |
|
|
|
#include <cstdlib> |
|
|
|
#endif |
|
|
|
#include "util/encoding.hpp" |
|
|
|
namespace util { |
|
|
|
#if defined(_WIN32) || !(INNOEXTRACT_HAVE_POSIX_SPAWNP \ |
|
|| (INNOEXTRACT_HAVE_FORK && INNOEXTRACT_HAVE_EXECVP)) |
|
static std::string format_command_line(const char * const args[]) { |
|
|
|
std::ostringstream oss; |
|
|
|
for(size_t i = 0; args[i]; i++) { |
|
if(i != 0) { |
|
oss << ' '; |
|
} |
|
oss << '"'; |
|
for(const char * arg = args[i]; *arg; arg++) { |
|
char c = *arg; |
|
if(c == '\\' || c == '\"' || c == ' ' || c == '\'' || c == '$' || c == '!') { |
|
oss << '\\'; |
|
} |
|
oss << c; |
|
} |
|
oss << '"'; |
|
} |
|
|
|
return oss.str(); |
|
} |
|
#endif |
|
|
|
int run(const char * const args[]) { |
|
|
|
std::cout.flush(); |
|
std::cerr.flush(); |
|
|
|
#if defined(_WIN32) |
|
|
|
// Format the command line arguments |
|
std::string exe; |
|
wtf8_to_utf16le(args[0], exe); |
|
exe.push_back('\0'); |
|
std::string cmdline; |
|
wtf8_to_utf16le(format_command_line(args + 1), exe); |
|
cmdline.push_back('\0'); |
|
|
|
STARTUPINFO si; |
|
memset(&si, 0, sizeof(STARTUPINFO)); |
|
si.cb = sizeof(STARTUPINFO); |
|
|
|
PROCESS_INFORMATION pi; |
|
memset(&pi, 0, sizeof(PROCESS_INFORMATION)); |
|
|
|
bool success = (CreateProcessW(reinterpret_cast<LPCWSTR>(exe.c_str()), |
|
reinterpret_cast<LPWSTR>(&cmdline[0]), 0, 0, 0, 0, 0, 0, &si, &pi) != 0); |
|
|
|
if(!success) { |
|
return -1; // Could not start process |
|
} |
|
|
|
int status = int(WaitForSingleObject(pi.hProcess, INFINITE)); |
|
|
|
CloseHandle(pi.hProcess); |
|
CloseHandle(pi.hThread); |
|
|
|
return status; |
|
|
|
#elif INNOEXTRACT_HAVE_POSIX_SPAWNP || (INNOEXTRACT_HAVE_FORK && INNOEXTRACT_HAVE_EXECVP) |
|
|
|
char ** argv = const_cast<char **>(args); |
|
|
|
pid_t pid = -1; |
|
|
|
#if INNOEXTRACT_HAVE_POSIX_SPAWNP |
|
|
|
// Fast POSIX implementation: posix_spawnp avoids unnecessary vm copies |
|
|
|
// Run the executable in a new process |
|
(void)posix_spawnp(&pid, argv[0], NULL, NULL, argv, environ); |
|
|
|
#else |
|
|
|
// Compatibility POSIX implementation |
|
|
|
// Start a new process |
|
pid = fork(); |
|
if(pid == 0) { |
|
|
|
// Run the executable |
|
(void)execvp(argv[0], argv); |
|
|
|
exit(-1); |
|
} |
|
|
|
#endif |
|
|
|
if(pid < 0) { |
|
return -1; |
|
} |
|
|
|
#if INNOEXTRACT_HAVE_WAITPID |
|
int status; |
|
(void)waitpid(pid, &status, 0); |
|
if(WIFEXITED(status) && WEXITSTATUS(status) < 127) { |
|
return WEXITSTATUS(status); |
|
} else if(WIFSIGNALED(status)) { |
|
return -WTERMSIG(status); |
|
} else { |
|
return -1; |
|
} |
|
#else |
|
# warning "Waiting for processes not supported on this system." |
|
#endif |
|
|
|
return 0; |
|
|
|
#else |
|
return std::system(format_command_line(args).c_str()); |
|
#endif |
|
|
|
} |
|
|
|
} // namespace util
|
|
|