2013-09-23 01:23:00 +04:00
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <winpr/crt.h>
|
|
|
|
#include <winpr/tchar.h>
|
|
|
|
#include <winpr/synch.h>
|
|
|
|
#include <winpr/thread.h>
|
2013-09-23 22:44:59 +04:00
|
|
|
#include <winpr/environment.h>
|
2015-08-28 11:31:22 +03:00
|
|
|
#include <winpr/pipe.h>
|
|
|
|
|
2016-05-25 16:47:58 +03:00
|
|
|
#define TESTENV_A "HELLO=WORLD"
|
|
|
|
#define TESTENV_T _T(TESTENV_A)
|
2013-09-23 01:23:00 +04:00
|
|
|
|
|
|
|
int TestThreadCreateProcess(int argc, char* argv[])
|
|
|
|
{
|
|
|
|
BOOL status;
|
2013-09-23 22:44:59 +04:00
|
|
|
DWORD exitCode;
|
2013-09-23 01:23:00 +04:00
|
|
|
LPCTSTR lpApplicationName;
|
|
|
|
LPTSTR lpCommandLine;
|
|
|
|
LPSECURITY_ATTRIBUTES lpProcessAttributes;
|
|
|
|
LPSECURITY_ATTRIBUTES lpThreadAttributes;
|
|
|
|
BOOL bInheritHandles;
|
|
|
|
DWORD dwCreationFlags;
|
|
|
|
LPVOID lpEnvironment;
|
|
|
|
LPCTSTR lpCurrentDirectory;
|
|
|
|
STARTUPINFO StartupInfo;
|
|
|
|
PROCESS_INFORMATION ProcessInformation;
|
2013-09-23 22:44:59 +04:00
|
|
|
LPTCH lpszEnvironmentBlock;
|
2015-08-28 11:31:22 +03:00
|
|
|
HANDLE pipe_read = NULL;
|
|
|
|
HANDLE pipe_write = NULL;
|
2016-05-25 16:47:58 +03:00
|
|
|
char buf[1024];
|
2015-08-28 11:31:22 +03:00
|
|
|
DWORD read_bytes;
|
|
|
|
int ret = 0;
|
2016-05-25 16:47:58 +03:00
|
|
|
SECURITY_ATTRIBUTES saAttr;
|
2013-09-23 22:44:59 +04:00
|
|
|
|
2021-07-29 11:18:52 +03:00
|
|
|
WINPR_UNUSED(argc);
|
|
|
|
WINPR_UNUSED(argv);
|
|
|
|
|
2013-09-23 22:44:59 +04:00
|
|
|
lpszEnvironmentBlock = GetEnvironmentStrings();
|
2013-09-23 01:23:00 +04:00
|
|
|
|
|
|
|
lpApplicationName = NULL;
|
2013-09-25 23:47:18 +04:00
|
|
|
|
|
|
|
#ifdef _WIN32
|
2016-05-25 16:47:58 +03:00
|
|
|
lpCommandLine = _T("cmd /C set");
|
2013-09-25 23:47:18 +04:00
|
|
|
#else
|
2013-09-23 22:44:59 +04:00
|
|
|
lpCommandLine = _T("printenv");
|
2013-09-25 23:47:18 +04:00
|
|
|
#endif
|
|
|
|
|
2013-09-23 01:23:00 +04:00
|
|
|
lpProcessAttributes = NULL;
|
|
|
|
lpThreadAttributes = NULL;
|
|
|
|
bInheritHandles = FALSE;
|
|
|
|
dwCreationFlags = 0;
|
2016-05-25 16:47:58 +03:00
|
|
|
#ifdef _UNICODE
|
|
|
|
dwCreationFlags |= CREATE_UNICODE_ENVIRONMENT;
|
|
|
|
#endif
|
2013-09-23 22:44:59 +04:00
|
|
|
lpEnvironment = lpszEnvironmentBlock;
|
2013-09-23 01:23:00 +04:00
|
|
|
lpCurrentDirectory = NULL;
|
|
|
|
ZeroMemory(&StartupInfo, sizeof(STARTUPINFO));
|
|
|
|
StartupInfo.cb = sizeof(STARTUPINFO);
|
|
|
|
ZeroMemory(&ProcessInformation, sizeof(PROCESS_INFORMATION));
|
|
|
|
|
2019-11-06 17:24:51 +03:00
|
|
|
status = CreateProcess(lpApplicationName, lpCommandLine, lpProcessAttributes,
|
|
|
|
lpThreadAttributes, bInheritHandles, dwCreationFlags, lpEnvironment,
|
|
|
|
lpCurrentDirectory, &StartupInfo, &ProcessInformation);
|
2013-09-23 01:23:00 +04:00
|
|
|
|
2015-05-05 18:25:17 +03:00
|
|
|
if (!status)
|
|
|
|
{
|
2019-11-06 17:24:51 +03:00
|
|
|
printf("CreateProcess failed. error=%" PRIu32 "\n", GetLastError());
|
2015-05-05 18:25:17 +03:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2016-05-25 16:47:58 +03:00
|
|
|
if (WaitForSingleObject(ProcessInformation.hProcess, 5000) != WAIT_OBJECT_0)
|
|
|
|
{
|
2019-11-06 17:24:51 +03:00
|
|
|
printf("Failed to wait for first process. error=%" PRIu32 "\n", GetLastError());
|
2016-05-25 16:47:58 +03:00
|
|
|
return 1;
|
|
|
|
}
|
2015-08-28 11:31:22 +03:00
|
|
|
|
|
|
|
exitCode = 0;
|
|
|
|
status = GetExitCodeProcess(ProcessInformation.hProcess, &exitCode);
|
|
|
|
|
2019-11-06 17:24:51 +03:00
|
|
|
printf("GetExitCodeProcess status: %" PRId32 "\n", status);
|
|
|
|
printf("Process exited with code: 0x%08" PRIX32 "\n", exitCode);
|
2015-08-28 11:31:22 +03:00
|
|
|
|
|
|
|
CloseHandle(ProcessInformation.hProcess);
|
|
|
|
CloseHandle(ProcessInformation.hThread);
|
2013-09-23 22:44:59 +04:00
|
|
|
FreeEnvironmentStrings(lpszEnvironmentBlock);
|
|
|
|
|
2015-08-28 11:31:22 +03:00
|
|
|
/* Test stdin,stdout,stderr redirection */
|
|
|
|
|
2016-05-25 16:47:58 +03:00
|
|
|
saAttr.nLength = sizeof(SECURITY_ATTRIBUTES);
|
|
|
|
saAttr.bInheritHandle = TRUE;
|
|
|
|
saAttr.lpSecurityDescriptor = NULL;
|
|
|
|
|
|
|
|
if (!CreatePipe(&pipe_read, &pipe_write, &saAttr, 0))
|
2015-08-28 11:31:22 +03:00
|
|
|
{
|
2019-11-06 17:24:51 +03:00
|
|
|
printf("Pipe creation failed. error=%" PRIu32 "\n", GetLastError());
|
2015-08-28 11:31:22 +03:00
|
|
|
return 1;
|
|
|
|
}
|
2016-05-25 16:47:58 +03:00
|
|
|
|
|
|
|
bInheritHandles = TRUE;
|
|
|
|
|
|
|
|
ZeroMemory(&StartupInfo, sizeof(STARTUPINFO));
|
|
|
|
StartupInfo.cb = sizeof(STARTUPINFO);
|
2015-08-28 11:31:22 +03:00
|
|
|
StartupInfo.hStdOutput = pipe_write;
|
|
|
|
StartupInfo.hStdError = pipe_write;
|
2016-05-25 16:47:58 +03:00
|
|
|
StartupInfo.dwFlags = STARTF_USESTDHANDLES;
|
|
|
|
|
|
|
|
ZeroMemory(&ProcessInformation, sizeof(PROCESS_INFORMATION));
|
|
|
|
|
|
|
|
if (!(lpEnvironment = calloc(1, sizeof(TESTENV_T) + sizeof(TCHAR))))
|
|
|
|
{
|
2019-11-06 17:24:51 +03:00
|
|
|
printf("Failed to allocate environment buffer. error=%" PRIu32 "\n", GetLastError());
|
2016-05-25 16:47:58 +03:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
memcpy(lpEnvironment, (void*)TESTENV_T, sizeof(TESTENV_T));
|
2015-08-28 11:31:22 +03:00
|
|
|
|
2019-11-06 17:24:51 +03:00
|
|
|
status = CreateProcess(lpApplicationName, lpCommandLine, lpProcessAttributes,
|
|
|
|
lpThreadAttributes, bInheritHandles, dwCreationFlags, lpEnvironment,
|
|
|
|
lpCurrentDirectory, &StartupInfo, &ProcessInformation);
|
2015-08-28 11:31:22 +03:00
|
|
|
|
|
|
|
free(lpEnvironment);
|
|
|
|
|
|
|
|
if (!status)
|
|
|
|
{
|
|
|
|
CloseHandle(pipe_read);
|
|
|
|
CloseHandle(pipe_write);
|
2019-11-06 17:24:51 +03:00
|
|
|
printf("CreateProcess failed. error=%" PRIu32 "\n", GetLastError());
|
2015-08-28 11:31:22 +03:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2016-05-25 16:47:58 +03:00
|
|
|
if (WaitForSingleObject(ProcessInformation.hProcess, 5000) != WAIT_OBJECT_0)
|
2015-08-28 11:31:22 +03:00
|
|
|
{
|
2019-11-06 17:24:51 +03:00
|
|
|
printf("Failed to wait for second process. error=%" PRIu32 "\n", GetLastError());
|
2016-05-25 16:47:58 +03:00
|
|
|
return 1;
|
2015-08-28 11:31:22 +03:00
|
|
|
}
|
2016-05-25 16:47:58 +03:00
|
|
|
|
|
|
|
ZeroMemory(buf, sizeof(buf));
|
2019-11-06 17:24:51 +03:00
|
|
|
ReadFile(pipe_read, buf, sizeof(buf) - 1, &read_bytes, NULL);
|
2016-05-25 16:47:58 +03:00
|
|
|
if (!strstr((const char*)buf, TESTENV_A))
|
2015-08-28 11:31:22 +03:00
|
|
|
{
|
2016-05-25 16:47:58 +03:00
|
|
|
printf("No or unexpected data read from pipe\n");
|
|
|
|
ret = 1;
|
2015-08-28 11:31:22 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
CloseHandle(pipe_read);
|
|
|
|
CloseHandle(pipe_write);
|
|
|
|
|
2013-09-25 23:47:18 +04:00
|
|
|
exitCode = 0;
|
2013-09-23 22:44:59 +04:00
|
|
|
status = GetExitCodeProcess(ProcessInformation.hProcess, &exitCode);
|
|
|
|
|
2019-11-06 17:24:51 +03:00
|
|
|
printf("GetExitCodeProcess status: %" PRId32 "\n", status);
|
|
|
|
printf("Process exited with code: 0x%08" PRIX32 "\n", exitCode);
|
2013-09-23 22:44:59 +04:00
|
|
|
|
2013-09-23 01:23:00 +04:00
|
|
|
CloseHandle(ProcessInformation.hProcess);
|
|
|
|
CloseHandle(ProcessInformation.hThread);
|
|
|
|
|
2015-08-28 11:31:22 +03:00
|
|
|
return ret;
|
2013-09-23 01:23:00 +04:00
|
|
|
}
|