2021-06-11 12:29:21 +00:00
|
|
|
/*
|
2007-03-14 20:24:57 +00:00
|
|
|
* COPYRIGHT: See COPYING in the top level directory
|
|
|
|
* PROJECT: ReactOS system libraries
|
2015-09-19 13:50:57 +00:00
|
|
|
* FILE: lib/sdk/crt/process/_system.c
|
2007-03-14 20:24:57 +00:00
|
|
|
* PURPOSE: Excutes a shell command
|
|
|
|
* PROGRAMER: Ariadne
|
2023-03-05 12:01:14 +00:00
|
|
|
* Katayama Hirofumi MZ
|
2007-03-14 20:24:57 +00:00
|
|
|
* UPDATE HISTORY:
|
|
|
|
* 04/03/99: Created
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <precomp.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <process.h>
|
|
|
|
|
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
|
|
|
int system(const char *command)
|
|
|
|
{
|
|
|
|
char *szCmdLine = NULL;
|
|
|
|
char *szComSpec = NULL;
|
|
|
|
PROCESS_INFORMATION ProcessInformation;
|
|
|
|
STARTUPINFOA StartupInfo;
|
|
|
|
BOOL result;
|
2023-03-05 14:28:00 +00:00
|
|
|
int exit_code;
|
2023-03-05 12:01:14 +00:00
|
|
|
char cmd_exe[MAX_PATH];
|
2007-03-14 20:24:57 +00:00
|
|
|
|
|
|
|
szComSpec = getenv("COMSPEC");
|
|
|
|
|
|
|
|
// system should return 0 if command is null and the shell is found
|
|
|
|
|
|
|
|
if (command == NULL) {
|
2023-03-05 12:01:14 +00:00
|
|
|
return (szComSpec == NULL) ? 0 : 1;
|
2010-05-29 09:23:23 +00:00
|
|
|
}
|
|
|
|
|
2023-03-05 12:01:14 +00:00
|
|
|
if (!szComSpec || GetFileAttributesA(szComSpec) == INVALID_FILE_ATTRIBUTES)
|
2007-03-14 20:24:57 +00:00
|
|
|
{
|
2023-03-05 12:01:14 +00:00
|
|
|
GetSystemDirectoryA(cmd_exe, _countof(cmd_exe));
|
|
|
|
strcat(cmd_exe, "\\cmd.exe");
|
|
|
|
szComSpec = cmd_exe;
|
2007-03-14 20:24:57 +00:00
|
|
|
}
|
|
|
|
|
2023-03-05 14:20:12 +00:00
|
|
|
szCmdLine = malloc(1 + strlen(szComSpec) + 5 + strlen(command) + 1);
|
2007-03-14 20:24:57 +00:00
|
|
|
if (szCmdLine == NULL)
|
|
|
|
{
|
2023-03-05 14:28:00 +00:00
|
|
|
_set_errno(ENOMEM);
|
|
|
|
return -1;
|
2007-03-14 20:24:57 +00:00
|
|
|
}
|
|
|
|
|
2023-03-05 12:01:14 +00:00
|
|
|
strcpy(szCmdLine, "\"");
|
|
|
|
strcat(szCmdLine, szComSpec);
|
|
|
|
strcat(szCmdLine, "\" /C ");
|
2007-03-14 20:24:57 +00:00
|
|
|
strcat(szCmdLine, command);
|
|
|
|
|
|
|
|
//command file has invalid format ENOEXEC
|
|
|
|
|
2023-03-05 12:01:14 +00:00
|
|
|
memset(&StartupInfo, 0, sizeof(StartupInfo));
|
2007-03-14 20:24:57 +00:00
|
|
|
StartupInfo.cb = sizeof(StartupInfo);
|
|
|
|
StartupInfo.dwFlags = STARTF_USESHOWWINDOW;
|
|
|
|
StartupInfo.wShowWindow = SW_SHOWDEFAULT;
|
|
|
|
|
2023-03-05 12:01:14 +00:00
|
|
|
// In order to disable Ctrl+C, the process is created with CREATE_NEW_PROCESS_GROUP.
|
|
|
|
// Thus, SetConsoleCtrlHandler(NULL, TRUE) is made on behalf of the new process.
|
2007-03-14 20:24:57 +00:00
|
|
|
|
2023-03-05 12:01:14 +00:00
|
|
|
//SIGCHILD should be blocked as well
|
2007-03-14 20:24:57 +00:00
|
|
|
|
|
|
|
result = CreateProcessA(szComSpec,
|
|
|
|
szCmdLine,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
TRUE,
|
|
|
|
CREATE_NEW_PROCESS_GROUP,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
&StartupInfo,
|
|
|
|
&ProcessInformation);
|
2023-03-05 14:20:12 +00:00
|
|
|
free(szCmdLine);
|
2007-03-14 20:24:57 +00:00
|
|
|
|
|
|
|
if (result == FALSE)
|
|
|
|
{
|
|
|
|
_dosmaperr(GetLastError());
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
CloseHandle(ProcessInformation.hThread);
|
|
|
|
|
2023-03-05 12:01:14 +00:00
|
|
|
/* Wait for the process to exit */
|
2023-03-05 14:28:00 +00:00
|
|
|
_cwait(&exit_code, (intptr_t)ProcessInformation.hProcess, 0);
|
2023-03-05 12:01:14 +00:00
|
|
|
|
2007-03-14 20:24:57 +00:00
|
|
|
CloseHandle(ProcessInformation.hProcess);
|
|
|
|
|
2023-03-05 12:01:14 +00:00
|
|
|
_set_errno(0);
|
2023-03-05 14:28:00 +00:00
|
|
|
return exit_code;
|
2007-03-14 20:24:57 +00:00
|
|
|
}
|
2008-05-28 21:08:23 +00:00
|
|
|
|
2023-03-05 14:20:12 +00:00
|
|
|
int CDECL _wsystem(const wchar_t *cmd)
|
2008-05-28 21:08:23 +00:00
|
|
|
{
|
2023-03-05 12:01:14 +00:00
|
|
|
wchar_t *cmdline = NULL;
|
|
|
|
wchar_t *comspec = NULL;
|
|
|
|
PROCESS_INFORMATION process_info;
|
|
|
|
STARTUPINFOW startup_info;
|
|
|
|
BOOL result;
|
2023-03-05 14:28:00 +00:00
|
|
|
int exit_code;
|
2023-03-05 12:01:14 +00:00
|
|
|
wchar_t cmd_exe[MAX_PATH];
|
|
|
|
|
|
|
|
comspec = _wgetenv(L"COMSPEC");
|
|
|
|
|
|
|
|
/* _wsystem should return 0 if cmd is null and the shell is found */
|
|
|
|
if (cmd == NULL)
|
|
|
|
{
|
|
|
|
return (comspec == NULL) ? 0 : 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (comspec == NULL || GetFileAttributesW(comspec) == INVALID_FILE_ATTRIBUTES)
|
|
|
|
{
|
|
|
|
GetSystemDirectoryW(cmd_exe, _countof(cmd_exe));
|
|
|
|
wcscat(cmd_exe, L"\\cmd.exe");
|
|
|
|
comspec = cmd_exe;
|
|
|
|
}
|
|
|
|
|
2023-03-05 14:20:12 +00:00
|
|
|
cmdline = malloc((1 + wcslen(comspec) + 5 + wcslen(cmd) + 1) * sizeof(wchar_t));
|
2023-03-05 12:01:14 +00:00
|
|
|
if (cmdline == NULL)
|
|
|
|
{
|
2023-03-05 14:28:00 +00:00
|
|
|
_set_errno(ENOMEM);
|
2023-03-05 12:01:14 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
wcscpy(cmdline, L"\"");
|
|
|
|
wcscat(cmdline, comspec);
|
|
|
|
wcscat(cmdline, L"\" /C ");
|
|
|
|
wcscat(cmdline, cmd);
|
|
|
|
|
|
|
|
/* command file has invalid format ENOEXEC */
|
|
|
|
|
|
|
|
memset(&startup_info, 0, sizeof(startup_info));
|
|
|
|
startup_info.cb = sizeof(startup_info);
|
|
|
|
startup_info.dwFlags = STARTF_USESHOWWINDOW;
|
|
|
|
startup_info.wShowWindow = SW_SHOWDEFAULT;
|
|
|
|
|
|
|
|
/* In order to disable Ctrl+C, the process is created with CREATE_NEW_PROCESS_GROUP.
|
|
|
|
Thus, SetConsoleCtrlHandler(NULL, TRUE) is made on behalf of the new process. */
|
|
|
|
|
|
|
|
/* SIGCHILD should be blocked as well */
|
|
|
|
|
|
|
|
/* Create the process to execute the command */
|
|
|
|
result = CreateProcessW(comspec,
|
|
|
|
cmdline,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
TRUE,
|
|
|
|
CREATE_NEW_PROCESS_GROUP,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
&startup_info,
|
|
|
|
&process_info);
|
2023-03-05 14:20:12 +00:00
|
|
|
free(cmdline);
|
2023-03-05 12:01:14 +00:00
|
|
|
|
|
|
|
if (!result)
|
|
|
|
{
|
|
|
|
_dosmaperr(GetLastError());
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
CloseHandle(process_info.hThread);
|
|
|
|
|
|
|
|
/* Wait for the process to exit */
|
2023-03-05 14:28:00 +00:00
|
|
|
_cwait(&exit_code, (intptr_t)process_info.hProcess, 0);
|
2023-03-05 12:01:14 +00:00
|
|
|
|
|
|
|
CloseHandle(process_info.hProcess);
|
|
|
|
|
|
|
|
_set_errno(0);
|
2023-03-05 14:28:00 +00:00
|
|
|
return exit_code;
|
2008-05-28 21:08:23 +00:00
|
|
|
}
|