Initial release
This commit is contained in:
commit
dbbf0b2f84
26
README.md
Normal file
26
README.md
Normal file
@ -0,0 +1,26 @@
|
||||
## PIJ - Run process tree in a job and wait for completion.
|
||||
|
||||
## Usage
|
||||
`pij my-program.exe arguments...`
|
||||
|
||||
## Purpose
|
||||
Use it when you want to run `my-program.exe` in a batch file, but
|
||||
my-program.exe starts another process(es) and returns before they
|
||||
complete, which is not what you want: batch file should continue only
|
||||
when all additional processes also exit. PIJ runs initial process in a
|
||||
job and waits for this job to become empty, before exiting.
|
||||
|
||||
## Result code (ERRORLEVEL)
|
||||
PIJ terminates with non-zero exit code if there's some problem during
|
||||
initial process creation or waiting phase. Zero exit code is returned
|
||||
when the last process in the job exits.
|
||||
|
||||
## Restrictions
|
||||
All arguments are passed to CreateProcess'es lpCommandLine, hence some
|
||||
restrictions on what you can start: it should be a real EXE file
|
||||
(please start cmd.exe if you want to run another batch file), standard
|
||||
directories and PATH are used, registry AppPaths aren't.
|
||||
|
||||
## Author
|
||||
Written by Anton Kovalenko <anton@sw4me.com>. My code is public domain.
|
||||
Uses some code by Raymond Chen.
|
1
build.cmd
Normal file
1
build.cmd
Normal file
@ -0,0 +1 @@
|
||||
x86_64-w64-mingw32-gcc -O2 -s pij.c pijhelp.cc -o pij.exe -DUNICODE
|
161
pij.c
Normal file
161
pij.c
Normal file
@ -0,0 +1,161 @@
|
||||
#ifndef UNICODE
|
||||
#define UNICODE
|
||||
#endif
|
||||
|
||||
#include <windows.h>
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
|
||||
// Skip initial argument in a command line. Adapted from my winserv
|
||||
// http://www.sw4me.com/wiki/Winserv
|
||||
static LPTSTR skip_arg(LPTSTR p)
|
||||
{
|
||||
int inquote, slashes;
|
||||
|
||||
// skip spaces
|
||||
while (isspace(*p)) { p++; }
|
||||
if (*p == TEXT('\0')) { return p; }
|
||||
inquote = 0; slashes = 0;
|
||||
// we found where the argument begins
|
||||
for (;;) {
|
||||
while (*p == TEXT('\\')) {
|
||||
slashes++;
|
||||
p++;
|
||||
}
|
||||
if (*p == TEXT('"')) {
|
||||
if ((slashes & 1) == 0) {
|
||||
if ((inquote) && (p[1] == TEXT('"'))) {
|
||||
p++;
|
||||
} else {
|
||||
inquote = !inquote;
|
||||
}
|
||||
}
|
||||
}
|
||||
slashes=0;
|
||||
if ((*p == TEXT('\0')) || (!inquote && isspace(*p))) {
|
||||
while (isspace(*p))
|
||||
p++;
|
||||
return p;
|
||||
}
|
||||
p++;
|
||||
}
|
||||
}
|
||||
|
||||
// Terminate unsuccessfully with an error message
|
||||
static void Fail(const char *desc) {
|
||||
fprintf(stderr, "Error %u in %s\n", GetLastError(), desc);
|
||||
exit(1);
|
||||
}
|
||||
|
||||
// Raymond Chen's CreateProcessInJob
|
||||
// see https://devblogs.microsoft.com/oldnewthing/20131209-00/?p=2433
|
||||
BOOL CreateProcessInJob(
|
||||
HANDLE hJob,
|
||||
LPCTSTR lpApplicationName,
|
||||
LPTSTR lpCommandLine,
|
||||
LPSECURITY_ATTRIBUTES lpProcessAttributes,
|
||||
LPSECURITY_ATTRIBUTES lpThreadAttributes,
|
||||
BOOL bInheritHandles,
|
||||
DWORD dwCreationFlags,
|
||||
LPVOID lpEnvironment,
|
||||
LPCTSTR lpCurrentDirectory,
|
||||
LPSTARTUPINFO lpStartupInfo,
|
||||
LPPROCESS_INFORMATION ppi)
|
||||
{
|
||||
BOOL fRc = CreateProcess(
|
||||
lpApplicationName,
|
||||
lpCommandLine,
|
||||
lpProcessAttributes,
|
||||
lpThreadAttributes,
|
||||
bInheritHandles,
|
||||
dwCreationFlags | CREATE_SUSPENDED,
|
||||
lpEnvironment,
|
||||
lpCurrentDirectory,
|
||||
lpStartupInfo,
|
||||
ppi);
|
||||
if (fRc) {
|
||||
fRc = AssignProcessToJobObject(hJob, ppi->hProcess);
|
||||
if (fRc && !(dwCreationFlags & CREATE_SUSPENDED)) {
|
||||
fRc = ResumeThread(ppi->hThread) != (DWORD)-1;
|
||||
}
|
||||
if (!fRc) {
|
||||
TerminateProcess(ppi->hProcess, 0);
|
||||
CloseHandle(ppi->hProcess);
|
||||
CloseHandle(ppi->hThread);
|
||||
ppi->hProcess = ppi->hThread = NULL;
|
||||
}
|
||||
}
|
||||
return fRc;
|
||||
}
|
||||
|
||||
// defined in pijhelp.cc, so I can use C++ raw literals
|
||||
extern const char *Help;
|
||||
|
||||
|
||||
int main(int argc, char*argv[])
|
||||
{
|
||||
if (argc <= 1) {
|
||||
printf("%s",Help);
|
||||
exit(0);
|
||||
}
|
||||
LPTSTR cmdline = GetCommandLineW();
|
||||
LPTSTR restCmdLine = skip_arg(cmdline);
|
||||
HANDLE hJob = CreateJobObject(NULL,NULL);
|
||||
if (hJob == NULL) {
|
||||
Fail("CreateJobObject");
|
||||
}
|
||||
HANDLE hPort = CreateIoCompletionPort(INVALID_HANDLE_VALUE, NULL, 0, 1);
|
||||
if (hPort == NULL) {
|
||||
Fail("CreateIoCompletionPort");
|
||||
}
|
||||
|
||||
// see https://devblogs.microsoft.com/oldnewthing/20130405-00/?p=4743
|
||||
JOBOBJECT_ASSOCIATE_COMPLETION_PORT jacp;
|
||||
jacp.CompletionKey = hJob;
|
||||
jacp.CompletionPort = hPort;
|
||||
if (!SetInformationJobObject(hJob,
|
||||
JobObjectAssociateCompletionPortInformation,
|
||||
&jacp, sizeof(jacp))) {
|
||||
Fail("SetInformationJobObject(JobObjectAssociateCompletionPortInformation)");
|
||||
}
|
||||
|
||||
STARTUPINFO si = { sizeof si };
|
||||
PROCESS_INFORMATION pi;
|
||||
if (!CreateProcessInJob(hJob,
|
||||
NULL,
|
||||
restCmdLine,
|
||||
NULL,
|
||||
NULL,
|
||||
FALSE,
|
||||
0,
|
||||
NULL,
|
||||
NULL,
|
||||
&si,
|
||||
&pi)) {
|
||||
Fail("CreateProcess");
|
||||
}
|
||||
CloseHandle(pi.hProcess);
|
||||
CloseHandle(pi.hThread);
|
||||
|
||||
|
||||
DWORD completionCode;
|
||||
ULONG_PTR completionKey;
|
||||
LPOVERLAPPED overlapped;
|
||||
|
||||
for (;;) {
|
||||
BOOL rc = GetQueuedCompletionStatus(hPort,
|
||||
&completionCode,
|
||||
&completionKey,
|
||||
&overlapped,
|
||||
INFINITE);
|
||||
if (!rc) {
|
||||
Fail("GetQueuedCompletionStatus");
|
||||
}
|
||||
if ((HANDLE)completionKey == hJob &&
|
||||
completionCode == JOB_OBJECT_MSG_ACTIVE_PROCESS_ZERO) {
|
||||
break;
|
||||
}
|
||||
}
|
||||
CloseHandle(hJob);
|
||||
return 0;
|
||||
}
|
24
pijhelp.cc
Normal file
24
pijhelp.cc
Normal file
@ -0,0 +1,24 @@
|
||||
const char *Help = R"HELP(
|
||||
PIJ - Run process tree in a job and wait for completion.
|
||||
|
||||
Usage:
|
||||
pij my-program.exe arguments...
|
||||
|
||||
Use it when you want to run my-program.exe in a batch file, but
|
||||
my-program.exe starts another process(es) and returns before they
|
||||
complete, which is not what you want: batch file should continue only
|
||||
when all additional processes also exit. PIJ runs initial process in a
|
||||
job and waits for this job to become empty, before exiting.
|
||||
|
||||
PIJ terminates with non-zero exit code if there's some problem during
|
||||
initial process creation or waiting phase. Zero exit code is returned
|
||||
when the last process in the job exits.
|
||||
|
||||
All arguments are passed to CreateProcess'es lpCommandLine, hence some
|
||||
restrictions on what you can start: it should be a real EXE file
|
||||
(please start cmd.exe if you want to run another batch file), standard
|
||||
directories and PATH are used, registry AppPaths aren't.
|
||||
|
||||
Written by Anton Kovalenko <anton@sw4me.com>. Public domain.
|
||||
)HELP";
|
||||
|
Loading…
Reference in New Issue
Block a user