summaryrefslogtreecommitdiffstats
path: root/src/kmk/w32
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--src/kmk/w32/Makefile.am26
-rw-r--r--src/kmk/w32/Makefile.kup0
-rw-r--r--src/kmk/w32/compat/Makefile.kup0
-rw-r--r--src/kmk/w32/compat/dirent.c212
-rw-r--r--src/kmk/w32/compat/posixfcn.c516
-rw-r--r--src/kmk/w32/imagecache.c219
-rw-r--r--src/kmk/w32/include/dirent.h66
-rw-r--r--src/kmk/w32/include/dlfcn.h29
-rw-r--r--src/kmk/w32/include/pathstuff.h30
-rw-r--r--src/kmk/w32/include/sub_proc.h72
-rw-r--r--src/kmk/w32/include/w32err.h26
-rw-r--r--src/kmk/w32/pathstuff.c321
-rw-r--r--src/kmk/w32/subproc/Makefile.kup0
-rw-r--r--src/kmk/w32/subproc/NMakefile60
-rw-r--r--src/kmk/w32/subproc/misc.c83
-rw-r--r--src/kmk/w32/subproc/proc.h29
-rw-r--r--src/kmk/w32/subproc/sub_proc.c1714
-rw-r--r--src/kmk/w32/subproc/w32err.c85
-rw-r--r--src/kmk/w32/tstFileInfo.c151
-rw-r--r--src/kmk/w32/w32os.c220
-rw-r--r--src/kmk/w32/winchildren.c3729
-rw-r--r--src/kmk/w32/winchildren.h115
22 files changed, 7703 insertions, 0 deletions
diff --git a/src/kmk/w32/Makefile.am b/src/kmk/w32/Makefile.am
new file mode 100644
index 0000000..5527f77
--- /dev/null
+++ b/src/kmk/w32/Makefile.am
@@ -0,0 +1,26 @@
+# Makefile.am to create libw32.a for mingw32 host.
+# Copyright (C) 1997-2016 Free Software Foundation, Inc.
+# This file is part of GNU Make.
+#
+# GNU Make 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 3 of the License, or (at your option) any later
+# version.
+#
+# GNU Make 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/>.
+
+AUTOMAKE_OPTIONS = subdir-objects
+
+noinst_LIBRARIES = libw32.a
+
+libw32_a_SOURCES = subproc/misc.c subproc/sub_proc.c subproc/w32err.c \
+ compat/posixfcn.c pathstuff.c w32os.c
+
+libw32_a_CPPFLAGS = -I$(srcdir)/include -I$(srcdir)/subproc -I$(top_srcdir) \
+ -I$(top_srcdir)/glob
diff --git a/src/kmk/w32/Makefile.kup b/src/kmk/w32/Makefile.kup
new file mode 100644
index 0000000..e69de29
--- /dev/null
+++ b/src/kmk/w32/Makefile.kup
diff --git a/src/kmk/w32/compat/Makefile.kup b/src/kmk/w32/compat/Makefile.kup
new file mode 100644
index 0000000..e69de29
--- /dev/null
+++ b/src/kmk/w32/compat/Makefile.kup
diff --git a/src/kmk/w32/compat/dirent.c b/src/kmk/w32/compat/dirent.c
new file mode 100644
index 0000000..9f992ec
--- /dev/null
+++ b/src/kmk/w32/compat/dirent.c
@@ -0,0 +1,212 @@
+/* Directory entry code for Window platforms.
+Copyright (C) 1996-2016 Free Software Foundation, Inc.
+This file is part of GNU Make.
+
+GNU Make 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 3 of the License, or (at your option) any later
+version.
+
+GNU Make 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 <config.h>
+#include <sys/types.h>
+#include <sys/stat.h>
+#include <errno.h>
+#include <string.h>
+#include <stdlib.h>
+#ifdef KMK_PRF
+# include <stdio.h>
+#endif
+#include "dirent.h"
+
+
+DIR*
+opendir(const char* pDirName)
+{
+ struct stat sb;
+ DIR* pDir;
+ char* pEndDirName;
+ int nBufferLen;
+
+ /* sanity checks */
+ if (!pDirName) {
+ errno = EINVAL;
+ return NULL;
+ }
+ if (stat(pDirName, &sb) != 0) {
+ errno = ENOENT;
+ return NULL;
+ }
+ if ((sb.st_mode & S_IFMT) != S_IFDIR) {
+ errno = ENOTDIR;
+ return NULL;
+ }
+
+ /* allocate a DIR structure to return */
+ pDir = (DIR *) malloc(sizeof (DIR));
+
+ if (!pDir)
+ return NULL;
+
+ /* input directory name length */
+ nBufferLen = strlen(pDirName);
+
+ /* copy input directory name to DIR buffer */
+ strcpy(pDir->dir_pDirectoryName, pDirName);
+
+ /* point to end of the copied directory name */
+ pEndDirName = &pDir->dir_pDirectoryName[nBufferLen - 1];
+
+ /* if directory name did not end in '/' or '\', add '/' */
+ if ((*pEndDirName != '/') && (*pEndDirName != '\\')) {
+ pEndDirName++;
+ *pEndDirName = '/';
+ }
+
+ /* now append the wildcard character to the buffer */
+ pEndDirName++;
+ *pEndDirName = '*';
+ pEndDirName++;
+ *pEndDirName = '\0';
+
+ /* other values defaulted */
+ pDir->dir_nNumFiles = 0;
+ pDir->dir_hDirHandle = INVALID_HANDLE_VALUE;
+ pDir->dir_ulCookie = __DIRENT_COOKIE;
+
+#ifdef KMK_PRF
+ fprintf(stderr, "opendir(%s) -> %p\n", pDirName, pDir);
+#endif
+ return pDir;
+}
+
+void
+closedir(DIR *pDir)
+{
+ /* got a valid pointer? */
+ if (!pDir) {
+ errno = EINVAL;
+ return;
+ }
+
+ /* sanity check that this is a DIR pointer */
+ if (pDir->dir_ulCookie != __DIRENT_COOKIE) {
+ errno = EINVAL;
+ return;
+ }
+
+ /* close the WINDOWS32 directory handle */
+ if (pDir->dir_hDirHandle != INVALID_HANDLE_VALUE)
+ FindClose(pDir->dir_hDirHandle);
+
+ free(pDir);
+
+ return;
+}
+
+struct dirent *
+readdir(DIR* pDir)
+{
+ WIN32_FIND_DATA wfdFindData;
+
+ if (!pDir) {
+ errno = EINVAL;
+ return NULL;
+ }
+
+ /* sanity check that this is a DIR pointer */
+ if (pDir->dir_ulCookie != __DIRENT_COOKIE) {
+ errno = EINVAL;
+ return NULL;
+ }
+
+ if (pDir->dir_nNumFiles == 0) {
+ pDir->dir_hDirHandle = FindFirstFile(pDir->dir_pDirectoryName, &wfdFindData);
+ if (pDir->dir_hDirHandle == INVALID_HANDLE_VALUE)
+ return NULL;
+ } else if (!FindNextFile(pDir->dir_hDirHandle, &wfdFindData))
+ return NULL;
+
+ /* bump count for next call to readdir() or telldir() */
+ pDir->dir_nNumFiles++;
+
+ /* fill in struct dirent values */
+ pDir->dir_sdReturn.d_ino = (ino_t)-1;
+ strcpy(pDir->dir_sdReturn.d_name, wfdFindData.cFileName);
+
+ return &pDir->dir_sdReturn;
+}
+
+void
+rewinddir(DIR* pDir)
+{
+ if (!pDir) {
+ errno = EINVAL;
+ return;
+ }
+
+ /* sanity check that this is a DIR pointer */
+ if (pDir->dir_ulCookie != __DIRENT_COOKIE) {
+ errno = EINVAL;
+ return;
+ }
+
+ /* close the WINDOWS32 directory handle */
+ if (pDir->dir_hDirHandle != INVALID_HANDLE_VALUE)
+ if (!FindClose(pDir->dir_hDirHandle))
+ errno = EBADF;
+
+ /* reset members which control readdir() */
+ pDir->dir_hDirHandle = INVALID_HANDLE_VALUE;
+ pDir->dir_nNumFiles = 0;
+
+ return;
+}
+
+int
+telldir(DIR* pDir)
+{
+ if (!pDir) {
+ errno = EINVAL;
+ return -1;
+ }
+
+ /* sanity check that this is a DIR pointer */
+ if (pDir->dir_ulCookie != __DIRENT_COOKIE) {
+ errno = EINVAL;
+ return -1;
+ }
+
+ /* return number of times readdir() called */
+ return pDir->dir_nNumFiles;
+}
+
+void
+seekdir(DIR* pDir, long nPosition)
+{
+ if (!pDir)
+ return;
+
+ /* sanity check that this is a DIR pointer */
+ if (pDir->dir_ulCookie != __DIRENT_COOKIE)
+ return;
+
+ /* go back to beginning of directory */
+ rewinddir(pDir);
+
+ /* loop until we have found position we care about */
+ for (--nPosition; nPosition && readdir(pDir); nPosition--);
+
+ /* flag invalid nPosition value */
+ if (nPosition)
+ errno = EINVAL;
+
+ return;
+}
diff --git a/src/kmk/w32/compat/posixfcn.c b/src/kmk/w32/compat/posixfcn.c
new file mode 100644
index 0000000..a537ca2
--- /dev/null
+++ b/src/kmk/w32/compat/posixfcn.c
@@ -0,0 +1,516 @@
+/* Replacements for Posix functions and Posix functionality for MS-Windows.
+
+Copyright (C) 2013-2016 Free Software Foundation, Inc.
+This file is part of GNU Make.
+
+GNU Make 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 3 of the License, or (at your option) any later
+version.
+
+GNU Make 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 <string.h>
+#include <io.h>
+#include <stdarg.h>
+#include <errno.h>
+#include <windows.h>
+
+#include "dlfcn.h"
+
+#include "makeint.h"
+#include "job.h"
+
+#ifndef NO_OUTPUT_SYNC
+/* Support for OUTPUT_SYNC and related functionality. */
+
+/* Emulation of fcntl that supports only F_GETFD and F_SETLKW. */
+int
+fcntl (intptr_t fd, int cmd, ...)
+{
+ va_list ap;
+
+ va_start (ap, cmd);
+
+ switch (cmd)
+ {
+ case F_GETFD:
+ va_end (ap);
+ /* Could have used GetHandleInformation, but that isn't
+ supported on Windows 9X. */
+ if (_get_osfhandle (fd) == -1)
+ return -1;
+ return 0;
+ case F_SETLKW:
+ {
+ void *buf = va_arg (ap, void *);
+ struct flock *fl = (struct flock *)buf;
+ HANDLE hmutex = (HANDLE)fd;
+ static struct flock last_fl;
+ short last_type = last_fl.l_type;
+
+ va_end (ap);
+
+ if (hmutex == INVALID_HANDLE_VALUE || !hmutex)
+ return -1;
+
+ last_fl = *fl;
+
+ switch (fl->l_type)
+ {
+
+ case F_WRLCK:
+ {
+ DWORD result;
+
+ if (last_type == F_WRLCK)
+ {
+ /* Don't call WaitForSingleObject if we already
+ own the mutex, because doing so will require
+ us to call ReleaseMutex an equal number of
+ times, before the mutex is actually
+ released. */
+ return 0;
+ }
+
+ result = WaitForSingleObject (hmutex, INFINITE);
+ switch (result)
+ {
+ case WAIT_OBJECT_0:
+ /* We don't care if the mutex owner crashed or
+ exited. */
+ case WAIT_ABANDONED:
+ return 0;
+ case WAIT_FAILED:
+ case WAIT_TIMEOUT: /* cannot happen, really */
+ {
+ DWORD err = GetLastError ();
+
+ /* Invalidate the last command. */
+ memset (&last_fl, 0, sizeof (last_fl));
+
+ switch (err)
+ {
+ case ERROR_INVALID_HANDLE:
+ case ERROR_INVALID_FUNCTION:
+ errno = EINVAL;
+ return -1;
+ default:
+ errno = EDEADLOCK;
+ return -1;
+ }
+ }
+ }
+ }
+ case F_UNLCK:
+ {
+ /* FIXME: Perhaps we should call ReleaseMutex
+ repatedly until it errors out, to make sure the
+ mutext is released even if we somehow managed to
+ to take ownership multiple times? */
+ BOOL status = ReleaseMutex (hmutex);
+
+ if (status)
+ return 0;
+ else
+ {
+ DWORD err = GetLastError ();
+
+ if (err == ERROR_NOT_OWNER)
+ errno = EPERM;
+ else
+ {
+ memset (&last_fl, 0, sizeof (last_fl));
+ errno = EINVAL;
+ }
+ return -1;
+ }
+ }
+ default:
+ errno = ENOSYS;
+ return -1;
+ }
+ }
+ default:
+ errno = ENOSYS;
+ va_end (ap);
+ return -1;
+ }
+}
+
+static intptr_t mutex_handle = -1;
+
+/* Record in a static variable the mutex handle we were requested to
+ use. That nameless mutex was created by the top-level Make, and
+ its handle was passed to us via inheritance. The value of that
+ handle is passed via the command-line arguments, so that we know
+ which handle to use. */
+void
+record_sync_mutex (const char *str)
+{
+#ifdef CONFIG_NEW_WIN_CHILDREN
+ HANDLE hmtx = OpenMutexA(SYNCHRONIZE, FALSE /*fInheritable*/, str);
+ if (hmtx)
+ mutex_handle = (intptr_t)hmtx;
+ else
+ {
+ mutex_handle = -1;
+ errno = ENOENT;
+ }
+#else
+ char *endp;
+ intptr_t hmutex = strtol (str, &endp, 16);
+
+ if (*endp == '\0')
+ mutex_handle = hmutex;
+ else
+ {
+ mutex_handle = -1;
+ errno = EINVAL;
+ }
+#endif
+}
+
+/* Create a new mutex or reuse one created by our parent. */
+intptr_t
+#ifdef CONFIG_NEW_WIN_CHILDREN
+create_mutex (char *mtxname, size_t size)
+#else
+create_mutex (void)
+#endif
+{
+#ifndef CONFIG_NEW_WIN_CHILDREN
+ SECURITY_ATTRIBUTES secattr;
+#endif
+ intptr_t hmutex = -1;
+
+ /* If we have a mutex handle passed from the parent Make, just use
+ that. */
+ if (mutex_handle > 0)
+ {
+#ifdef CONFIG_NEW_WIN_CHILDREN
+ mtxname[0] = '\0';
+#endif
+ return mutex_handle;
+ }
+
+#ifdef CONFIG_NEW_WIN_CHILDREN
+ /* We're the top-level Make. Child Make processes will open our mutex, since
+ children does not inherit any handles other than the three standard ones. */
+ snprintf(mtxname, size, "Make-output-%u-%u-%u", GetCurrentProcessId(),
+ GetCurrentThreadId(), GetTickCount());
+ hmutex = (intptr_t)CreateMutexA (NULL, FALSE /*Locked*/, mtxname);
+#else
+ /* We are the top-level Make, and we want the handle to be inherited
+ by our child processes. */
+ secattr.nLength = sizeof (secattr);
+ secattr.lpSecurityDescriptor = NULL; /* use default security descriptor */
+ secattr.bInheritHandle = TRUE;
+
+ hmutex = (intptr_t)CreateMutex (&secattr, FALSE, NULL);
+#endif
+ if (!hmutex)
+ {
+ DWORD err = GetLastError ();
+
+ fprintf (stderr, "CreateMutex: error %lu\n", err);
+ errno = ENOLCK;
+ hmutex = -1;
+ }
+
+ mutex_handle = hmutex;
+ return hmutex;
+}
+
+/* Return non-zero if F1 and F2 are 2 streams representing the same
+ file or pipe or device. */
+int
+same_stream (FILE *f1, FILE *f2)
+{
+ HANDLE fh1 = (HANDLE)_get_osfhandle (fileno (f1));
+ HANDLE fh2 = (HANDLE)_get_osfhandle (fileno (f2));
+
+ /* Invalid file descriptors get treated as different streams. */
+ if (fh1 && fh1 != INVALID_HANDLE_VALUE
+ && fh2 && fh2 != INVALID_HANDLE_VALUE)
+ {
+ if (fh1 == fh2)
+ return 1;
+ else
+ {
+ DWORD ftyp1 = GetFileType (fh1), ftyp2 = GetFileType (fh2);
+
+ if (ftyp1 != ftyp2
+ || ftyp1 == FILE_TYPE_UNKNOWN || ftyp2 == FILE_TYPE_UNKNOWN)
+ return 0;
+ else if (ftyp1 == FILE_TYPE_CHAR)
+ {
+ /* For character devices, check if they both refer to a
+ console. This loses if both handles refer to the
+ null device (FIXME!), but in that case we don't care
+ in the context of Make. */
+ DWORD conmode1, conmode2;
+
+ /* Each process on Windows can have at most 1 console,
+ so if both handles are for the console device, they
+ are the same. We also compare the console mode to
+ distinguish between stdin and stdout/stderr. */
+ if (GetConsoleMode (fh1, &conmode1)
+ && GetConsoleMode (fh2, &conmode2)
+ && conmode1 == conmode2)
+ return 1;
+ }
+ else
+ {
+ /* For disk files and pipes, compare their unique
+ attributes. */
+ BY_HANDLE_FILE_INFORMATION bhfi1, bhfi2;
+
+ /* Pipes get zero in the volume serial number, but do
+ appear to have meaningful information in file index
+ attributes. We test file attributes as well, for a
+ good measure. */
+ if (GetFileInformationByHandle (fh1, &bhfi1)
+ && GetFileInformationByHandle (fh2, &bhfi2))
+ return (bhfi1.dwVolumeSerialNumber == bhfi2.dwVolumeSerialNumber
+ && bhfi1.nFileIndexLow == bhfi2.nFileIndexLow
+ && bhfi1.nFileIndexHigh == bhfi2.nFileIndexHigh
+ && bhfi1.dwFileAttributes == bhfi2.dwFileAttributes);
+ }
+ }
+ }
+ return 0;
+}
+
+/* A replacement for tmpfile, since the MSVCRT implementation creates
+ the file in the root directory of the current drive, which might
+ not be writable by our user. Most of the code borrowed from
+ create_batch_file, see job.c. */
+FILE *
+tmpfile (void)
+{
+ char temp_path[MAXPATHLEN];
+ unsigned path_size = GetTempPath (sizeof temp_path, temp_path);
+ int path_is_dot = 0;
+ /* The following variable is static so we won't try to reuse a name
+ that was generated a little while ago, because that file might
+ not be on disk yet, since we use FILE_ATTRIBUTE_TEMPORARY below,
+ which tells the OS it doesn't need to flush the cache to disk.
+ If the file is not yet on disk, we might think the name is
+ available, while it really isn't. This happens in parallel
+ builds, where Make doesn't wait for one job to finish before it
+ launches the next one. */
+ static unsigned uniq = 0;
+ static int second_loop = 0;
+ const char base[] = "gmake_tmpf";
+ const unsigned sizemax = sizeof base - 1 + 4 + 10 + 10;
+ unsigned pid = GetCurrentProcessId ();
+
+ if (path_size == 0)
+ {
+ path_size = GetCurrentDirectory (sizeof temp_path, temp_path);
+ path_is_dot = 1;
+ }
+
+ ++uniq;
+ if (uniq >= 0x10000 && !second_loop)
+ {
+ /* If we already had 64K batch files in this
+ process, make a second loop through the numbers,
+ looking for free slots, i.e. files that were
+ deleted in the meantime. */
+ second_loop = 1;
+ uniq = 1;
+ }
+ while (path_size > 0 &&
+ path_size + sizemax < sizeof temp_path &&
+ !(uniq >= 0x10000 && second_loop))
+ {
+ HANDLE h;
+
+ sprintf (temp_path + path_size,
+ "%s%s%u-%x.tmp",
+ temp_path[path_size - 1] == '\\' ? "" : "\\",
+ base, pid, uniq);
+ h = CreateFile (temp_path, /* file name */
+ GENERIC_READ | GENERIC_WRITE | DELETE, /* desired access */
+ FILE_SHARE_READ | FILE_SHARE_WRITE, /* share mode */
+ NULL, /* default security attributes */
+ CREATE_NEW, /* creation disposition */
+ FILE_ATTRIBUTE_NORMAL | /* flags and attributes */
+ FILE_ATTRIBUTE_TEMPORARY |
+ FILE_FLAG_DELETE_ON_CLOSE,
+ NULL); /* no template file */
+
+ if (h == INVALID_HANDLE_VALUE)
+ {
+ const DWORD er = GetLastError ();
+
+ if (er == ERROR_FILE_EXISTS || er == ERROR_ALREADY_EXISTS)
+ {
+ ++uniq;
+ if (uniq == 0x10000 && !second_loop)
+ {
+ second_loop = 1;
+ uniq = 1;
+ }
+ }
+
+ /* The temporary path is not guaranteed to exist, or might
+ not be writable by user. Use the current directory as
+ fallback. */
+ else if (path_is_dot == 0)
+ {
+ path_size = GetCurrentDirectory (sizeof temp_path, temp_path);
+ path_is_dot = 1;
+ }
+
+ else
+ {
+ errno = EACCES;
+ break;
+ }
+ }
+ else
+ {
+ int fd = _open_osfhandle ((intptr_t)h, 0);
+
+ return _fdopen (fd, "w+b");
+ }
+ }
+
+ if (uniq >= 0x10000)
+ errno = EEXIST;
+ return NULL;
+}
+
+#endif /* !NO_OUTPUT_SYNC */
+
+#if MAKE_LOAD
+
+/* Support for dynamic loading of objects. */
+
+
+static DWORD last_err;
+
+void *
+dlopen (const char *file, int mode)
+{
+ char dllfn[MAX_PATH], *p;
+ HANDLE dllhandle;
+
+ if ((mode & ~(RTLD_LAZY | RTLD_NOW | RTLD_GLOBAL)) != 0)
+ {
+ errno = EINVAL;
+ last_err = ERROR_INVALID_PARAMETER;
+ return NULL;
+ }
+
+ if (!file)
+ dllhandle = GetModuleHandle (NULL);
+ else
+ {
+ /* MSDN says to be sure to use backslashes in the DLL file name. */
+ strcpy (dllfn, file);
+ for (p = dllfn; *p; p++)
+ if (*p == '/')
+ *p = '\\';
+
+ dllhandle = LoadLibrary (dllfn);
+ }
+ if (!dllhandle)
+ last_err = GetLastError ();
+
+ return dllhandle;
+}
+
+char *
+dlerror (void)
+{
+ static char errbuf[1024];
+ DWORD ret;
+
+ if (!last_err)
+ return NULL;
+
+ ret = FormatMessage (FORMAT_MESSAGE_FROM_SYSTEM
+ | FORMAT_MESSAGE_IGNORE_INSERTS,
+ NULL, last_err, 0, errbuf, sizeof (errbuf), NULL);
+ while (ret > 0 && (errbuf[ret - 1] == '\n' || errbuf[ret - 1] == '\r'))
+ --ret;
+
+ errbuf[ret] = '\0';
+ if (!ret)
+ sprintf (errbuf, "Error code %lu", last_err);
+
+ last_err = 0;
+ return errbuf;
+}
+
+void *
+dlsym (void *handle, const char *name)
+{
+ FARPROC addr = NULL;
+
+ if (!handle || handle == INVALID_HANDLE_VALUE)
+ {
+ last_err = ERROR_INVALID_PARAMETER;
+ return NULL;
+ }
+
+ addr = GetProcAddress (handle, name);
+ if (!addr)
+ last_err = GetLastError ();
+
+ return (void *)addr;
+}
+
+int
+dlclose (void *handle)
+{
+ if (!handle || handle == INVALID_HANDLE_VALUE)
+ return -1;
+ if (!FreeLibrary (handle))
+ return -1;
+
+ return 0;
+}
+
+
+#endif /* MAKE_LOAD */
+
+
+/* MS runtime's isatty returns non-zero for any character device,
+ including the null device, which is not what we want. */
+int
+isatty (int fd)
+{
+ HANDLE fh = (HANDLE) _get_osfhandle (fd);
+ DWORD con_mode;
+
+ if (fh == INVALID_HANDLE_VALUE)
+ {
+ errno = EBADF;
+ return 0;
+ }
+ if (GetConsoleMode (fh, &con_mode))
+ return 1;
+
+ errno = ENOTTY;
+ return 0;
+}
+
+char *
+ttyname (int fd)
+{
+ /* This "knows" that Make only asks about stdout and stderr. A more
+ sophisticated implementation should test whether FD is open for
+ input or output. We can do that by looking at the mode returned
+ by GetConsoleMode. */
+ return "CONOUT$";
+}
diff --git a/src/kmk/w32/imagecache.c b/src/kmk/w32/imagecache.c
new file mode 100644
index 0000000..2b26dac
--- /dev/null
+++ b/src/kmk/w32/imagecache.c
@@ -0,0 +1,219 @@
+/* $Id: imagecache.c 3195 2018-03-27 18:09:23Z bird $ */
+/** @file
+ * kBuild specific executable image cache for Windows.
+ */
+
+/*
+ * Copyright (c) 2012 knut st. osmundsen <bird-kBuild-spamx@anduin.net>
+ *
+ * This file is part of kBuild.
+ *
+ * kBuild 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 3 of the License, or
+ * (at your option) any later version.
+ *
+ * kBuild 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 kBuild. If not, see <http://www.gnu.org/licenses/>
+ *
+ */
+
+/* No GNU coding style here! */
+
+/*********************************************************************************************************************************
+* Header Files *
+*********************************************************************************************************************************/
+#include "makeint.h"
+
+#include <Windows.h>
+
+
+/*********************************************************************************************************************************
+* Structures and Typedefs *
+*********************************************************************************************************************************/
+typedef struct EXECCACHEENTRY
+{
+ /** The name hash value. */
+ unsigned uHash;
+ /** The name length. */
+ unsigned cwcName;
+ /** Pointer to the next name with the same hash. */
+ struct EXECCACHEENTRY *pNext;
+ /** When it was last referenced. */
+ unsigned uLastRef;
+ /** The module handle, LOAD_LIBRARY_AS_DATAFILE. */
+ HMODULE hmod1;
+ /** The module handle, DONT_RESOLVE_DLL_REFERENCES. */
+ HMODULE hmod2;
+ /** The executable path. */
+ wchar_t wszName[1];
+} EXECCACHEENTRY;
+typedef EXECCACHEENTRY *PEXECCACHEENTRY;
+
+
+/*********************************************************************************************************************************
+* Global Variables *
+*********************************************************************************************************************************/
+/** Critical section serializing all access. */
+static CRITICAL_SECTION g_CritSect;
+/** Set if initialized. */
+static int volatile g_fInitialized = 0;
+/** The number of cached images. */
+static unsigned g_cCached;
+/** Used noting when entries was last used.
+ * Increased on each kmk_cache_exec_image call. */
+static unsigned g_uNow;
+
+/** The size of the hash table. */
+#define EXECCACHE_HASHTAB_SIZE 128
+/** The hash table. */
+static PEXECCACHEENTRY g_apHashTab[EXECCACHE_HASHTAB_SIZE];
+
+
+/** A sleepy approach to do-once. */
+static void kmk_cache_lazy_init(void)
+{
+ if (_InterlockedCompareExchange(&g_fInitialized, -1, 0) == 0)
+ {
+ InitializeCriticalSection(&g_CritSect);
+ _InterlockedExchange(&g_fInitialized, 1);
+ }
+ else
+ while (g_fInitialized != 1)
+ Sleep(1);
+}
+
+
+/* sdbm:
+ This algorithm was created for sdbm (a public-domain reimplementation of
+ ndbm) database library. it was found to do well in scrambling bits,
+ causing better distribution of the keys and fewer splits. it also happens
+ to be a good general hashing function with good distribution. the actual
+ function is hash(i) = hash(i - 1) * 65599 + str[i]; what is included below
+ is the faster version used in gawk. [there is even a faster, duff-device
+ version] the magic constant 65599 was picked out of thin air while
+ experimenting with different constants, and turns out to be a prime.
+ this is one of the algorithms used in berkeley db (see sleepycat) and
+ elsewhere. */
+
+static unsigned execcache_calc_hash(const wchar_t *pwsz, size_t *pcch)
+{
+ wchar_t const * const pwszStart = pwsz;
+ unsigned hash = 0;
+ int ch;
+
+ while ((ch = *pwsz++) != L'\0')
+ hash = ch + (hash << 6) + (hash << 16) - hash;
+
+ *pcch = (size_t)(pwsz - pwszStart - 1);
+ return hash;
+}
+
+/**
+ * Caches two memory mappings of the specified image so that it isn't flushed
+ * from the kernel's cache mananger.
+ *
+ * Not sure exactly how much this actually helps, but whatever...
+ *
+ * @param pwszExec The executable.
+ */
+extern void kmk_cache_exec_image_w(const wchar_t *pwszExec)
+{
+ /*
+ * Prepare name lookup and to lazy init.
+ */
+ size_t cwcName;
+ const unsigned uHash = execcache_calc_hash(pwszExec, &cwcName);
+ PEXECCACHEENTRY *ppCur = &g_apHashTab[uHash % EXECCACHE_HASHTAB_SIZE];
+ PEXECCACHEENTRY pCur;
+
+ if (g_fInitialized != 1)
+ kmk_cache_lazy_init();
+
+ /*
+ * Do the lookup.
+ */
+ EnterCriticalSection(&g_CritSect);
+ pCur = *ppCur;
+ while (pCur)
+ {
+ if ( pCur->uHash == uHash
+ && pCur->cwcName == cwcName
+ && !memcmp(pCur->wszName, pwszExec, cwcName * sizeof(wchar_t)))
+ {
+ pCur->uLastRef = ++g_uNow;
+ LeaveCriticalSection(&g_CritSect);
+ return;
+ }
+ ppCur = &pCur->pNext;
+ pCur = pCur->pNext;
+ }
+ LeaveCriticalSection(&g_CritSect);
+
+ /*
+ * Not found, create a new entry.
+ */
+ pCur = xmalloc(sizeof(*pCur) + cwcName * sizeof(wchar_t));
+ pCur->uHash = uHash;
+ pCur->cwcName = (unsigned)cwcName;
+ pCur->pNext = NULL;
+ pCur->uLastRef = ++g_uNow;
+ memcpy(pCur->wszName, pwszExec, (cwcName + 1) * sizeof(wchar_t));
+ pCur->hmod1 = LoadLibraryExW(pwszExec, NULL, LOAD_LIBRARY_AS_DATAFILE);
+ if (pCur->hmod1 != NULL)
+ pCur->hmod2 = LoadLibraryExW(pwszExec, NULL, DONT_RESOLVE_DLL_REFERENCES);
+ else
+ pCur->hmod2 = NULL;
+
+ /*
+ * Insert it.
+ * Take into account that we might've been racing other threads,
+ * fortunately we don't evict anything from the cache.
+ */
+ EnterCriticalSection(&g_CritSect);
+ if (*ppCur != NULL)
+ {
+ /* Find new end of chain and check for duplicate. */
+ PEXECCACHEENTRY pCur2 = *ppCur;
+ while (pCur2)
+ {
+ if ( pCur->uHash == uHash
+ && pCur->cwcName == cwcName
+ && !memcmp(pCur->wszName, pwszExec, cwcName * sizeof(wchar_t)))
+ break;
+ ppCur = &pCur->pNext;
+ pCur = pCur->pNext;
+ }
+
+ }
+ if (*ppCur == NULL)
+ {
+ *ppCur = pCur;
+ g_cCached++;
+ LeaveCriticalSection(&g_CritSect);
+ }
+ else
+ {
+ LeaveCriticalSection(&g_CritSect);
+
+ if (pCur->hmod1 != NULL)
+ FreeLibrary(pCur->hmod1);
+ if (pCur->hmod2 != NULL)
+ FreeLibrary(pCur->hmod2);
+ free(pCur);
+ }
+}
+
+extern void kmk_cache_exec_image_a(const char *pszExec)
+{
+ wchar_t wszExec[260];
+ int cwc = MultiByteToWideChar(CP_ACP, 0 /*fFlags*/, pszExec, strlen(pszExec) + 1, wszExec, 260);
+ if (cwc > 0)
+ kmk_cache_exec_image_w(wszExec);
+}
+
diff --git a/src/kmk/w32/include/dirent.h b/src/kmk/w32/include/dirent.h
new file mode 100644
index 0000000..c349e85
--- /dev/null
+++ b/src/kmk/w32/include/dirent.h
@@ -0,0 +1,66 @@
+/* Windows version of dirent.h
+Copyright (C) 1996-2016 Free Software Foundation, Inc.
+This file is part of GNU Make.
+
+GNU Make 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 3 of the License, or (at your option) any later
+version.
+
+GNU Make 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/>. */
+
+#ifndef _DIRENT_H
+#define _DIRENT_H
+
+#ifdef KMK
+# include <windows.h>
+# include "nt/ntdir.h"
+
+#else /* !KMK */
+
+#ifdef __MINGW32__
+# include <windows.h>
+# include_next <dirent.h>
+#else
+
+#include <stdlib.h>
+#include <windows.h>
+#include <limits.h>
+#include <sys/types.h>
+
+#ifndef NAME_MAX
+#define NAME_MAX 255
+#endif
+
+#define __DIRENT_COOKIE 0xfefeabab
+
+
+struct dirent
+{
+ ino_t d_ino; /* unused - no equivalent on WINDOWS32 */
+ char d_name[NAME_MAX+1];
+};
+
+typedef struct dir_struct {
+ ULONG dir_ulCookie;
+ HANDLE dir_hDirHandle;
+ DWORD dir_nNumFiles;
+ char dir_pDirectoryName[NAME_MAX+1];
+ struct dirent dir_sdReturn;
+} DIR;
+
+DIR *opendir(const char *);
+struct dirent *readdir(DIR *);
+void rewinddir(DIR *);
+void closedir(DIR *);
+int telldir(DIR *);
+void seekdir(DIR *, long);
+
+#endif /* !__MINGW32__ */
+#endif /* !KMK */
+#endif
diff --git a/src/kmk/w32/include/dlfcn.h b/src/kmk/w32/include/dlfcn.h
new file mode 100644
index 0000000..5a2ae28
--- /dev/null
+++ b/src/kmk/w32/include/dlfcn.h
@@ -0,0 +1,29 @@
+/* dlfcn.h replacement for MS-Windows build.
+Copyright (C) 2013-2016 Free Software Foundation, Inc.
+This file is part of GNU Make.
+
+GNU Make 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 3 of the License, or (at your option) any later
+version.
+
+GNU Make 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/>. */
+
+#ifndef DLFCN_H
+#define DLFCN_H
+
+#define RTLD_LAZY 1
+#define RTLD_NOW 2
+#define RTLD_GLOBAL 4
+
+extern void *dlopen (const char *, int);
+extern void *dlsym (void *, const char *);
+extern char *dlerror (void);
+extern int dlclose (void *);
+
+#endif /* DLFCN_H */
diff --git a/src/kmk/w32/include/pathstuff.h b/src/kmk/w32/include/pathstuff.h
new file mode 100644
index 0000000..3f839ec
--- /dev/null
+++ b/src/kmk/w32/include/pathstuff.h
@@ -0,0 +1,30 @@
+/* Definitions for Windows path manipulation.
+Copyright (C) 1996-2016 Free Software Foundation, Inc.
+This file is part of GNU Make.
+
+GNU Make 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 3 of the License, or (at your option) any later
+version.
+
+GNU Make 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/>. */
+
+#ifndef _PATHSTUFF_H
+#define _PATHSTUFF_H
+
+char *convert_Path_to_windows32(char *Path, char to_delim);
+char *convert_vpath_to_windows32(char *Path, char to_delim);
+#if 1
+char *unix_slashes(char *filename); /* bird */
+char *unix_slashes_resolved(const char *src, char *dst, unsigned len); /* bird */
+#else
+char *w32ify(const char *filename, int resolve);
+#endif
+char *getcwd_fs(char *buf, int len);
+
+#endif
diff --git a/src/kmk/w32/include/sub_proc.h b/src/kmk/w32/include/sub_proc.h
new file mode 100644
index 0000000..51e3830
--- /dev/null
+++ b/src/kmk/w32/include/sub_proc.h
@@ -0,0 +1,72 @@
+/* Definitions for Windows process invocation.
+Copyright (C) 1996-2016 Free Software Foundation, Inc.
+This file is part of GNU Make.
+
+GNU Make 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 3 of the License, or (at your option) any later
+version.
+
+GNU Make 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/>. */
+
+#ifndef SUB_PROC_H
+#define SUB_PROC_H
+#ifdef CONFIG_NEW_WIN_CHILDREN
+# error "Just checking..."
+#endif
+
+/*
+ * Component Name:
+ *
+ * $Date$
+ *
+ * $Source$
+ *
+ * $Id$
+ */
+
+#define EXTERN_DECL(entry, args) extern entry args
+#define VOID_DECL void
+
+EXTERN_DECL(HANDLE process_init, (VOID_DECL));
+EXTERN_DECL(HANDLE process_init_fd, (HANDLE stdinh, HANDLE stdouth,
+ HANDLE stderrh));
+EXTERN_DECL(long process_begin, (HANDLE proc, char **argv, char **envp,
+ char *exec_path, char *as_user));
+EXTERN_DECL(long process_pipe_io, (HANDLE proc, char *stdin_data,
+ int stdin_data_len));
+#ifndef KMK /* unused */
+EXTERN_DECL(long process_file_io, (HANDLE proc));
+#endif
+EXTERN_DECL(void process_cleanup, (HANDLE proc));
+EXTERN_DECL(HANDLE process_wait_for_any, (int block, DWORD* pdwWaitStatus));
+EXTERN_DECL(void process_register, (HANDLE proc));
+EXTERN_DECL(HANDLE process_easy, (char** argv, char** env,
+ int outfd, int errfd));
+EXTERN_DECL(BOOL process_kill, (HANDLE proc, int signal));
+EXTERN_DECL(int process_used_slots, (VOID_DECL));
+EXTERN_DECL(DWORD process_set_handles, (HANDLE *handles));
+
+#ifdef KMK
+EXTERN_DECL(int process_kmk_register_submit, (HANDLE hEvent, intptr_t clue, pid_t *pPid));
+EXTERN_DECL(int process_kmk_register_redirect, (HANDLE hProcess, pid_t *pPid));
+#endif
+
+/* support routines */
+EXTERN_DECL(long process_errno, (HANDLE proc));
+EXTERN_DECL(long process_last_err, (HANDLE proc));
+EXTERN_DECL(long process_exit_code, (HANDLE proc));
+EXTERN_DECL(long process_signal, (HANDLE proc));
+EXTERN_DECL(char * process_outbuf, (HANDLE proc));
+EXTERN_DECL(char * process_errbuf, (HANDLE proc));
+EXTERN_DECL(int process_outcnt, (HANDLE proc));
+EXTERN_DECL(int process_errcnt, (HANDLE proc));
+EXTERN_DECL(void process_pipes, (HANDLE proc, int pipes[3]));
+EXTERN_DECL(void process_noinherit, (int fildes));
+
+#endif
diff --git a/src/kmk/w32/include/w32err.h b/src/kmk/w32/include/w32err.h
new file mode 100644
index 0000000..b4292f2
--- /dev/null
+++ b/src/kmk/w32/include/w32err.h
@@ -0,0 +1,26 @@
+/* Definitions for Windows error handling.
+Copyright (C) 1996-2016 Free Software Foundation, Inc.
+This file is part of GNU Make.
+
+GNU Make 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 3 of the License, or (at your option) any later
+version.
+
+GNU Make 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/>. */
+
+#ifndef _W32ERR_H_
+#define _W32ERR_H_
+
+#ifndef EXTERN_DECL
+#define EXTERN_DECL(entry, args) entry args
+#endif
+
+EXTERN_DECL(const char * map_windows32_error_to_string, (DWORD error));
+
+#endif /* !_W32ERR_H */
diff --git a/src/kmk/w32/pathstuff.c b/src/kmk/w32/pathstuff.c
new file mode 100644
index 0000000..f80cbf1
--- /dev/null
+++ b/src/kmk/w32/pathstuff.c
@@ -0,0 +1,321 @@
+/* Path conversion for Windows pathnames.
+Copyright (C) 1996-2016 Free Software Foundation, Inc.
+This file is part of GNU Make.
+
+GNU Make 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 3 of the License, or (at your option) any later
+version.
+
+GNU Make 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 "makeint.h"
+#include <string.h>
+#include <stdlib.h>
+#include "pathstuff.h"
+#if 1 /* bird */
+# include "nt_fullpath.h"
+# include <assert.h>
+#endif
+
+/*
+ * Convert delimiter separated vpath to Canonical format.
+ */
+char *
+convert_vpath_to_windows32(char *Path, char to_delim)
+{
+ char *etok; /* token separator for old Path */
+
+ /*
+ * Convert all spaces to delimiters. Note that pathnames which
+ * contain blanks get trounced here. Use 8.3 format as a workaround.
+ */
+ for (etok = Path; etok && *etok; etok++)
+ if (ISBLANK ((unsigned char) *etok))
+ *etok = to_delim;
+
+ return (convert_Path_to_windows32(Path, to_delim));
+}
+
+/*
+ * Convert delimiter separated path to Canonical format.
+ */
+char *
+convert_Path_to_windows32(char *Path, char to_delim)
+{
+ char *etok; /* token separator for old Path */
+ char *p; /* points to element of old Path */
+
+ /* is this a multi-element Path ? */
+ /* FIXME: Perhaps use ":;\"" in strpbrk to convert all quotes to
+ delimiters as well, as a way to handle quoted directories in
+ PATH? */
+ for (p = Path, etok = strpbrk(p, ":;");
+ etok;
+ etok = strpbrk(p, ":;"))
+ if ((etok - p) == 1) {
+ if (*(etok - 1) == ';' ||
+ *(etok - 1) == ':') {
+ etok[-1] = to_delim;
+ etok[0] = to_delim;
+ p = ++etok;
+ continue; /* ignore empty bucket */
+ } else if (!isalpha ((unsigned char) *p)) {
+ /* found one to count, handle things like '.' */
+ *etok = to_delim;
+ p = ++etok;
+ } else if ((*etok == ':') && (etok = strpbrk(etok+1, ":;"))) {
+ /* found one to count, handle drive letter */
+ *etok = to_delim;
+ p = ++etok;
+ } else
+ /* all finished, force abort */
+ p += strlen(p);
+ } else if (*p == '"') { /* a quoted directory */
+ for (p++; *p && *p != '"'; p++) /* skip quoted part */
+ ;
+ etok = strpbrk(p, ":;"); /* find next delimiter */
+ if (etok) {
+ *etok = to_delim;
+ p = ++etok;
+ } else
+ p += strlen(p);
+ } else {
+ /* found another one, no drive letter */
+ *etok = to_delim;
+ p = ++etok;
+ }
+
+ return Path;
+}
+
+/*
+ * Convert to forward slashes directly (w32ify(filename, 0)).
+ */
+char *unix_slashes(char *filename) /* bird */
+{
+ char *slash = filename ;
+ while ((slash = strchr(slash, '\\')) != NULL)
+ *slash++ = '/';
+ return filename;
+}
+
+/*
+ * Resolve and convert to forward slashes directly (w32ify(filename, 1)).
+ * Returns if out of buffer space.
+ */
+char *unix_slashes_resolved(const char *src, char *dst, unsigned len)
+{
+ assert(len >= FILENAME_MAX);
+ *dst = '\0'; /** @todo nt_fullpath_cached needs to return some indication of overflow. */
+#if 1
+ nt_fullpath_cached(src, dst, len);
+#else
+ _fullpath(dst, src, len);
+#endif
+
+ return unix_slashes(dst);
+}
+
+#if 0 /* bird: replaced by unix_slashes and unix_slahes_resolved. */
+/*
+ * Convert to forward slashes. Resolve to full pathname optionally
+ */
+char *
+w32ify(const char *filename, int resolve)
+{
+ static char w32_path[FILENAME_MAX];
+#if 1 /* bird */
+
+ if (resolve) {
+ nt_fullpath_cached(filename, w32_path, sizeof(w32_path));
+ } else {
+ w32_path[0] = '\0';
+ strncat(w32_path, filename, sizeof(w32_path));
+ }
+ return unix_slashes(w32_path);
+
+#else /* !bird */
+ char *p;
+
+ if (resolve) {
+ _fullpath(w32_path, filename, sizeof (w32_path));
+ } else
+ strncpy(w32_path, filename, sizeof (w32_path));
+
+ for (p = w32_path; p && *p; p++)
+ if (*p == '\\')
+ *p = '/';
+
+ return w32_path;
+#endif /* !bird */
+}
+#endif
+
+char *
+getcwd_fs(char* buf, int len)
+{
+ char *p = getcwd(buf, len);
+
+ if (p) {
+#if 1
+ p = unix_slashes(p);
+#else
+ char *q = w32ify(buf, 0);
+#if 1 /* bird - UPSTREAM? */
+ buf[0] = '\0';
+ strncat(buf, q, len);
+#else /* !bird */
+ strncpy(buf, q, len);
+#endif
+#endif
+ }
+
+ return p;
+}
+
+#ifdef unused
+/*
+ * Convert delimiter separated pathnames (e.g. PATH) or single file pathname
+ * (e.g. c:/foo, c:\bar) to NutC format. If we are handed a string that
+ * _NutPathToNutc() fails to convert, just return the path we were handed
+ * and assume the caller will know what to do with it (It was probably
+ * a mistake to try and convert it anyway due to some of the bizarre things
+ * that might look like pathnames in makefiles).
+ */
+char *
+convert_path_to_nutc(char *path)
+{
+ int count; /* count of path elements */
+ char *nutc_path; /* new NutC path */
+ int nutc_path_len; /* length of buffer to allocate for new path */
+ char *pathp; /* pointer to nutc_path used to build it */
+ char *etok; /* token separator for old path */
+ char *p; /* points to element of old path */
+ char sep; /* what flavor of separator used in old path */
+ char *rval;
+
+ /* is this a multi-element path ? */
+ for (p = path, etok = strpbrk(p, ":;"), count = 0;
+ etok;
+ etok = strpbrk(p, ":;"))
+ if ((etok - p) == 1) {
+ if (*(etok - 1) == ';' ||
+ *(etok - 1) == ':') {
+ p = ++etok;
+ continue; /* ignore empty bucket */
+ } else if (etok = strpbrk(etok+1, ":;"))
+ /* found one to count, handle drive letter */
+ p = ++etok, count++;
+ else
+ /* all finished, force abort */
+ p += strlen(p);
+ } else
+ /* found another one, no drive letter */
+ p = ++etok, count++;
+
+ if (count) {
+ count++; /* x1;x2;x3 <- need to count x3 */
+
+ /*
+ * Hazard a guess on how big the buffer needs to be.
+ * We have to convert things like c:/foo to /c=/foo.
+ */
+ nutc_path_len = strlen(path) + (count*2) + 1;
+ nutc_path = xmalloc(nutc_path_len);
+ pathp = nutc_path;
+ *pathp = '\0';
+
+ /*
+ * Loop through PATH and convert one elemnt of the path at at
+ * a time. Single file pathnames will fail this and fall
+ * to the logic below loop.
+ */
+ for (p = path, etok = strpbrk(p, ":;");
+ etok;
+ etok = strpbrk(p, ":;")) {
+
+ /* don't trip up on device specifiers or empty path slots */
+ if ((etok - p) == 1)
+ if (*(etok - 1) == ';' ||
+ *(etok - 1) == ':') {
+ p = ++etok;
+ continue;
+ } else if ((etok = strpbrk(etok+1, ":;")) == NULL)
+ break; /* thing found was a WINDOWS32 pathname */
+
+ /* save separator */
+ sep = *etok;
+
+ /* terminate the current path element -- temporarily */
+ *etok = '\0';
+
+#ifdef __NUTC__
+ /* convert to NutC format */
+ if (_NutPathToNutc(p, pathp, 0) == FALSE) {
+ free(nutc_path);
+ rval = savestring(path, strlen(path));
+ return rval;
+ }
+#else
+ *pathp++ = '/';
+ *pathp++ = p[0];
+ *pathp++ = '=';
+ *pathp++ = '/';
+ strcpy(pathp, &p[2]);
+#endif
+
+ pathp += strlen(pathp);
+ *pathp++ = ':'; /* use Unix style path separtor for new path */
+ *pathp = '\0'; /* make sure we are null terminaed */
+
+ /* restore path separator */
+ *etok = sep;
+
+ /* point p to first char of next path element */
+ p = ++etok;
+
+ }
+ } else {
+ nutc_path_len = strlen(path) + 3;
+ nutc_path = xmalloc(nutc_path_len);
+ pathp = nutc_path;
+ *pathp = '\0';
+ p = path;
+ }
+
+ /*
+ * OK, here we handle the last element in PATH (e.g. c of a;b;c)
+ * or the path was a single filename and will be converted
+ * here. Note, testing p here assures that we don't trip up
+ * on paths like a;b; which have trailing delimiter followed by
+ * nothing.
+ */
+ if (*p != '\0') {
+#ifdef __NUTC__
+ if (_NutPathToNutc(p, pathp, 0) == FALSE) {
+ free(nutc_path);
+ rval = savestring(path, strlen(path));
+ return rval;
+ }
+#else
+ *pathp++ = '/';
+ *pathp++ = p[0];
+ *pathp++ = '=';
+ *pathp++ = '/';
+ strcpy(pathp, &p[2]);
+#endif
+ } else
+ *(pathp-1) = '\0'; /* we're already done, don't leave trailing : */
+
+ rval = savestring(nutc_path, strlen(nutc_path));
+ free(nutc_path);
+ return rval;
+}
+
+#endif
diff --git a/src/kmk/w32/subproc/Makefile.kup b/src/kmk/w32/subproc/Makefile.kup
new file mode 100644
index 0000000..e69de29
--- /dev/null
+++ b/src/kmk/w32/subproc/Makefile.kup
diff --git a/src/kmk/w32/subproc/NMakefile b/src/kmk/w32/subproc/NMakefile
new file mode 100644
index 0000000..325e55c
--- /dev/null
+++ b/src/kmk/w32/subproc/NMakefile
@@ -0,0 +1,60 @@
+# NOTE: If you have no 'make' program at all to process this makefile, run
+# 'build.bat' instead.
+#
+# Copyright (C) 1996-2016 Free Software Foundation, Inc.
+# This file is part of GNU Make.
+#
+# GNU Make 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 3 of the License, or (at your option) any later
+# version.
+#
+# GNU Make 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/>.
+
+#
+# NMakefile for GNU Make (subproc library)
+#
+LIB = lib
+CC = cl
+MAKE = nmake
+
+OUTDIR=.
+MAKEFILE=NMakefile
+
+CFLAGS_any = /nologo /MT /W4 /GX /Z7 /YX /D WIN32 /D WINDOWS32 /D _WINDOWS -I. -I../include -I../../
+CFLAGS_debug = $(CFLAGS_any) /Od /D _DEBUG /FR.\WinDebug\ /Fp.\WinDebug\subproc.pch /Fo.\WinDebug/
+CFLAGS_release = $(CFLAGS_any) /O2 /FR.\WinRel\ /Fp.\WinRel\subproc.pch /Fo.\WinRel/
+
+all: Release Debug
+
+Release:
+ $(MAKE) /f $(MAKEFILE) OUTDIR=WinRel CFLAGS="$(CFLAGS_release)" WinRel/subproc.lib
+Debug:
+ $(MAKE) /f $(MAKEFILE) OUTDIR=WinDebug CFLAGS="$(CFLAGS_debug)" WinDebug/subproc.lib
+
+clean:
+ rmdir /s /q WinRel WinDebug
+ erase *.pdb
+
+$(OUTDIR):
+ if not exist .\$@\nul mkdir .\$@
+
+OBJS = $(OUTDIR)/misc.obj $(OUTDIR)/w32err.obj $(OUTDIR)/sub_proc.obj
+
+$(OUTDIR)/subproc.lib: $(OUTDIR) $(OBJS)
+ $(LIB) -out:$@ @<<
+ $(OBJS)
+<<
+
+.c{$(OUTDIR)}.obj:
+ $(CC) $(CFLAGS) /c $<
+
+$(OUTDIR)/misc.obj: misc.c proc.h
+$(OUTDIR)/sub_proc.obj: sub_proc.c ../include/sub_proc.h ../include/w32err.h proc.h
+$(OUTDIR)/w32err.obj: w32err.c ../include/w32err.h
diff --git a/src/kmk/w32/subproc/misc.c b/src/kmk/w32/subproc/misc.c
new file mode 100644
index 0000000..8b17413
--- /dev/null
+++ b/src/kmk/w32/subproc/misc.c
@@ -0,0 +1,83 @@
+/* Process handling for Windows
+Copyright (C) 1996-2016 Free Software Foundation, Inc.
+This file is part of GNU Make.
+
+GNU Make 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 3 of the License, or (at your option) any later
+version.
+
+GNU Make 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 <config.h>
+#include <stddef.h>
+#include <stdlib.h>
+#include <string.h>
+#include <windows.h>
+#include "proc.h"
+
+
+/*
+ * Description: Convert a NULL string terminated UNIX environment block to
+ * an environment block suitable for a windows32 system call
+ *
+ * Returns: TRUE= success, FALSE=fail
+ *
+ * Notes/Dependencies: the environment block is sorted in case-insensitive
+ * order, is double-null terminated, and is a char *, not a char **
+ */
+int _cdecl compare(const void *a1, const void *a2)
+{
+ return _stricoll(*((char**)a1),*((char**)a2));
+}
+bool_t
+arr2envblk(char **arr, char **envblk_out, int *envsize_needed)
+{
+ char **tmp;
+ int size_needed;
+ int arrcnt;
+ char *ptr;
+
+ arrcnt = 0;
+ while (arr[arrcnt]) {
+ arrcnt++;
+ }
+
+ tmp = (char**) calloc(arrcnt + 1, sizeof(char *));
+ if (!tmp) {
+ return FALSE;
+ }
+
+ arrcnt = 0;
+ size_needed = *envsize_needed = 0;
+ while (arr[arrcnt]) {
+ tmp[arrcnt] = arr[arrcnt];
+ size_needed += strlen(arr[arrcnt]) + 1;
+ arrcnt++;
+ }
+ size_needed++;
+ *envsize_needed = size_needed;
+
+ qsort((void *) tmp, (size_t) arrcnt, sizeof (char*), compare);
+
+ ptr = *envblk_out = calloc(size_needed, 1);
+ if (!ptr) {
+ free(tmp);
+ return FALSE;
+ }
+
+ arrcnt = 0;
+ while (tmp[arrcnt]) {
+ strcpy(ptr, tmp[arrcnt]);
+ ptr += strlen(tmp[arrcnt]) + 1;
+ arrcnt++;
+ }
+
+ free(tmp);
+ return TRUE;
+}
diff --git a/src/kmk/w32/subproc/proc.h b/src/kmk/w32/subproc/proc.h
new file mode 100644
index 0000000..7ccb5ea
--- /dev/null
+++ b/src/kmk/w32/subproc/proc.h
@@ -0,0 +1,29 @@
+/* Definitions for Windows
+Copyright (C) 1996-2016 Free Software Foundation, Inc.
+This file is part of GNU Make.
+
+GNU Make 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 3 of the License, or (at your option) any later
+version.
+
+GNU Make 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/>. */
+
+#ifndef _PROC_H
+#define _PROC_H
+
+typedef int bool_t;
+
+#define E_SCALL 101
+#define E_IO 102
+#define E_NO_MEM 103
+#define E_FORK 104
+
+extern bool_t arr2envblk(char **arr, char **envblk_out, int *envsize_needed);
+
+#endif
diff --git a/src/kmk/w32/subproc/sub_proc.c b/src/kmk/w32/subproc/sub_proc.c
new file mode 100644
index 0000000..6ccfa47
--- /dev/null
+++ b/src/kmk/w32/subproc/sub_proc.c
@@ -0,0 +1,1714 @@
+/* Process handling for Windows.
+Copyright (C) 1996-2016 Free Software Foundation, Inc.
+This file is part of GNU Make.
+
+GNU Make 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 3 of the License, or (at your option) any later
+version.
+
+GNU Make 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 <config.h>
+#include <stdlib.h>
+#include <stdio.h>
+#include <io.h> /* for _get_osfhandle */
+#ifdef _MSC_VER
+# include <stddef.h> /* for intptr_t */
+#else
+# include <stdint.h>
+#endif
+#include <string.h>
+#include <process.h> /* for msvc _beginthreadex, _endthreadex */
+#include <signal.h>
+#include <windows.h>
+
+#include "makeint.h"
+#include "filedef.h"
+#include "variable.h"
+#include "sub_proc.h"
+#include "proc.h"
+#include "w32err.h"
+#include "debug.h"
+
+#ifdef KMK
+# include <assert.h>
+# include "kmkbuiltin.h"
+extern void kmk_cache_exec_image_a(const char *); /* imagecache.c */
+#endif
+
+static char *make_command_line(char *shell_name, char *exec_path, char **argv);
+
+typedef struct sub_process_t {
+#ifdef KMK
+ enum { kRegular = 0, kSubmit, kSubProcFreed } enmType;
+ intptr_t clue;
+#endif
+ intptr_t sv_stdin[2];
+ intptr_t sv_stdout[2];
+ intptr_t sv_stderr[2];
+ int using_pipes;
+ char *inp;
+ DWORD incnt;
+ char * volatile outp;
+ volatile DWORD outcnt;
+ char * volatile errp;
+ volatile DWORD errcnt;
+ pid_t pid;
+ int exit_code;
+ int signal;
+ long last_err;
+ long lerrno;
+} sub_process;
+
+static long process_file_io_private(sub_process *pproc, BOOL fNeedToWait); /* bird */
+
+/* keep track of children so we can implement a waitpid-like routine */
+static sub_process *proc_array[MAXIMUM_WAIT_OBJECTS];
+static int proc_index = 0;
+static int fake_exits_pending = 0;
+
+
+/*
+ * Fill a HANDLE list with handles to wait for.
+ */
+DWORD
+process_set_handles(HANDLE *handles)
+{
+ DWORD count = 0;
+ int i;
+
+ /* Build array of handles to wait for */
+ for (i = 0; i < proc_index; i++) {
+ /* Don't wait on child processes that have already finished */
+ if (fake_exits_pending && proc_array[i]->exit_code)
+ continue;
+
+ handles[count++] = (HANDLE) proc_array[i]->pid;
+ }
+
+ return count;
+}
+
+#ifndef KMK /* Inefficient! */
+/*
+ * When a process has been waited for, adjust the wait state
+ * array so that we don't wait for it again
+ */
+static void
+process_adjust_wait_state(sub_process* pproc)
+{
+ int i;
+
+ if (!proc_index)
+ return;
+
+ for (i = 0; i < proc_index; i++)
+ if (proc_array[i]->pid == pproc->pid)
+ break;
+
+ if (i < proc_index) {
+ proc_index--;
+ if (i != proc_index)
+ memmove(&proc_array[i], &proc_array[i+1],
+ (proc_index-i) * sizeof(sub_process*));
+ proc_array[proc_index] = NULL;
+ }
+}
+
+#endif /* !KMK */
+
+/*
+ * Waits for any of the registered child processes to finish.
+ */
+static sub_process *
+process_wait_for_any_private(int block, DWORD* pdwWaitStatus)
+{
+ HANDLE handles[MAXIMUM_WAIT_OBJECTS];
+ DWORD retval, which;
+ int i;
+
+ if (!proc_index)
+ return NULL;
+
+ /* build array of handles to wait for */
+ for (i = 0; i < proc_index; i++) {
+ handles[i] = (HANDLE) proc_array[i]->pid;
+
+ if (fake_exits_pending && proc_array[i]->exit_code)
+ break;
+ }
+
+ /* wait for someone to exit */
+ if (!fake_exits_pending) {
+#ifdef KMK
+l_wait_again:
+#endif
+ retval = WaitForMultipleObjects(proc_index, handles, FALSE, (block ? INFINITE : 0));
+ which = retval - WAIT_OBJECT_0;
+ } else {
+ fake_exits_pending--;
+ retval = !WAIT_FAILED;
+ which = i;
+ }
+
+ /* If the pointer is not NULL, set the wait status result variable. */
+ if (pdwWaitStatus)
+ *pdwWaitStatus = retval;
+
+ /* return pointer to process */
+ if ((retval == WAIT_TIMEOUT) || (retval == WAIT_FAILED)) {
+ return NULL;
+ }
+ else {
+ sub_process* pproc = proc_array[which];
+#ifdef KMK
+ if (pproc->enmType == kSubmit) {
+ /* Try get the result from kSubmit.c. This may not succeed if the whole
+ result hasn't arrived yet, in which we just restart the wait. */
+ if (kSubmitSubProcGetResult(pproc->clue, &pproc->exit_code, &pproc->signal) != 0) {
+ goto l_wait_again;
+ }
+ }
+#endif
+#ifndef KMK /* Inefficient! */
+ process_adjust_wait_state(pproc);
+#else
+ proc_index--;
+ if ((int)which < proc_index)
+ proc_array[which] = proc_array[proc_index];
+ proc_array[proc_index] = NULL;
+#endif
+ return pproc;
+ }
+}
+
+/*
+ * Terminate a process.
+ */
+BOOL
+ process_kill(HANDLE proc, int signal)
+{
+ sub_process* pproc = (sub_process*) proc;
+#ifdef KMK
+ if (pproc->enmType == kRegular) {
+#endif
+ pproc->signal = signal;
+ return (TerminateProcess((HANDLE) pproc->pid, signal));
+#ifdef KMK
+ } else if (pproc->enmType == kSubmit)
+ return kSubmitSubProcKill(pproc->clue, signal) == 0;
+ assert(0);
+ return FALSE;
+#endif
+}
+
+/*
+ * Use this function to register processes you wish to wait for by
+ * calling process_file_io(NULL) or process_wait_any(). This must be done
+ * because it is possible for callers of this library to reuse the same
+ * handle for multiple processes launches :-(
+ */
+void
+process_register(HANDLE proc)
+{
+#ifdef KMK
+ assert(((sub_process *)proc)->enmType == kRegular);
+#endif
+ if (proc_index < MAXIMUM_WAIT_OBJECTS)
+ proc_array[proc_index++] = (sub_process *) proc;
+}
+
+#ifdef KMK
+
+/**
+ * Interface used by kmkbuiltin/kSubmit.c to register stuff going down in a
+ * worker process.
+ *
+ * @returns 0 on success, -1 if there are too many sub-processes already.
+ * @param hEvent The event semaphore to wait on.
+ * @param clue The clue to base.
+ * @param pPid Where to return the pid that job.c expects.
+ */
+int
+process_kmk_register_submit(HANDLE hEvent, intptr_t clue, pid_t *pPid)
+{
+ if (proc_index < MAXIMUM_WAIT_OBJECTS) {
+ sub_process *pSubProc = (sub_process *)xcalloc(sizeof(*pSubProc));
+ pSubProc->enmType = kSubmit;
+ pSubProc->clue = clue;
+ pSubProc->pid = (intptr_t)hEvent;
+
+ proc_array[proc_index++] = pSubProc;
+ *pPid = (intptr_t)pSubProc;
+ return 0;
+ }
+ return -1;
+}
+
+/**
+ * Interface used by kmkbuiltin/kRedirect.c to register a spawned process.
+ *
+ * @returns 0 on success, -1 if there are too many sub-processes already.
+ * @param hProcess The process handle.
+ * @param pPid Where to return the pid that job.c expects.
+ */
+int
+process_kmk_register_redirect(HANDLE hProcess, pid_t *pPid)
+{
+ if (proc_index < MAXIMUM_WAIT_OBJECTS) {
+ sub_process *pSubProc = (sub_process *)xcalloc(sizeof(*pSubProc));
+ pSubProc->enmType = kRegular;
+ pSubProc->pid = (intptr_t)hProcess;
+
+ proc_array[proc_index++] = pSubProc;
+ *pPid = (intptr_t)pSubProc;
+ return 0;
+ }
+ return -1;
+}
+
+#endif /* KMK */
+
+/*
+ * Return the number of processes that we are still waiting for.
+ */
+int
+process_used_slots(void)
+{
+ return proc_index;
+}
+
+/*
+ * Public function which works kind of like waitpid(). Wait for any
+ * of the children to die and return results. To call this function,
+ * you must do 1 of things:
+ *
+ * x = process_easy(...);
+ *
+ * or
+ *
+ * x = process_init_fd();
+ * process_register(x);
+ *
+ * or
+ *
+ * x = process_init();
+ * process_register(x);
+ *
+ * You must NOT then call process_pipe_io() because this function is
+ * not capable of handling automatic notification of any child
+ * death.
+ */
+
+HANDLE
+process_wait_for_any(int block, DWORD* pdwWaitStatus)
+{
+ sub_process* pproc = process_wait_for_any_private(block, pdwWaitStatus);
+
+ if (!pproc)
+ return NULL;
+ else {
+ /*
+ * Ouch! can't tell caller if this fails directly. Caller
+ * will have to use process_last_err()
+ */
+#ifdef KMK
+ /* Invalidate negative directory cache entries now that a
+ job has completed and possibly created new files that
+ was missing earlier. */
+ dir_cache_invalid_after_job ();
+
+ if (pproc->enmType == kRegular) {
+ (void)process_file_io_private(pproc, FALSE);
+ }
+#else
+ (void) process_file_io(pproc);
+#endif
+ return ((HANDLE) pproc);
+ }
+}
+
+long
+process_signal(HANDLE proc)
+{
+ if (proc == INVALID_HANDLE_VALUE) return 0;
+ return (((sub_process *)proc)->signal);
+}
+
+long
+process_last_err(HANDLE proc)
+{
+ if (proc == INVALID_HANDLE_VALUE) return ERROR_INVALID_HANDLE;
+ return (((sub_process *)proc)->last_err);
+}
+
+long
+process_exit_code(HANDLE proc)
+{
+ if (proc == INVALID_HANDLE_VALUE) return EXIT_FAILURE;
+ return (((sub_process *)proc)->exit_code);
+}
+
+void
+process_noinherit(int fd)
+{
+ HANDLE fh = (HANDLE)_get_osfhandle(fd);
+
+ if (fh && fh != INVALID_HANDLE_VALUE)
+ SetHandleInformation(fh, HANDLE_FLAG_INHERIT, 0);
+}
+
+/*
+2006-02:
+All the following functions are currently unused.
+All of them would crash gmake if called with argument INVALID_HANDLE_VALUE.
+Hence whoever wants to use one of this functions must invent and implement
+a reasonable error handling for this function.
+
+char *
+process_outbuf(HANDLE proc)
+{
+ return (((sub_process *)proc)->outp);
+}
+
+char *
+process_errbuf(HANDLE proc)
+{
+ return (((sub_process *)proc)->errp);
+}
+
+int
+process_outcnt(HANDLE proc)
+{
+ return (((sub_process *)proc)->outcnt);
+}
+
+int
+process_errcnt(HANDLE proc)
+{
+ return (((sub_process *)proc)->errcnt);
+}
+
+void
+process_pipes(HANDLE proc, int pipes[3])
+{
+ pipes[0] = ((sub_process *)proc)->sv_stdin[0];
+ pipes[1] = ((sub_process *)proc)->sv_stdout[0];
+ pipes[2] = ((sub_process *)proc)->sv_stderr[0];
+ return;
+}
+*/
+
+ HANDLE
+process_init()
+{
+ sub_process *pproc;
+ /*
+ * open file descriptors for attaching stdin/stdout/sterr
+ */
+ HANDLE stdin_pipes[2];
+ HANDLE stdout_pipes[2];
+ HANDLE stderr_pipes[2];
+ SECURITY_ATTRIBUTES inherit;
+ BYTE sd[SECURITY_DESCRIPTOR_MIN_LENGTH];
+
+ pproc = malloc(sizeof(*pproc));
+ memset(pproc, 0, sizeof(*pproc));
+
+ /* We can't use NULL for lpSecurityDescriptor because that
+ uses the default security descriptor of the calling process.
+ Instead we use a security descriptor with no DACL. This
+ allows nonrestricted access to the associated objects. */
+
+ if (!InitializeSecurityDescriptor((PSECURITY_DESCRIPTOR)(&sd),
+ SECURITY_DESCRIPTOR_REVISION)) {
+ pproc->last_err = GetLastError();
+ pproc->lerrno = E_SCALL;
+ return((HANDLE)pproc);
+ }
+
+ inherit.nLength = sizeof(inherit);
+ inherit.lpSecurityDescriptor = (PSECURITY_DESCRIPTOR)(&sd);
+ inherit.bInheritHandle = TRUE;
+
+ // By convention, parent gets pipe[0], and child gets pipe[1]
+ // This means the READ side of stdin pipe goes into pipe[1]
+ // and the WRITE side of the stdout and stderr pipes go into pipe[1]
+ if (CreatePipe( &stdin_pipes[1], &stdin_pipes[0], &inherit, 0) == FALSE ||
+ CreatePipe( &stdout_pipes[0], &stdout_pipes[1], &inherit, 0) == FALSE ||
+ CreatePipe( &stderr_pipes[0], &stderr_pipes[1], &inherit, 0) == FALSE) {
+
+ pproc->last_err = GetLastError();
+ pproc->lerrno = E_SCALL;
+ return((HANDLE)pproc);
+ }
+
+ //
+ // Mark the parent sides of the pipes as non-inheritable
+ //
+ if (SetHandleInformation(stdin_pipes[0],
+ HANDLE_FLAG_INHERIT, 0) == FALSE ||
+ SetHandleInformation(stdout_pipes[0],
+ HANDLE_FLAG_INHERIT, 0) == FALSE ||
+ SetHandleInformation(stderr_pipes[0],
+ HANDLE_FLAG_INHERIT, 0) == FALSE) {
+
+ pproc->last_err = GetLastError();
+ pproc->lerrno = E_SCALL;
+ return((HANDLE)pproc);
+ }
+ pproc->sv_stdin[0] = (intptr_t) stdin_pipes[0];
+ pproc->sv_stdin[1] = (intptr_t) stdin_pipes[1];
+ pproc->sv_stdout[0] = (intptr_t) stdout_pipes[0];
+ pproc->sv_stdout[1] = (intptr_t) stdout_pipes[1];
+ pproc->sv_stderr[0] = (intptr_t) stderr_pipes[0];
+ pproc->sv_stderr[1] = (intptr_t) stderr_pipes[1];
+
+ pproc->using_pipes = 1;
+
+ pproc->lerrno = 0;
+
+ return((HANDLE)pproc);
+}
+
+
+ HANDLE
+process_init_fd(HANDLE stdinh, HANDLE stdouth, HANDLE stderrh)
+{
+ sub_process *pproc;
+
+ pproc = malloc(sizeof(*pproc));
+ if (pproc) {
+ memset(pproc, 0, sizeof(*pproc));
+
+ /*
+ * Just pass the provided file handles to the 'child
+ * side' of the pipe, bypassing pipes altogether.
+ */
+ pproc->sv_stdin[1] = (intptr_t) stdinh;
+ pproc->sv_stdout[1] = (intptr_t) stdouth;
+ pproc->sv_stderr[1] = (intptr_t) stderrh;
+
+ pproc->last_err = pproc->lerrno = 0;
+ }
+
+ return((HANDLE)pproc);
+}
+
+
+static HANDLE
+find_file(const char *exec_path, const char *path_var,
+ char *full_fname, DWORD full_len)
+{
+ HANDLE exec_handle;
+ char *fname;
+ char *ext;
+ DWORD req_len;
+ int i;
+ static const char *extensions[] =
+ /* Should .com come before no-extension case? */
+ { ".exe", ".cmd", ".bat", "", ".com", NULL };
+
+ fname = xmalloc(strlen(exec_path) + 5);
+ strcpy(fname, exec_path);
+ ext = fname + strlen(fname);
+
+ for (i = 0; extensions[i]; i++) {
+ strcpy(ext, extensions[i]);
+ if (((req_len = SearchPath (path_var, fname, NULL, full_len,
+ full_fname, NULL)) > 0
+ /* For compatibility with previous code, which
+ used OpenFile, and with Windows operation in
+ general, also look in various default
+ locations, such as Windows directory and
+ Windows System directory. Warning: this also
+ searches PATH in the Make's environment, which
+ might not be what the Makefile wants, but it
+ seems to be OK as a fallback, after the
+ previous SearchPath failed to find on child's
+ PATH. */
+ || (req_len = SearchPath (NULL, fname, NULL, full_len,
+ full_fname, NULL)) > 0)
+ && req_len <= full_len
+ && (exec_handle =
+ CreateFile(full_fname,
+ GENERIC_READ,
+ FILE_SHARE_READ | FILE_SHARE_WRITE,
+ NULL,
+ OPEN_EXISTING,
+ FILE_ATTRIBUTE_NORMAL,
+ NULL)) != INVALID_HANDLE_VALUE) {
+ free(fname);
+ return(exec_handle);
+ }
+ }
+
+ free(fname);
+ return INVALID_HANDLE_VALUE;
+}
+
+/*
+ * Return non-zero of FNAME specifies a batch file and its name
+ * includes embedded whitespace.
+ */
+
+static int
+batch_file_with_spaces(const char *fname)
+{
+ size_t fnlen = strlen(fname);
+
+ return (fnlen > 4
+ && (_strnicmp(fname + fnlen - 4, ".bat", 4) == 0
+ || _strnicmp(fname + fnlen - 4, ".cmd", 4) == 0)
+ /* The set of characters in the 2nd arg to strpbrk
+ should be the same one used by make_command_line
+ below to decide whether an argv[] element needs
+ quoting. */
+ && strpbrk(fname, " \t") != NULL);
+}
+
+
+/*
+ * Description: Create the child process to be helped
+ *
+ * Returns: success <=> 0
+ *
+ * Notes/Dependencies:
+ */
+long
+process_begin(
+ HANDLE proc,
+ char **argv,
+ char **envp,
+ char *exec_path,
+ char *as_user)
+{
+ sub_process *pproc = (sub_process *)proc;
+ char *shell_name = 0;
+ int file_not_found=0;
+ HANDLE exec_handle;
+ char exec_fname[MAX_PATH];
+ const char *path_var = NULL;
+ char **ep;
+ char buf[MAX_PATH];
+ DWORD bytes_returned;
+ DWORD flags;
+ char *command_line;
+ STARTUPINFO startInfo;
+ PROCESS_INFORMATION procInfo;
+ char *envblk=NULL;
+ int envsize_needed = 0;
+ int pass_null_exec_path = 0;
+#ifdef KMK
+ size_t exec_path_len;
+ extern int process_priority;
+
+ assert (pproc->enmType == kRegular);
+#endif
+
+ /*
+ * Shell script detection... if the exec_path starts with #! then
+ * we want to exec shell-script-name exec-path, not just exec-path
+ * NT doesn't recognize #!/bin/sh or #!/etc/Tivoli/bin/perl. We do not
+ * hard-code the path to the shell or perl or whatever: Instead, we
+ * assume it's in the path somewhere (generally, the NT tools
+ * bin directory)
+ */
+
+#ifdef KMK
+ /* kmk performance: Don't bother looking for shell scripts in .exe files. */
+ exec_path_len = strlen(exec_path);
+ if (exec_path_len > 4
+ && exec_path[exec_path_len - 4] == '.'
+ && !stricmp(exec_path + exec_path_len - 3, "exe")) {
+ exec_handle = INVALID_HANDLE_VALUE;
+ exec_fname[0] = '\0';
+ }
+ else {
+#endif /* KMK */
+
+ /* Use the Makefile's value of PATH to look for the program to
+ execute, because it could be different from Make's PATH
+ (e.g., if the target sets its own value. */
+ if (envp)
+ for (ep = envp; *ep; ep++) {
+ if (strncmp (*ep, "PATH=", 5) == 0
+ || strncmp (*ep, "Path=", 5) == 0) {
+ path_var = *ep + 5;
+ break;
+ }
+ }
+ exec_handle = find_file(exec_path, path_var,
+ exec_fname, sizeof(exec_fname));
+#ifdef KMK
+ }
+#endif
+
+ /*
+ * If we couldn't open the file, just assume that Windows will be
+ * somehow able to find and execute it. If the first character
+ * of the command is '/', assume they set SHELL to a Unixy shell
+ * that have some magic mounts known only to it, and run the whole
+ * command via $SHELL -c "COMMAND" instead.
+ */
+ if (exec_handle == INVALID_HANDLE_VALUE) {
+ if (exec_path[0] == '/') {
+ char *new_argv0;
+ char **argvi = argv;
+ int arglen = 0;
+
+ strcpy(buf, variable_expand ("$(SHELL)"));
+ shell_name = &buf[0];
+ strcpy(exec_fname, "-c");
+ /* Construct a single command string in argv[0]. */
+ while (*argvi) {
+ arglen += strlen(*argvi) + 1;
+ argvi++;
+ }
+ new_argv0 = xmalloc(arglen + 1);
+ new_argv0[0] = '\0';
+ for (argvi = argv; *argvi; argvi++) {
+ strcat(new_argv0, *argvi);
+ strcat(new_argv0, " ");
+ }
+ /* Remove the extra blank at the end. */
+ new_argv0[arglen-1] = '\0';
+ free(argv[0]);
+ argv[0] = new_argv0;
+ argv[1] = NULL;
+ }
+ else
+ file_not_found++;
+ }
+ else {
+ /* Attempt to read the first line of the file */
+ if (ReadFile( exec_handle,
+ buf, sizeof(buf) - 1, /* leave room for trailing NULL */
+ &bytes_returned, 0) == FALSE || bytes_returned < 2) {
+
+ pproc->last_err = GetLastError();
+ pproc->lerrno = E_IO;
+ CloseHandle(exec_handle);
+ return(-1);
+ }
+ if (buf[0] == '#' && buf[1] == '!') {
+ /*
+ * This is a shell script... Change the command line from
+ * exec_path args to shell_name exec_path args
+ */
+ char *p;
+
+ /* Make sure buf is NULL terminated */
+ buf[bytes_returned] = 0;
+ /*
+ * Depending on the file system type, etc. the first line
+ * of the shell script may end with newline or newline-carriage-return
+ * Whatever it ends with, cut it off.
+ */
+ p= strchr(buf, '\n');
+ if (p)
+ *p = 0;
+ p = strchr(buf, '\r');
+ if (p)
+ *p = 0;
+
+ /*
+ * Find base name of shell
+ */
+ shell_name = strrchr( buf, '/');
+ if (shell_name) {
+ shell_name++;
+ } else {
+ shell_name = &buf[2];/* skipping "#!" */
+ }
+
+ }
+ CloseHandle(exec_handle);
+ }
+
+ flags = 0;
+
+ if (file_not_found)
+ command_line = make_command_line( shell_name, exec_path, argv);
+ else {
+ /* If exec_fname includes whitespace, CreateProcess
+ behaves erratically and unreliably, and often fails
+ if argv[0] also includes whitespace (and thus will
+ be quoted by make_command_line below). So in that
+ case, we don't pass exec_fname as the 1st arg to
+ CreateProcess, but instead replace argv[0] with
+ exec_fname (to keep its leading directories and
+ extension as found by find_file), and pass NULL to
+ CreateProcess as its 1st arg. This works around
+ the bugs in CreateProcess, which are probably
+ caused by its passing the command to cmd.exe with
+ some incorrect quoting. */
+ if (!shell_name
+ && batch_file_with_spaces(exec_fname)
+ && _stricmp(exec_path, argv[0]) == 0) {
+ char *new_argv, *p;
+ char **argvi;
+ int arglen, i;
+ pass_null_exec_path = 1;
+ /* Rewrite argv[] replacing argv[0] with exec_fname. */
+ for (argvi = argv + 1, arglen = strlen(exec_fname) + 1;
+ *argvi;
+ argvi++) {
+ arglen += strlen(*argvi) + 1;
+ }
+ new_argv = xmalloc(arglen);
+ p = strcpy(new_argv, exec_fname) + strlen(exec_fname) + 1;
+ for (argvi = argv + 1, i = 1; *argvi; argvi++, i++) {
+ strcpy(p, *argvi);
+ argv[i] = p;
+ p += strlen(*argvi) + 1;
+ }
+ argv[i] = NULL;
+ free (argv[0]);
+ argv[0] = new_argv;
+ }
+ command_line = make_command_line( shell_name, exec_fname, argv);
+ }
+
+ if ( command_line == NULL ) {
+ pproc->last_err = 0;
+ pproc->lerrno = E_NO_MEM;
+ return(-1);
+ }
+
+ if (envp) {
+ if (arr2envblk(envp, &envblk, &envsize_needed) == FALSE) {
+ pproc->lerrno = E_NO_MEM;
+ free( command_line );
+ if ((pproc->last_err == ERROR_INVALID_PARAMETER
+ || pproc->last_err == ERROR_MORE_DATA)
+ && envsize_needed > 32*1024) {
+ fprintf (stderr, "CreateProcess failed, probably because environment is too large (%d bytes).\n",
+ envsize_needed);
+ }
+ pproc->last_err = 0;
+ return(-1);
+ }
+ }
+
+ if (shell_name || file_not_found || pass_null_exec_path) {
+ exec_path = 0; /* Search for the program in %Path% */
+ } else {
+ exec_path = exec_fname;
+ }
+
+ /*
+ * Set up inherited stdin, stdout, stderr for child
+ */
+ memset(&startInfo, '\0', sizeof(startInfo));
+ GetStartupInfo(&startInfo);
+#ifndef KMK
+ startInfo.dwFlags = STARTF_USESTDHANDLES;
+#endif
+ startInfo.lpReserved = 0;
+ startInfo.cbReserved2 = 0;
+ startInfo.lpReserved2 = 0;
+#ifndef KMK
+ startInfo.hStdInput = (HANDLE)pproc->sv_stdin[1];
+ startInfo.hStdOutput = (HANDLE)pproc->sv_stdout[1];
+ startInfo.hStdError = (HANDLE)pproc->sv_stderr[1];
+#else
+ if ( ((HANDLE)pproc->sv_stdin[1] != INVALID_HANDLE_VALUE && pproc->sv_stdin[1])
+ || ((HANDLE)pproc->sv_stdout[1] != INVALID_HANDLE_VALUE && pproc->sv_stdout[1])
+ || ((HANDLE)pproc->sv_stderr[1] != INVALID_HANDLE_VALUE && pproc->sv_stderr[1]) ) {
+ startInfo.dwFlags = STARTF_USESTDHANDLES;
+ startInfo.hStdInput = (HANDLE)pproc->sv_stdin[1];
+ startInfo.hStdOutput = (HANDLE)pproc->sv_stdout[1];
+ startInfo.hStdError = (HANDLE)pproc->sv_stderr[1];
+ } else {
+ startInfo.dwFlags = 0;
+ startInfo.hStdInput = 0;
+ startInfo.hStdOutput = 0;
+ startInfo.hStdError = 0;
+ }
+#endif
+
+ if (as_user) {
+ free(envblk);
+ return -1;
+ } else {
+ DB (DB_JOBS, ("CreateProcess(%s,%s,...)\n",
+ exec_path ? exec_path : "NULL",
+ command_line ? command_line : "NULL"));
+#ifdef KMK
+ if (exec_fname[0])
+ kmk_cache_exec_image_a(exec_fname);
+ else if (exec_path)
+ kmk_cache_exec_image_a(exec_path);
+ else if (argv[0])
+ kmk_cache_exec_image_a(argv[0]);
+
+ switch (process_priority) {
+ case 1: flags |= CREATE_SUSPENDED | IDLE_PRIORITY_CLASS; break;
+ case 2: flags |= CREATE_SUSPENDED | BELOW_NORMAL_PRIORITY_CLASS; break;
+ case 3: flags |= CREATE_SUSPENDED | NORMAL_PRIORITY_CLASS; break;
+ case 4: flags |= CREATE_SUSPENDED | HIGH_PRIORITY_CLASS; break;
+ case 5: flags |= CREATE_SUSPENDED | REALTIME_PRIORITY_CLASS; break;
+ }
+#endif
+ if (CreateProcess(
+ exec_path,
+ command_line,
+ NULL,
+ 0, /* default security attributes for thread */
+ TRUE, /* inherit handles (e.g. helper pipes, oserv socket) */
+ flags,
+ envblk,
+ 0, /* default starting directory */
+ &startInfo,
+ &procInfo) == FALSE) {
+
+ pproc->last_err = GetLastError();
+ pproc->lerrno = E_FORK;
+ fprintf(stderr, "process_begin: CreateProcess(%s, %s, ...) failed.\n",
+ exec_path ? exec_path : "NULL", command_line);
+ free(envblk);
+ free( command_line );
+ return(-1);
+ }
+#ifdef KMK
+ switch (process_priority) {
+ case 1: SetThreadPriority(procInfo.hThread, THREAD_PRIORITY_IDLE); break;
+ case 2: SetThreadPriority(procInfo.hThread, THREAD_PRIORITY_BELOW_NORMAL); break;
+ case 3: SetThreadPriority(procInfo.hThread, THREAD_PRIORITY_NORMAL); break;
+ case 4: SetThreadPriority(procInfo.hThread, THREAD_PRIORITY_HIGHEST); break;
+ case 5: SetThreadPriority(procInfo.hThread, THREAD_PRIORITY_TIME_CRITICAL); break;
+ }
+ ResumeThread(procInfo.hThread);
+#endif
+ }
+
+ pproc->pid = (pid_t)procInfo.hProcess;
+ /* Close the thread handle -- we'll just watch the process */
+ CloseHandle(procInfo.hThread);
+
+ /* Close the halves of the pipes we don't need */
+ if ((HANDLE)pproc->sv_stdin[1] != INVALID_HANDLE_VALUE && pproc->sv_stdin[1])
+ CloseHandle((HANDLE)pproc->sv_stdin[1]);
+ if ((HANDLE)pproc->sv_stdout[1] != INVALID_HANDLE_VALUE && pproc->sv_stdout[1])
+ CloseHandle((HANDLE)pproc->sv_stdout[1]);
+ if ((HANDLE)pproc->sv_stderr[1] != INVALID_HANDLE_VALUE && pproc->sv_stderr[1])
+ CloseHandle((HANDLE)pproc->sv_stderr[1]);
+ pproc->sv_stdin[1] = 0;
+ pproc->sv_stdout[1] = 0;
+ pproc->sv_stderr[1] = 0;
+
+ free( command_line );
+ free(envblk);
+ pproc->lerrno=0;
+ return 0;
+}
+
+
+
+#if 0 /* unused */
+static DWORD
+proc_stdin_thread(sub_process *pproc)
+{
+ DWORD in_done;
+ for (;;) {
+ if (WriteFile( (HANDLE) pproc->sv_stdin[0], pproc->inp, pproc->incnt,
+ &in_done, NULL) == FALSE)
+ _endthreadex(0);
+ // This if should never be true for anonymous pipes, but gives
+ // us a chance to change I/O mechanisms later
+ if (in_done < pproc->incnt) {
+ pproc->incnt -= in_done;
+ pproc->inp += in_done;
+ } else {
+ _endthreadex(0);
+ }
+ }
+ return 0; // for compiler warnings only.. not reached
+}
+
+static DWORD
+proc_stdout_thread(sub_process *pproc)
+{
+ DWORD bufsize = 1024;
+ char c;
+ DWORD nread;
+ pproc->outp = malloc(bufsize);
+ if (pproc->outp == NULL)
+ _endthreadex(0);
+ pproc->outcnt = 0;
+
+ for (;;) {
+ if (ReadFile( (HANDLE)pproc->sv_stdout[0], &c, 1, &nread, NULL)
+ == FALSE) {
+/* map_windows32_error_to_string(GetLastError());*/
+ _endthreadex(0);
+ }
+ if (nread == 0)
+ _endthreadex(0);
+ if (pproc->outcnt + nread > bufsize) {
+ bufsize += nread + 512;
+ pproc->outp = realloc(pproc->outp, bufsize);
+ if (pproc->outp == NULL) {
+ pproc->outcnt = 0;
+ _endthreadex(0);
+ }
+ }
+ pproc->outp[pproc->outcnt++] = c;
+ }
+ return 0;
+}
+
+static DWORD
+proc_stderr_thread(sub_process *pproc)
+{
+ DWORD bufsize = 1024;
+ char c;
+ DWORD nread;
+ pproc->errp = malloc(bufsize);
+ if (pproc->errp == NULL)
+ _endthreadex(0);
+ pproc->errcnt = 0;
+
+ for (;;) {
+ if (ReadFile( (HANDLE)pproc->sv_stderr[0], &c, 1, &nread, NULL) == FALSE) {
+ map_windows32_error_to_string(GetLastError());
+ _endthreadex(0);
+ }
+ if (nread == 0)
+ _endthreadex(0);
+ if (pproc->errcnt + nread > bufsize) {
+ bufsize += nread + 512;
+ pproc->errp = realloc(pproc->errp, bufsize);
+ if (pproc->errp == NULL) {
+ pproc->errcnt = 0;
+ _endthreadex(0);
+ }
+ }
+ pproc->errp[pproc->errcnt++] = c;
+ }
+ return 0;
+}
+
+
+/*
+ * Purpose: collects output from child process and returns results
+ *
+ * Description:
+ *
+ * Returns:
+ *
+ * Notes/Dependencies:
+ */
+ long
+process_pipe_io(
+ HANDLE proc,
+ char *stdin_data,
+ int stdin_data_len)
+{
+ sub_process *pproc = (sub_process *)proc;
+ bool_t stdin_eof = FALSE, stdout_eof = FALSE, stderr_eof = FALSE;
+ HANDLE childhand = (HANDLE) pproc->pid;
+ HANDLE tStdin = NULL, tStdout = NULL, tStderr = NULL;
+ unsigned int dwStdin, dwStdout, dwStderr;
+ HANDLE wait_list[4];
+ DWORD wait_count;
+ DWORD wait_return;
+ HANDLE ready_hand;
+ bool_t child_dead = FALSE;
+ BOOL GetExitCodeResult;
+#ifdef KMK
+ assert (pproc->enmType == kRegular);
+#endif
+
+ /*
+ * Create stdin thread, if needed
+ */
+ pproc->inp = stdin_data;
+ pproc->incnt = stdin_data_len;
+ if (!pproc->inp) {
+ stdin_eof = TRUE;
+ CloseHandle((HANDLE)pproc->sv_stdin[0]);
+ pproc->sv_stdin[0] = 0;
+ } else {
+ tStdin = (HANDLE) _beginthreadex( 0, 1024,
+ (unsigned (__stdcall *) (void *))proc_stdin_thread,
+ pproc, 0, &dwStdin);
+ if (tStdin == 0) {
+ pproc->last_err = GetLastError();
+ pproc->lerrno = E_SCALL;
+ goto done;
+ }
+ }
+
+ /*
+ * Assume child will produce stdout and stderr
+ */
+ tStdout = (HANDLE) _beginthreadex( 0, 1024,
+ (unsigned (__stdcall *) (void *))proc_stdout_thread, pproc, 0,
+ &dwStdout);
+ tStderr = (HANDLE) _beginthreadex( 0, 1024,
+ (unsigned (__stdcall *) (void *))proc_stderr_thread, pproc, 0,
+ &dwStderr);
+
+ if (tStdout == 0 || tStderr == 0) {
+
+ pproc->last_err = GetLastError();
+ pproc->lerrno = E_SCALL;
+ goto done;
+ }
+
+
+ /*
+ * Wait for all I/O to finish and for the child process to exit
+ */
+
+ while (!stdin_eof || !stdout_eof || !stderr_eof || !child_dead) {
+ wait_count = 0;
+ if (!stdin_eof) {
+ wait_list[wait_count++] = tStdin;
+ }
+ if (!stdout_eof) {
+ wait_list[wait_count++] = tStdout;
+ }
+ if (!stderr_eof) {
+ wait_list[wait_count++] = tStderr;
+ }
+ if (!child_dead) {
+ wait_list[wait_count++] = childhand;
+ }
+
+ wait_return = WaitForMultipleObjects(wait_count, wait_list,
+ FALSE, /* don't wait for all: one ready will do */
+ child_dead? 1000 :INFINITE); /* after the child dies, subthreads have
+ one second to collect all remaining output */
+
+ if (wait_return == WAIT_FAILED) {
+/* map_windows32_error_to_string(GetLastError());*/
+ pproc->last_err = GetLastError();
+ pproc->lerrno = E_SCALL;
+ goto done;
+ }
+
+ ready_hand = wait_list[wait_return - WAIT_OBJECT_0];
+
+ if (ready_hand == tStdin) {
+ CloseHandle((HANDLE)pproc->sv_stdin[0]);
+ pproc->sv_stdin[0] = 0;
+ CloseHandle(tStdin);
+ tStdin = 0;
+ stdin_eof = TRUE;
+
+ } else if (ready_hand == tStdout) {
+
+ CloseHandle((HANDLE)pproc->sv_stdout[0]);
+ pproc->sv_stdout[0] = 0;
+ CloseHandle(tStdout);
+ tStdout = 0;
+ stdout_eof = TRUE;
+
+ } else if (ready_hand == tStderr) {
+
+ CloseHandle((HANDLE)pproc->sv_stderr[0]);
+ pproc->sv_stderr[0] = 0;
+ CloseHandle(tStderr);
+ tStderr = 0;
+ stderr_eof = TRUE;
+
+ } else if (ready_hand == childhand) {
+
+ DWORD ierr;
+ GetExitCodeResult = GetExitCodeProcess(childhand, &ierr);
+ if (ierr == CONTROL_C_EXIT) {
+ pproc->signal = SIGINT;
+ } else {
+ pproc->exit_code = ierr;
+ }
+ if (GetExitCodeResult == FALSE) {
+ pproc->last_err = GetLastError();
+ pproc->lerrno = E_SCALL;
+ goto done;
+ }
+ child_dead = TRUE;
+
+ } else {
+
+ /* ?? Got back a handle we didn't query ?? */
+ pproc->last_err = 0;
+ pproc->lerrno = E_FAIL;
+ goto done;
+ }
+ }
+
+ done:
+ if (tStdin != 0)
+ CloseHandle(tStdin);
+ if (tStdout != 0)
+ CloseHandle(tStdout);
+ if (tStderr != 0)
+ CloseHandle(tStderr);
+
+ if (pproc->lerrno)
+ return(-1);
+ else
+ return(0);
+
+}
+#endif /* unused */
+
+#ifndef KMK /* unused */
+/*
+ * Purpose: collects output from child process and returns results
+ *
+ * Description:
+ *
+ * Returns:
+ *
+ * Notes/Dependencies:
+ */
+ long
+process_file_io(
+ HANDLE proc)
+{
+ sub_process *pproc;
+
+ if (proc == NULL)
+ pproc = process_wait_for_any_private(1, 0);
+ else
+ pproc = (sub_process *)proc;
+
+ /* some sort of internal error */
+ if (!pproc)
+ return -1;
+
+ return process_file_io_private(proc, TRUE);
+}
+#endif /* !KMK - unused */
+
+/* private function, avoid some kernel calls. (bird) */
+static long
+process_file_io_private(
+ sub_process *pproc,
+ BOOL fNeedToWait)
+{
+ HANDLE childhand;
+ DWORD wait_return;
+ BOOL GetExitCodeResult;
+ DWORD ierr;
+
+ childhand = (HANDLE) pproc->pid;
+
+ /*
+ * This function is poorly named, and could also be used just to wait
+ * for child death if you're doing your own pipe I/O. If that is
+ * the case, close the pipe handles here.
+ */
+ if (pproc->sv_stdin[0]) {
+ CloseHandle((HANDLE)pproc->sv_stdin[0]);
+ pproc->sv_stdin[0] = 0;
+ }
+ if (pproc->sv_stdout[0]) {
+ CloseHandle((HANDLE)pproc->sv_stdout[0]);
+ pproc->sv_stdout[0] = 0;
+ }
+ if (pproc->sv_stderr[0]) {
+ CloseHandle((HANDLE)pproc->sv_stderr[0]);
+ pproc->sv_stderr[0] = 0;
+ }
+
+#ifdef KMK
+ if (childhand == NULL || childhand == INVALID_HANDLE_VALUE) {
+ goto done2;
+ }
+#endif
+
+ /*
+ * Wait for the child process to exit
+ */
+
+ if (fNeedToWait) { /* bird */
+ wait_return = WaitForSingleObject(childhand, INFINITE);
+
+ if (wait_return != WAIT_OBJECT_0) {
+/* map_windows32_error_to_string(GetLastError());*/
+ pproc->last_err = GetLastError();
+ pproc->lerrno = E_SCALL;
+ goto done2;
+ }
+ } /* bird */
+
+ GetExitCodeResult = GetExitCodeProcess(childhand, &ierr);
+ if (ierr == CONTROL_C_EXIT) {
+ pproc->signal = SIGINT;
+ } else {
+ pproc->exit_code = ierr;
+ }
+ if (GetExitCodeResult == FALSE) {
+ pproc->last_err = GetLastError();
+ pproc->lerrno = E_SCALL;
+ }
+
+done2:
+ if (pproc->lerrno)
+ return(-1);
+ else
+ return(0);
+
+}
+
+/*
+ * Description: Clean up any leftover handles, etc. It is up to the
+ * caller to manage and free the input, output, and stderr buffers.
+ */
+ void
+process_cleanup(
+ HANDLE proc)
+{
+ sub_process *pproc = (sub_process *)proc;
+ int i;
+
+#ifdef KMK
+ if (pproc->enmType == kRegular) {
+#endif
+ if (pproc->using_pipes) {
+ for (i= 0; i <= 1; i++) {
+ if ((HANDLE)pproc->sv_stdin[i]
+ && (HANDLE)pproc->sv_stdin[i] != INVALID_HANDLE_VALUE)
+ CloseHandle((HANDLE)pproc->sv_stdin[i]);
+ if ((HANDLE)pproc->sv_stdout[i]
+ && (HANDLE)pproc->sv_stdout[i] != INVALID_HANDLE_VALUE)
+ CloseHandle((HANDLE)pproc->sv_stdout[i]);
+ if ((HANDLE)pproc->sv_stderr[i]
+ && (HANDLE)pproc->sv_stderr[i] != INVALID_HANDLE_VALUE)
+ CloseHandle((HANDLE)pproc->sv_stderr[i]);
+ }
+ }
+ if ((HANDLE)pproc->pid)
+ CloseHandle((HANDLE)pproc->pid);
+#ifdef KMK
+ } else if (pproc->enmType == kSubmit) {
+ kSubmitSubProcCleanup(pproc->clue);
+ } else {
+ assert(0);
+ return;
+ }
+ pproc->enmType = kSubProcFreed;
+#endif
+
+ free(pproc);
+}
+
+
+/*
+ * Description:
+ * Create a command line buffer to pass to CreateProcess
+ *
+ * Returns: the buffer or NULL for failure
+ * Shell case: sh_name a:/full/path/to/script argv[1] argv[2] ...
+ * Otherwise: argv[0] argv[1] argv[2] ...
+ *
+ * Notes/Dependencies:
+ * CreateProcess does not take an argv, so this command creates a
+ * command line for the executable.
+ */
+
+static char *
+make_command_line( char *shell_name, char *full_exec_path, char **argv)
+{
+ int argc = 0;
+ char** argvi;
+ int* enclose_in_quotes = NULL;
+ int* enclose_in_quotes_i;
+ unsigned int bytes_required = 0;
+ char* command_line;
+ char* command_line_i;
+ int cygwin_mode = 0; /* HAVE_CYGWIN_SHELL */
+ int have_sh = 0; /* HAVE_CYGWIN_SHELL */
+#undef HAVE_CYGWIN_SHELL /* bird: paranoia */
+#ifdef HAVE_CYGWIN_SHELL
+ have_sh = (shell_name != NULL || strstr(full_exec_path, "sh.exe"));
+ cygwin_mode = 1;
+#endif
+
+ if (shell_name && full_exec_path) {
+ bytes_required
+ = strlen(shell_name) + 1 + strlen(full_exec_path);
+ /*
+ * Skip argv[0] if any, when shell_name is given.
+ * The special case of "-c" in full_exec_path means
+ * argv[0] is not the shell name, but the command string
+ * to pass to the shell.
+ */
+ if (*argv && strcmp(full_exec_path, "-c")) argv++;
+ /*
+ * Add one for the intervening space.
+ */
+ if (*argv) bytes_required++;
+ }
+
+ argvi = argv;
+ while (*(argvi++)) argc++;
+
+ if (argc) {
+ enclose_in_quotes = (int*) calloc(1, argc * sizeof(int));
+
+ if (!enclose_in_quotes) {
+ return NULL;
+ }
+ }
+
+ /* We have to make one pass through each argv[i] to see if we need
+ * to enclose it in ", so we might as well figure out how much
+ * memory we'll need on the same pass.
+ */
+
+ argvi = argv;
+ enclose_in_quotes_i = enclose_in_quotes;
+ while(*argvi) {
+ char* p = *argvi;
+ unsigned int backslash_count = 0;
+
+ /*
+ * We have to enclose empty arguments in ".
+ */
+ if (!(*p)) *enclose_in_quotes_i = 1;
+
+ while(*p) {
+ switch (*p) {
+ case '\"':
+ /*
+ * We have to insert a backslash for each "
+ * and each \ that precedes the ".
+ */
+ bytes_required += (backslash_count + 1);
+ backslash_count = 0;
+ break;
+
+#if !defined(HAVE_MKS_SHELL) && !defined(HAVE_CYGWIN_SHELL)
+ case '\\':
+ backslash_count++;
+ break;
+#endif
+ /*
+ * At one time we set *enclose_in_quotes_i for '*' or '?' to suppress
+ * wildcard expansion in programs linked with MSVC's SETARGV.OBJ so
+ * that argv in always equals argv out. This was removed. Say you have
+ * such a program named glob.exe. You enter
+ * glob '*'
+ * at the sh command prompt. Obviously the intent is to make glob do the
+ * wildcarding instead of sh. If we set *enclose_in_quotes_i for '*' or '?',
+ * then the command line that glob would see would be
+ * glob "*"
+ * and the _setargv in SETARGV.OBJ would _not_ expand the *.
+ */
+ case ' ':
+ case '\t':
+ *enclose_in_quotes_i = 1;
+ /* fall through */
+
+ default:
+ backslash_count = 0;
+ break;
+ }
+
+ /*
+ * Add one for each character in argv[i].
+ */
+ bytes_required++;
+
+ p++;
+ }
+
+ if (*enclose_in_quotes_i) {
+ /*
+ * Add one for each enclosing ",
+ * and one for each \ that precedes the
+ * closing ".
+ */
+ bytes_required += (backslash_count + 2);
+ }
+
+ /*
+ * Add one for the intervening space.
+ */
+ if (*(++argvi)) bytes_required++;
+ enclose_in_quotes_i++;
+ }
+
+ /*
+ * Add one for the terminating NULL.
+ */
+ bytes_required++;
+#ifdef KMK /* for the space before the final " in case we need it. */
+ bytes_required++;
+#endif
+
+ command_line = (char*) malloc(bytes_required);
+
+ if (!command_line) {
+ free(enclose_in_quotes);
+ return NULL;
+ }
+
+ command_line_i = command_line;
+
+ if (shell_name && full_exec_path) {
+ while(*shell_name) {
+ *(command_line_i++) = *(shell_name++);
+ }
+
+ *(command_line_i++) = ' ';
+
+ while(*full_exec_path) {
+ *(command_line_i++) = *(full_exec_path++);
+ }
+
+ if (*argv) {
+ *(command_line_i++) = ' ';
+ }
+ }
+
+ argvi = argv;
+ enclose_in_quotes_i = enclose_in_quotes;
+
+ while(*argvi) {
+ char* p = *argvi;
+ unsigned int backslash_count = 0;
+
+ if (*enclose_in_quotes_i) {
+ *(command_line_i++) = '\"';
+ }
+
+ while(*p) {
+ if (*p == '\"') {
+ if (cygwin_mode && have_sh) { /* HAVE_CYGWIN_SHELL */
+ /* instead of a \", cygwin likes "" */
+ *(command_line_i++) = '\"';
+ } else {
+
+ /*
+ * We have to insert a backslash for the "
+ * and each \ that precedes the ".
+ */
+ backslash_count++;
+
+ while(backslash_count) {
+ *(command_line_i++) = '\\';
+ backslash_count--;
+ };
+ }
+#if !defined(HAVE_MKS_SHELL) && !defined(HAVE_CYGWIN_SHELL)
+ } else if (*p == '\\') {
+ backslash_count++;
+ } else {
+ backslash_count = 0;
+#endif
+ }
+
+ /*
+ * Copy the character.
+ */
+ *(command_line_i++) = *(p++);
+ }
+
+ if (*enclose_in_quotes_i) {
+#if !defined(HAVE_MKS_SHELL) && !defined(HAVE_CYGWIN_SHELL)
+ /*
+ * Add one \ for each \ that precedes the
+ * closing ".
+ */
+ while(backslash_count--) {
+ *(command_line_i++) = '\\';
+ };
+#endif
+#ifdef KMK
+ /*
+ * ash it put off by echo "hello world" ending up as:
+ * G:/.../kmk_ash.exe -c "echo ""hello world"""
+ * It wants a space before the last '"'.
+ * (The 'test_shell' goals in Makefile.kmk tests this problem.)
+ */
+ if (command_line_i[-1] == '\"' /* && cygwin_mode && have_sh*/ && !argvi[1]) {
+ *(command_line_i++) = ' ';
+ }
+#endif
+
+ *(command_line_i++) = '\"';
+ }
+
+ /*
+ * Append an intervening space.
+ */
+ if (*(++argvi)) {
+ *(command_line_i++) = ' ';
+ }
+
+ enclose_in_quotes_i++;
+ }
+
+ /*
+ * Append the terminating NULL.
+ */
+ *command_line_i = '\0';
+
+ free(enclose_in_quotes);
+ return command_line;
+}
+
+/*
+ * Description: Given an argv and optional envp, launch the process
+ * using the default stdin, stdout, and stderr handles.
+ * Also, register process so that process_wait_for_any_private()
+ * can be used via process_file_io(NULL) or
+ * process_wait_for_any().
+ *
+ * Returns:
+ *
+ * Notes/Dependencies:
+ */
+HANDLE
+process_easy(
+ char **argv,
+ char **envp,
+ int outfd,
+ int errfd)
+{
+ HANDLE hIn = INVALID_HANDLE_VALUE;
+ HANDLE hOut = INVALID_HANDLE_VALUE;
+ HANDLE hErr = INVALID_HANDLE_VALUE;
+ HANDLE hProcess, tmpIn, tmpOut, tmpErr;
+ DWORD e;
+
+ if (proc_index >= MAXIMUM_WAIT_OBJECTS) {
+ DB (DB_JOBS, ("process_easy: All process slots used up\n"));
+ return INVALID_HANDLE_VALUE;
+ }
+#ifdef KMK /* We can effort here by lettering CreateProcess/kernel do the standard handle duplication. */
+ if (outfd != -1 || errfd != -1) {
+#endif
+ /* Standard handles returned by GetStdHandle can be NULL or
+ INVALID_HANDLE_VALUE if the parent process closed them. If that
+ happens, we open the null device and pass its handle to
+ CreateProcess as the corresponding handle to inherit. */
+ tmpIn = GetStdHandle(STD_INPUT_HANDLE);
+ if (DuplicateHandle(GetCurrentProcess(),
+ tmpIn,
+ GetCurrentProcess(),
+ &hIn,
+ 0,
+ TRUE,
+ DUPLICATE_SAME_ACCESS) == FALSE) {
+ if ((e = GetLastError()) == ERROR_INVALID_HANDLE) {
+ tmpIn = CreateFile("NUL", GENERIC_READ,
+ FILE_SHARE_READ | FILE_SHARE_WRITE, NULL,
+ OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
+ if (tmpIn != INVALID_HANDLE_VALUE
+ && DuplicateHandle(GetCurrentProcess(),
+ tmpIn,
+ GetCurrentProcess(),
+ &hIn,
+ 0,
+ TRUE,
+ DUPLICATE_SAME_ACCESS) == FALSE)
+ CloseHandle(tmpIn);
+ }
+ if (hIn == INVALID_HANDLE_VALUE) {
+ fprintf(stderr, "process_easy: DuplicateHandle(In) failed (e=%ld)\n", e);
+ return INVALID_HANDLE_VALUE;
+ }
+ }
+ if (outfd >= 0)
+ tmpOut = (HANDLE)_get_osfhandle (outfd);
+ else
+ tmpOut = GetStdHandle (STD_OUTPUT_HANDLE);
+ if (DuplicateHandle(GetCurrentProcess(),
+ tmpOut,
+ GetCurrentProcess(),
+ &hOut,
+ 0,
+ TRUE,
+ DUPLICATE_SAME_ACCESS) == FALSE) {
+ if ((e = GetLastError()) == ERROR_INVALID_HANDLE) {
+ tmpOut = CreateFile("NUL", GENERIC_WRITE,
+ FILE_SHARE_READ | FILE_SHARE_WRITE, NULL,
+ OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
+ if (tmpOut != INVALID_HANDLE_VALUE
+ && DuplicateHandle(GetCurrentProcess(),
+ tmpOut,
+ GetCurrentProcess(),
+ &hOut,
+ 0,
+ TRUE,
+ DUPLICATE_SAME_ACCESS) == FALSE)
+ CloseHandle(tmpOut);
+ }
+ if (hOut == INVALID_HANDLE_VALUE) {
+ fprintf(stderr, "process_easy: DuplicateHandle(Out) failed (e=%ld)\n", e);
+ return INVALID_HANDLE_VALUE;
+ }
+ }
+ if (errfd >= 0)
+ tmpErr = (HANDLE)_get_osfhandle (errfd);
+ else
+ tmpErr = GetStdHandle(STD_ERROR_HANDLE);
+ if (DuplicateHandle(GetCurrentProcess(),
+ tmpErr,
+ GetCurrentProcess(),
+ &hErr,
+ 0,
+ TRUE,
+ DUPLICATE_SAME_ACCESS) == FALSE) {
+ if ((e = GetLastError()) == ERROR_INVALID_HANDLE) {
+ tmpErr = CreateFile("NUL", GENERIC_WRITE,
+ FILE_SHARE_READ | FILE_SHARE_WRITE, NULL,
+ OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
+ if (tmpErr != INVALID_HANDLE_VALUE
+ && DuplicateHandle(GetCurrentProcess(),
+ tmpErr,
+ GetCurrentProcess(),
+ &hErr,
+ 0,
+ TRUE,
+ DUPLICATE_SAME_ACCESS) == FALSE)
+ CloseHandle(tmpErr);
+ }
+ if (hErr == INVALID_HANDLE_VALUE) {
+ fprintf(stderr, "process_easy: DuplicateHandle(Err) failed (e=%ld)\n", e);
+ return INVALID_HANDLE_VALUE;
+ }
+ }
+#ifdef KMK /* saving effort */
+ }
+#endif
+
+ hProcess = process_init_fd(hIn, hOut, hErr);
+
+ if (process_begin(hProcess, argv, envp, argv[0], NULL)) {
+ fake_exits_pending++;
+ /* process_begin() failed: make a note of that. */
+ if (!((sub_process*) hProcess)->last_err)
+ ((sub_process*) hProcess)->last_err = -1;
+#ifdef KMK
+ if (!((sub_process*) hProcess)->exit_code)
+#endif
+ ((sub_process*) hProcess)->exit_code = process_last_err(hProcess);
+
+ /* close up unused handles */
+ if (hIn != INVALID_HANDLE_VALUE)
+ CloseHandle(hIn);
+ if (hOut != INVALID_HANDLE_VALUE)
+ CloseHandle(hOut);
+ if (hErr != INVALID_HANDLE_VALUE)
+ CloseHandle(hErr);
+ }
+
+ process_register(hProcess);
+
+ return hProcess;
+}
diff --git a/src/kmk/w32/subproc/w32err.c b/src/kmk/w32/subproc/w32err.c
new file mode 100644
index 0000000..14eebed
--- /dev/null
+++ b/src/kmk/w32/subproc/w32err.c
@@ -0,0 +1,85 @@
+/* Error handling for Windows
+Copyright (C) 1996-2016 Free Software Foundation, Inc.
+This file is part of GNU Make.
+
+GNU Make 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 3 of the License, or (at your option) any later
+version.
+
+GNU Make 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 <stdlib.h>
+#include <windows.h>
+#include "makeint.h"
+#include "w32err.h"
+
+/*
+ * Description: the windows32 version of perror()
+ *
+ * Returns: a pointer to a static error
+ *
+ * Notes/Dependencies: I got this from
+ * comp.os.ms-windows.programmer.win32
+ */
+const char *
+map_windows32_error_to_string (DWORD ercode) {
+/*
+ * We used to have an MSVC-specific '__declspec (thread)' qualifier
+ * here, with the following comment:
+ *
+ * __declspec (thread) necessary if you will use multiple threads on MSVC
+ *
+ * However, Make was never multithreaded on Windows (except when
+ * Ctrl-C is hit, in which case the main thread is stopped
+ * immediately, so it doesn't matter in this context). The functions
+ * on sub_proc.c that started and stopped additional threads were
+ * never used, and are now #ifdef'ed away. Until we need more than
+ * one thread, we have no problems with the following buffer being
+ * static. (If and when we do need it to be in thread-local storage,
+ * the corresponding GCC qualifier is '__thread'.)
+ */
+ static char szMessageBuffer[128];
+ /* Fill message buffer with a default message in
+ * case FormatMessage fails
+ */
+ wsprintf (szMessageBuffer, "Error %ld\n", ercode);
+
+ /*
+ * Special code for winsock error handling.
+ */
+ if (ercode > WSABASEERR) {
+#if 0
+ HMODULE hModule = GetModuleHandle("wsock32");
+ if (hModule != NULL) {
+ FormatMessage(FORMAT_MESSAGE_FROM_HMODULE,
+ hModule,
+ ercode,
+ LANG_NEUTRAL,
+ szMessageBuffer,
+ sizeof(szMessageBuffer),
+ NULL);
+ FreeLibrary(hModule);
+ }
+#else
+ O (fatal, NILF, szMessageBuffer);
+#endif
+ } else {
+ /*
+ * Default system message handling
+ */
+ FormatMessage(FORMAT_MESSAGE_FROM_SYSTEM,
+ NULL,
+ ercode,
+ LANG_NEUTRAL,
+ szMessageBuffer,
+ sizeof(szMessageBuffer),
+ NULL);
+ }
+ return szMessageBuffer;
+}
diff --git a/src/kmk/w32/tstFileInfo.c b/src/kmk/w32/tstFileInfo.c
new file mode 100644
index 0000000..94d47bb
--- /dev/null
+++ b/src/kmk/w32/tstFileInfo.c
@@ -0,0 +1,151 @@
+/* $Id: $ */
+/** @file
+ * Test program for some NtQueryInformationFile functionality.
+ */
+
+/*
+ * Copyright (c) 2007-2010 knut st. osmundsen <bird-kBuild-spamx@anduin.net>
+ *
+ * This file is part of kBuild.
+ *
+ * kBuild 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 3 of the License, or
+ * (at your option) any later version.
+ *
+ * kBuild 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 kBuild. If not, see <http://www.gnu.org/licenses/>
+ *
+ */
+
+
+
+#include <stdio.h>
+#include <Windows.h>
+
+typedef enum _FILE_INFORMATION_CLASS {
+ FileDirectoryInformation = 1,
+ FileFullDirectoryInformation, // 2
+ FileBothDirectoryInformation, // 3
+ FileBasicInformation, // 4 wdm
+ FileStandardInformation, // 5 wdm
+ FileInternalInformation, // 6
+ FileEaInformation, // 7
+ FileAccessInformation, // 8
+ FileNameInformation, // 9
+ FileRenameInformation, // 10
+ FileLinkInformation, // 11
+ FileNamesInformation, // 12
+ FileDispositionInformation, // 13
+ FilePositionInformation, // 14 wdm
+ FileFullEaInformation, // 15
+ FileModeInformation, // 16
+ FileAlignmentInformation, // 17
+ FileAllInformation, // 18
+ FileAllocationInformation, // 19
+ FileEndOfFileInformation, // 20 wdm
+ FileAlternateNameInformation, // 21
+ FileStreamInformation, // 22
+ FilePipeInformation, // 23
+ FilePipeLocalInformation, // 24
+ FilePipeRemoteInformation, // 25
+ FileMailslotQueryInformation, // 26
+ FileMailslotSetInformation, // 27
+ FileCompressionInformation, // 28
+ FileObjectIdInformation, // 29
+ FileCompletionInformation, // 30
+ FileMoveClusterInformation, // 31
+ FileQuotaInformation, // 32
+ FileReparsePointInformation, // 33
+ FileNetworkOpenInformation, // 34
+ FileAttributeTagInformation, // 35
+ FileTrackingInformation, // 36
+ FileIdBothDirectoryInformation, // 37
+ FileIdFullDirectoryInformation, // 38
+ FileMaximumInformation
+} FILE_INFORMATION_CLASS, *PFILE_INFORMATION_CLASS;
+
+typedef struct _FILE_NAME_INFORMATION
+{
+ ULONG FileNameLength;
+ WCHAR FileName[1];
+} FILE_NAME_INFORMATION, *PFILE_NAME_INFORMATION;
+
+typedef LONG NTSTATUS;
+
+typedef struct _IO_STATUS_BLOCK
+{
+ union
+ {
+ NTSTATUS Status;
+ PVOID Pointer;
+ };
+ ULONG_PTR Information;
+} IO_STATUS_BLOCK, *PIO_STATUS_BLOCK;
+
+NTSYSAPI
+NTSTATUS
+NTAPI
+NtQueryInformationFile(HANDLE FileHandle, PIO_STATUS_BLOCK IoStatusBlock, PVOID FileInformation,
+ ULONG Length, FILE_INFORMATION_CLASS FileInformationClass);
+
+
+
+int main(int argc, char **argv)
+{
+ int rc = 0;
+ int i;
+
+ NTSTATUS (NTAPI *pfnNtQueryInformationFile)(HANDLE FileHandle, PIO_STATUS_BLOCK IoStatusBlock, PVOID FileInformation,
+ ULONG Length, FILE_INFORMATION_CLASS FileInformationClass);
+
+ pfnNtQueryInformationFile = GetProcAddress(LoadLibrary("ntdll.dll"), "NtQueryInformationFile");
+
+ for (i = 1; i < argc; i++)
+ {
+ HANDLE hFile = CreateFile(argv[i],
+ GENERIC_READ,
+ FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE,
+ NULL,
+ OPEN_EXISTING,
+ FILE_FLAG_BACKUP_SEMANTICS,
+ NULL);
+ if (hFile)
+ {
+ long rcNt;
+ char abBuf[4096];
+ IO_STATUS_BLOCK Ios;
+
+ memset(abBuf, 0, sizeof(abBuf));
+ memset(&Ios, 0, sizeof(Ios));
+ rcNt = pfnNtQueryInformationFile(hFile, &Ios, abBuf, sizeof(abBuf), FileNameInformation);
+ if (rcNt >= 0)
+ {
+ PFILE_NAME_INFORMATION pFileNameInfo = (PFILE_NAME_INFORMATION)abBuf;
+ printf("#%d: %s - rcNt=%#x - FileNameInformation:\n"
+ " FileName: %ls\n"
+ " FileNameLength: %lu\n",
+ i, argv[i], rcNt,
+ pFileNameInfo->FileName,
+ pFileNameInfo->FileNameLength
+ );
+ }
+ else
+ printf("#%d: %s - rcNt=%#x - FileNameInformation!\n", i, argv[i], rcNt);
+
+ CloseHandle(hFile);
+ }
+ else
+ {
+ printf("#%d: %s - open failed, last error %d\n", i, argv[i], GetLastError());
+ rc = 1;
+ }
+ }
+ return rc;
+}
+
diff --git a/src/kmk/w32/w32os.c b/src/kmk/w32/w32os.c
new file mode 100644
index 0000000..fd30661
--- /dev/null
+++ b/src/kmk/w32/w32os.c
@@ -0,0 +1,220 @@
+/* Windows32-based operating system interface for GNU Make.
+Copyright (C) 2016 Free Software Foundation, Inc.
+This file is part of GNU Make.
+
+GNU Make 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 3 of the License, or (at your option) any later
+version.
+
+GNU Make 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 "makeint.h"
+
+#include <stdio.h>
+#include <string.h>
+
+#include <windows.h>
+#include <process.h>
+#include <io.h>
+#include "pathstuff.h"
+#ifndef CONFIG_NEW_WIN_CHILDREN
+# include "sub_proc.h"
+#else
+# include "winchildren.h"
+#endif
+#include "w32err.h"
+#include "os.h"
+#include "debug.h"
+
+/* This section provides OS-specific functions to support the jobserver. */
+
+static char jobserver_semaphore_name[MAX_PATH + 1];
+static HANDLE jobserver_semaphore = NULL;
+
+unsigned int
+jobserver_setup (int slots)
+{
+#ifndef CONFIG_NEW_WIN_CHILDREN
+ /* sub_proc.c cannot wait for more than MAXIMUM_WAIT_OBJECTS objects
+ * and one of them is the job-server semaphore object. Limit the
+ * number of available job slots to (MAXIMUM_WAIT_OBJECTS - 1). */
+
+ if (slots >= MAXIMUM_WAIT_OBJECTS)
+ {
+ slots = MAXIMUM_WAIT_OBJECTS - 1;
+ DB (DB_JOBS, (_("Jobserver slots limited to %d\n"), slots));
+ }
+#endif
+
+ sprintf (jobserver_semaphore_name, "gmake_semaphore_%d", _getpid ());
+
+ jobserver_semaphore = CreateSemaphore (
+ NULL, /* Use default security descriptor */
+ slots, /* Initial count */
+ slots, /* Maximum count */
+ jobserver_semaphore_name); /* Semaphore name */
+
+ if (jobserver_semaphore == NULL)
+ {
+ DWORD err = GetLastError ();
+ const char *estr = map_windows32_error_to_string (err);
+ ONS (fatal, NILF,
+ _("creating jobserver semaphore: (Error %ld: %s)"), err, estr);
+ }
+
+ return 1;
+}
+
+unsigned int
+jobserver_parse_auth (const char *auth)
+{
+ jobserver_semaphore = OpenSemaphore (
+ SEMAPHORE_ALL_ACCESS, /* Semaphore access setting */
+ FALSE, /* Child processes DON'T inherit */
+ auth); /* Semaphore name */
+
+ if (jobserver_semaphore == NULL)
+ {
+ DWORD err = GetLastError ();
+ const char *estr = map_windows32_error_to_string (err);
+ fatal (NILF, strlen (auth) + INTSTR_LENGTH + strlen (estr),
+ _("internal error: unable to open jobserver semaphore '%s': (Error %ld: %s)"),
+ auth, err, estr);
+ }
+ DB (DB_JOBS, (_("Jobserver client (semaphore %s)\n"), auth));
+
+ return 1;
+}
+
+char *
+jobserver_get_auth ()
+{
+ return xstrdup (jobserver_semaphore_name);
+}
+
+unsigned int
+jobserver_enabled ()
+{
+ return jobserver_semaphore != NULL;
+}
+
+/* Close jobserver semaphore */
+void
+jobserver_clear ()
+{
+ if (jobserver_semaphore != NULL)
+ {
+ CloseHandle (jobserver_semaphore);
+ jobserver_semaphore = NULL;
+ }
+}
+
+void
+jobserver_release (int is_fatal)
+{
+ if (! ReleaseSemaphore (
+ jobserver_semaphore, /* handle to semaphore */
+ 1, /* increase count by one */
+ NULL)) /* not interested in previous count */
+ {
+ if (is_fatal)
+ {
+ DWORD err = GetLastError ();
+ const char *estr = map_windows32_error_to_string (err);
+ ONS (fatal, NILF,
+ _("release jobserver semaphore: (Error %ld: %s)"), err, estr);
+ }
+ perror_with_name ("release_jobserver_semaphore", "");
+ }
+}
+
+unsigned int
+jobserver_acquire_all ()
+{
+ unsigned int tokens = 0;
+ while (1)
+ {
+ DWORD dwEvent = WaitForSingleObject (
+ jobserver_semaphore, /* Handle to semaphore */
+ 0); /* DON'T wait on semaphore */
+
+ if (dwEvent != WAIT_OBJECT_0)
+ return tokens;
+
+ ++tokens;
+ }
+}
+
+void
+jobserver_signal ()
+{
+}
+
+void jobserver_pre_child (int recursive)
+{
+}
+
+void jobserver_post_child (int recursive)
+{
+}
+
+void
+jobserver_pre_acquire ()
+{
+}
+
+/* Returns 1 if we got a token, or 0 if a child has completed.
+ The Windows implementation doesn't support load detection. */
+unsigned int
+jobserver_acquire (int timeout)
+{
+#ifndef CONFIG_NEW_WIN_CHILDREN
+ HANDLE handles[MAXIMUM_WAIT_OBJECTS + 1]; /* bird: + 1 to prevent trashing the stack. */
+#else
+ HANDLE handles[2];
+#endif
+ DWORD dwHandleCount;
+ DWORD dwEvent;
+
+ /* Add jobserver semaphore to first slot. */
+ handles[0] = jobserver_semaphore;
+
+#ifndef CONFIG_NEW_WIN_CHILDREN
+ /* Build array of handles to wait for. */
+ dwHandleCount = 1 + process_set_handles (&handles[1]);
+ dwEvent = WaitForMultipleObjects (
+ dwHandleCount, /* number of objects in array */
+ handles, /* array of objects */
+ FALSE, /* wait for any object */
+ INFINITE); /* wait until object is signalled */
+#else
+ /* Add the completed children event as the 2nd one. */
+ handles[1] = (HANDLE)MkWinChildGetCompleteEventHandle ();
+ if (handles[1] == NULL)
+ return 0;
+ dwHandleCount = 2;
+ dwEvent = WaitForMultipleObjectsEx (dwHandleCount,
+ handles,
+ FALSE /*bWaitAll*/,
+ 256, /* INFINITE - paranoia, only wait 256 ms before checking again. */
+ TRUE /*bAlertable*/);
+#endif
+
+ if (dwEvent == WAIT_FAILED)
+ {
+ DWORD err = GetLastError ();
+ const char *estr = map_windows32_error_to_string (err);
+ ONS (fatal, NILF,
+ _("semaphore or child process wait: (Error %ld: %s)"),
+ err, estr);
+ }
+
+ /* WAIT_OBJECT_0 indicates that the semaphore was signalled. */
+ return dwEvent == WAIT_OBJECT_0;
+}
diff --git a/src/kmk/w32/winchildren.c b/src/kmk/w32/winchildren.c
new file mode 100644
index 0000000..635fe99
--- /dev/null
+++ b/src/kmk/w32/winchildren.c
@@ -0,0 +1,3729 @@
+/* $Id: winchildren.c 3359 2020-06-05 16:17:17Z bird $ */
+/** @file
+ * Child process creation and management for kmk.
+ */
+
+/*
+ * Copyright (c) 2018 knut st. osmundsen <bird-kBuild-spamx@anduin.net>
+ *
+ * This file is part of kBuild.
+ *
+ * kBuild 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 3 of the License, or
+ * (at your option) any later version.
+ *
+ * kBuild 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 kBuild. If not, see <http://www.gnu.org/licenses/>
+ *
+ */
+
+/* No GNU coding style here atm, convert if upstreamed. */
+
+/** @page pg_win_children Windows child process creation and managment
+ *
+ * This new implementation aims at addressing the following:
+ *
+ * 1. Speed up process creation by doing the expensive CreateProcess call
+ * in a worker thread.
+ *
+ * 2. No 64 process limit imposed by WaitForMultipleObjects.
+ *
+ * 3. Better distribute jobs among processor groups.
+ *
+ * 4. Offloading more expensive kmkbuiltin operations to worker threads,
+ * making the main thread focus on managing child processes.
+ *
+ * 5. Output synchronization using reusable pipes.
+ *
+ *
+ * To be quite honest, the first item (CreateProcess expense) didn't occur to me
+ * at first and was more of a sideeffect discovered along the way. A test
+ * rebuilding IPRT went from 4m52s to 3m19s on a 8 thread system.
+ *
+ * The 2nd and 3rd goals are related to newer build servers that have lots of
+ * CPU threads and various Windows NT (aka NT OS/2 at the time) design choices
+ * made in the late 1980ies.
+ *
+ * WaitForMultipleObjects does not support waiting for more than 64 objects,
+ * unlike poll and select. This is just something everyone ends up having to
+ * work around in the end.
+ *
+ * Affinity masks are uintptr_t sized, so 64-bit hosts can only manage 64
+ * processors and 32-bit only 32. Workaround was introduced with Windows 7
+ * (IIRC) and is called processor groups. The CPU threads are grouped into 1 or
+ * more groups of up to 64 processors. Processes are generally scheduled to a
+ * signle processor group at first, but threads may be changed to be scheduled
+ * on different groups. This code will try distribute children evenly among the
+ * processor groups, using a very simple algorithm (see details in code).
+ *
+ *
+ * @section sec_win_children_av Remarks on Microsoft Defender and other AV
+ *
+ * Part of the motivation for writing this code was horrible CPU utilization on
+ * a brand new AMD Threadripper 1950X system with lots of memory and SSDs,
+ * running 64-bit Windows 10 build 16299.
+ *
+ * Turns out Microsoft defender adds some overhead to CreateProcess
+ * and other stuff:
+ * - Old make with CreateProcess on main thread:
+ * - With runtime defender enabled: 14 min 6 seconds
+ * - With runtime defender disabled: 4 min 49 seconds
+ * - New make with CreateProcess on worker thread (this code):
+ * - With runtime defender enabled: 6 min 29 seconds
+ * - With runtime defender disabled: 4 min 36 seconds
+ * - With runtime defender disabled out dir only: 5 min 59 seconds
+ *
+ * See also kWorker / kSubmit for more bickering about AV & disk encryption.
+ */
+
+
+/*********************************************************************************************************************************
+* Header Files *
+*********************************************************************************************************************************/
+#include <Windows.h>
+#include <Winternl.h>
+
+#include "../makeint.h"
+#include "../job.h"
+#include "../filedef.h"
+#include "../debug.h"
+#include "../kmkbuiltin.h"
+#include "winchildren.h"
+
+#include <assert.h>
+#include <process.h>
+#include <intrin.h>
+
+#include "nt/nt_child_inject_standard_handles.h"
+#include "console.h"
+
+#ifndef KMK_BUILTIN_STANDALONE
+extern void kmk_cache_exec_image_w(const wchar_t *); /* imagecache.c */
+#endif
+
+/* Option values from main.c: */
+extern const char *win_job_object_mode;
+extern const char *win_job_object_name;
+extern int win_job_object_no_kill;
+
+
+/*********************************************************************************************************************************
+* Defined Constants And Macros *
+*********************************************************************************************************************************/
+#define MKWINCHILD_MAX_PATH 1024
+
+#define MKWINCHILD_DO_SET_PROCESSOR_GROUP
+
+/** Checks the UTF-16 environment variable pointed to is the PATH. */
+#define IS_PATH_ENV_VAR(a_cwcVar, a_pwszVar) \
+ ( (a_cwcVar) >= 5 \
+ && (a_pwszVar)[4] == L'=' \
+ && ((a_pwszVar)[0] == L'P' || (a_pwszVar)[0] == L'p') \
+ && ((a_pwszVar)[1] == L'A' || (a_pwszVar)[1] == L'a') \
+ && ((a_pwszVar)[2] == L'T' || (a_pwszVar)[2] == L't') \
+ && ((a_pwszVar)[3] == L'H' || (a_pwszVar)[3] == L'h') )
+
+
+/*********************************************************************************************************************************
+* Structures and Typedefs *
+*********************************************************************************************************************************/
+/** Pointer to a childcare worker thread. */
+typedef struct WINCHILDCAREWORKER *PWINCHILDCAREWORKER;
+/** Pointer to a windows child process. */
+typedef struct WINCHILD *PWINCHILD;
+
+
+/**
+ * Child process type.
+ */
+typedef enum WINCHILDTYPE
+{
+ WINCHILDTYPE_INVALID = 0,
+ /** Normal child process. */
+ WINCHILDTYPE_PROCESS,
+#ifdef KMK
+ /** kmkbuiltin command. */
+ WINCHILDTYPE_BUILT_IN,
+ /** kmkbuiltin_append result write out. */
+ WINCHILDTYPE_APPEND,
+ /** kSubmit job. */
+ WINCHILDTYPE_SUBMIT,
+ /** kmk_redirect job. */
+ WINCHILDTYPE_REDIRECT,
+#endif
+ /** End of valid child types. */
+ WINCHILDTYPE_END
+} WINCHILDTYPE;
+
+
+/**
+ * Windows child process.
+ */
+typedef struct WINCHILD
+{
+ /** Magic / eyecatcher (WINCHILD_MAGIC). */
+ ULONG uMagic;
+ /** Child type. */
+ WINCHILDTYPE enmType;
+ /** Pointer to the next child process. */
+ PWINCHILD pNext;
+ /** The pid for this child. */
+ pid_t pid;
+ /** The make child structure associated with this child. */
+ struct child *pMkChild;
+
+ /** The process exit code. */
+ int iExitCode;
+ /** Kill signal, in case we or someone else killed it. */
+ int iSignal;
+ /** Set if core was dumped. */
+ int fCoreDumped;
+ /** Set if the a child process is a candidate for cl.exe where we supress
+ * annoying source name output. */
+ BOOL fProbableClExe;
+ /** The worker executing this child. */
+ PWINCHILDCAREWORKER pWorker;
+
+ /** Type specific data. */
+ union
+ {
+ /** Data for WINCHILDTYPE_PROCESS. */
+ struct
+ {
+ /** Argument vector (single allocation, strings following array). */
+ char **papszArgs;
+ /** Length of the argument strings. */
+ size_t cbArgsStrings;
+ /** Environment vector. Only a copy if fEnvIsCopy is set. */
+ char **papszEnv;
+ /** If we made a copy of the environment, this is the size of the
+ * strings and terminator string (not in array). This is done to
+ * speed up conversion, since MultiByteToWideChar can handle '\0'. */
+ size_t cbEnvStrings;
+ /** The make shell to use (copy). */
+ char *pszShell;
+ /** Handle to use for standard out. */
+ HANDLE hStdOut;
+ /** Handle to use for standard out. */
+ HANDLE hStdErr;
+ /** Whether to close hStdOut after creating the process. */
+ BOOL fCloseStdOut;
+ /** Whether to close hStdErr after creating the process. */
+ BOOL fCloseStdErr;
+ /** Whether to catch output from the process. */
+ BOOL fCatchOutput;
+
+ /** Child process handle. */
+ HANDLE hProcess;
+ } Process;
+
+ /** Data for WINCHILDTYPE_BUILT_IN. */
+ struct
+ {
+ /** The built-in command. */
+ PCKMKBUILTINENTRY pBuiltIn;
+ /** Number of arguments. */
+ int cArgs;
+ /** Argument vector (single allocation, strings following array). */
+ char **papszArgs;
+ /** Environment vector. Only a copy if fEnvIsCopy is set. */
+ char **papszEnv;
+ } BuiltIn;
+
+ /** Data for WINCHILDTYPE_APPEND. */
+ struct
+ {
+ /** The filename. */
+ char *pszFilename;
+ /** How much to append. */
+ size_t cbAppend;
+ /** What to append. */
+ char *pszAppend;
+ /** Whether to truncate the file. */
+ int fTruncate;
+ } Append;
+
+ /** Data for WINCHILDTYPE_SUBMIT. */
+ struct
+ {
+ /** The event we're to wait on (hooked up to a pipe) */
+ HANDLE hEvent;
+ /** Parameter for the cleanup callback. */
+ void *pvSubmitWorker;
+ /** Standard output catching pipe. Optional. */
+ PWINCCWPIPE pStdOut;
+ /** Standard error catching pipe. Optional. */
+ PWINCCWPIPE pStdErr;
+ } Submit;
+
+ /** Data for WINCHILDTYPE_REDIRECT. */
+ struct
+ {
+ /** Child process handle. */
+ HANDLE hProcess;
+ } Redirect;
+ } u;
+
+} WINCHILD;
+/** WINCHILD::uMagic value. */
+#define WINCHILD_MAGIC 0xbabebabeU
+
+
+/**
+ * Data for a windows childcare worker thread.
+ *
+ * We use one worker thread per child, reusing the threads when possible.
+ *
+ * This setup helps avoid the 64-bit handle with the WaitForMultipleObject API.
+ *
+ * It also helps using all CPUs on systems with more than one CPU group
+ * (typically systems with more than 64 CPU threads or/and multiple sockets, or
+ * special configs).
+ *
+ * This helps facilitates using pipes for collecting output child rather
+ * than temporary files. Pipes doesn't involve NTFS and can easily be reused.
+ *
+ * Finally, kBuild specific, this allows running kmkbuiltin_xxxx commands in
+ * threads.
+ */
+typedef struct WINCHILDCAREWORKER
+{
+ /** Magic / eyecatcher (WINCHILDCAREWORKER_MAGIC). */
+ ULONG uMagic;
+ /** The worker index. */
+ unsigned int idxWorker;
+ /** The processor group for this worker. */
+ unsigned int iProcessorGroup;
+ /** The thread ID. */
+ unsigned int tid;
+ /** The thread handle. */
+ HANDLE hThread;
+ /** The event the thread is idling on. */
+ HANDLE hEvtIdle;
+ /** The pipe catching standard output from a child. */
+ PWINCCWPIPE pStdOut;
+ /** The pipe catching standard error from a child. */
+ PWINCCWPIPE pStdErr;
+
+ /** Pointer to the current child. */
+ PWINCHILD volatile pCurChild;
+ /** List of children pending execution on this worker.
+ * This is updated atomitically just like g_pTailCompletedChildren. */
+ PWINCHILD volatile pTailTodoChildren;
+ /** TRUE if idle, FALSE if not. */
+ long volatile fIdle;
+} WINCHILDCAREWORKER;
+/** WINCHILD::uMagic value. */
+#define WINCHILDCAREWORKER_MAGIC 0xdad0dad0U
+
+
+/*********************************************************************************************************************************
+* Global Variables *
+*********************************************************************************************************************************/
+/** Whether it's initialized or not. */
+static BOOL g_fInitialized = FALSE;
+/** Set when we're shutting down everything. */
+static BOOL volatile g_fShutdown = FALSE;
+/** Event used to wait for children. */
+static HANDLE g_hEvtWaitChildren = INVALID_HANDLE_VALUE;
+/** Number of childcare workers currently in g_papChildCareworkers. */
+static unsigned g_cChildCareworkers = 0;
+/** Maximum number of childcare workers in g_papChildCareworkers. */
+static unsigned g_cChildCareworkersMax = 0;
+/** Pointer to childcare workers. */
+static PWINCHILDCAREWORKER *g_papChildCareworkers = NULL;
+/** The processor group allocator state. */
+static MKWINCHILDCPUGROUPALLOCSTATE g_ProcessorGroupAllocator;
+/** Number of processor groups in the system. */
+static unsigned g_cProcessorGroups = 1;
+/** Array detailing how many active processors there are in each group. */
+static unsigned const *g_pacProcessorsInGroup = &g_cProcessorGroups;
+/** Kernel32!GetActiveProcessorGroupCount */
+static WORD (WINAPI *g_pfnGetActiveProcessorGroupCount)(VOID);
+/** Kernel32!GetActiveProcessorCount */
+static DWORD (WINAPI *g_pfnGetActiveProcessorCount)(WORD);
+/** Kernel32!SetThreadGroupAffinity */
+static BOOL (WINAPI *g_pfnSetThreadGroupAffinity)(HANDLE, CONST GROUP_AFFINITY *, GROUP_AFFINITY *);
+/** NTDLL!NtQueryInformationProcess */
+static NTSTATUS (NTAPI *g_pfnNtQueryInformationProcess)(HANDLE, PROCESSINFOCLASS, PVOID, ULONG, PULONG);
+/** Set if the windows host is 64-bit. */
+static BOOL g_f64BitHost = (K_ARCH_BITS == 64);
+/** Windows version info.
+ * @note Putting this before the volatile stuff, hoping to keep it in a
+ * different cache line than the static bits above. */
+static OSVERSIONINFOA g_VersionInfo = { sizeof(g_VersionInfo), 4, 0, 1381, VER_PLATFORM_WIN32_NT, {0} };
+
+/** Children that has been completed.
+ * This is updated atomically, pushing completed children in LIFO fashion
+ * (thus 'tail'), then hitting g_hEvtWaitChildren if head. */
+static PWINCHILD volatile g_pTailCompletedChildren = NULL;
+
+/** Number of idle pending children.
+ * This is updated before g_hEvtWaitChildren is signalled. */
+static unsigned volatile g_cPendingChildren = 0;
+
+/** Number of idle childcare worker threads. */
+static unsigned volatile g_cIdleChildcareWorkers = 0;
+/** Index of the last idle child careworker (just a hint). */
+static unsigned volatile g_idxLastChildcareWorker = 0;
+
+#ifdef WITH_RW_LOCK
+/** RW lock for serializing kmkbuiltin_redirect and CreateProcess. */
+static SRWLOCK g_RWLock;
+#endif
+
+/** The job object for this make instance, if we created/opened one. */
+static HANDLE g_hJob = NULL;
+
+
+/*********************************************************************************************************************************
+* Internal Functions *
+*********************************************************************************************************************************/
+static void mkWinChildInitJobObjectAssociation(void);
+
+#if K_ARCH_BITS == 32 && !defined(_InterlockedCompareExchangePointer)
+/** _InterlockedCompareExchangePointer is missing? (VS2010) */
+K_INLINE void *_InterlockedCompareExchangePointer(void * volatile *ppvDst, void *pvNew, void *pvOld)
+{
+ return (void *)_InterlockedCompareExchange((long volatile *)ppvDst, (intptr_t)pvNew, (intptr_t)pvOld);
+}
+#endif
+
+
+/**
+ * Initializes the windows child module.
+ *
+ * @param cJobSlots The number of job slots.
+ */
+void MkWinChildInit(unsigned int cJobSlots)
+{
+ HMODULE hmod;
+
+ /*
+ * Figure out how many childcare workers first.
+ */
+ static unsigned int const s_cMaxWorkers = 4096;
+ unsigned cWorkers;
+ if (cJobSlots >= 1 && cJobSlots < s_cMaxWorkers)
+ cWorkers = cJobSlots;
+ else
+ cWorkers = s_cMaxWorkers;
+
+ /*
+ * Allocate the array and the child completed event object.
+ */
+ g_papChildCareworkers = (PWINCHILDCAREWORKER *)xcalloc(cWorkers * sizeof(g_papChildCareworkers[0]));
+ g_cChildCareworkersMax = cWorkers;
+
+ g_hEvtWaitChildren = CreateEvent(NULL, FALSE /*fManualReset*/, FALSE /*fInitialState*/, NULL /*pszName*/);
+ if (!g_hEvtWaitChildren)
+ fatal(NILF, INTSTR_LENGTH, _("MkWinChildInit: CreateEvent failed: %u"), GetLastError());
+
+ /*
+ * NTDLL imports that we need.
+ */
+ hmod = GetModuleHandleA("NTDLL.DLL");
+ *(FARPROC *)&g_pfnNtQueryInformationProcess = GetProcAddress(hmod, "NtQueryInformationProcess");
+ if (!g_pfnNtQueryInformationProcess)
+ fatal(NILF, 0, _("MkWinChildInit: NtQueryInformationProcess not found"));
+
+#if K_ARCH_BITS == 32
+ /*
+ * Initialize g_f64BitHost.
+ */
+ if (!IsWow64Process(GetCurrentProcess(), &g_f64BitHost))
+ fatal(NILF, INTSTR_LENGTH, _("MkWinChildInit: IsWow64Process failed: %u"), GetLastError());
+#elif K_ARCH_BITS == 64
+ assert(g_f64BitHost);
+#else
+# error "K_ARCH_BITS is bad/missing"
+#endif
+
+ /*
+ * Figure out how many processor groups there are.
+ * For that we need to first figure the windows version.
+ */
+ if (!GetVersionExA(&g_VersionInfo))
+ {
+ DWORD uRawVer = GetVersion();
+ g_VersionInfo.dwMajorVersion = uRawVer & 0xff;
+ g_VersionInfo.dwMinorVersion = (uRawVer >> 8) & 0xff;
+ g_VersionInfo.dwBuildNumber = (uRawVer >> 16) & 0x7fff;
+ }
+ if (g_VersionInfo.dwMajorVersion >= 6)
+ {
+ hmod = GetModuleHandleA("KERNEL32.DLL");
+ *(FARPROC *)&g_pfnGetActiveProcessorGroupCount = GetProcAddress(hmod, "GetActiveProcessorGroupCount");
+ *(FARPROC *)&g_pfnGetActiveProcessorCount = GetProcAddress(hmod, "GetActiveProcessorCount");
+ *(FARPROC *)&g_pfnSetThreadGroupAffinity = GetProcAddress(hmod, "SetThreadGroupAffinity");
+ if ( g_pfnSetThreadGroupAffinity
+ && g_pfnGetActiveProcessorCount
+ && g_pfnGetActiveProcessorGroupCount)
+ {
+ unsigned int *pacProcessorsInGroup;
+ unsigned iGroup;
+ g_cProcessorGroups = g_pfnGetActiveProcessorGroupCount();
+ if (g_cProcessorGroups == 0)
+ g_cProcessorGroups = 1;
+
+ pacProcessorsInGroup = (unsigned int *)xmalloc(sizeof(g_pacProcessorsInGroup[0]) * g_cProcessorGroups);
+ g_pacProcessorsInGroup = pacProcessorsInGroup;
+ for (iGroup = 0; iGroup < g_cProcessorGroups; iGroup++)
+ pacProcessorsInGroup[iGroup] = g_pfnGetActiveProcessorCount(iGroup);
+
+ MkWinChildInitCpuGroupAllocator(&g_ProcessorGroupAllocator);
+ }
+ else
+ {
+ g_pfnSetThreadGroupAffinity = NULL;
+ g_pfnGetActiveProcessorCount = NULL;
+ g_pfnGetActiveProcessorGroupCount = NULL;
+ }
+ }
+
+#ifdef WITH_RW_LOCK
+ /*
+ * For serializing with standard file handle manipulation (kmkbuiltin_redirect).
+ */
+ InitializeSRWLock(&g_RWLock);
+#endif
+
+ /*
+ * Associate with a job object.
+ */
+ mkWinChildInitJobObjectAssociation();
+
+ /*
+ * This is dead code that was thought to fix a problem observed doing
+ * `tcc.exe /c "kmk |& tee bld.log"` and leading to a crash in cl.exe
+ * when spawned with fInheritHandles = FALSE, see hStdErr=NULL in the
+ * child. However, it turns out this was probably caused by not clearing
+ * the CRT file descriptor and handle table in the startup info.
+ * Leaving the code here in case it comes in handy after all.
+ */
+#if 0
+ {
+ struct
+ {
+ DWORD uStdHandle;
+ HANDLE hHandle;
+ } aHandles[3] = { { STD_INPUT_HANDLE, NULL }, { STD_OUTPUT_HANDLE, NULL }, { STD_ERROR_HANDLE, NULL } };
+ int i;
+
+ for (i = 0; i < 3; i++)
+ aHandles[i].hHandle = GetStdHandle(aHandles[i].uStdHandle);
+
+ for (i = 0; i < 3; i++)
+ if ( aHandles[i].hHandle == NULL
+ || aHandles[i].hHandle == INVALID_HANDLE_VALUE)
+ {
+ int fd = open("nul", _O_RDWR);
+ if (fd >= 0)
+ {
+ if (_dup2(fd, i) >= 0)
+ {
+ assert((HANDLE)_get_osfhandle(i) != aHandles[i].hHandle);
+ assert((HANDLE)_get_osfhandle(i) == GetStdHandle(aHandles[i].uStdHandle));
+ }
+ else
+ ONNNS(fatal, NILF, "_dup2(%d('nul'), %d) failed: %u (%s)", fd, i, errno, strerror(errno));
+ if (fd != i)
+ close(fd);
+ }
+ else
+ ONNS(fatal, NILF, "open(nul,RW) failed: %u (%s)", i, errno, strerror(errno));
+ }
+ else
+ {
+ int j;
+ for (j = i + 1; j < 3; j++)
+ if (aHandles[j].hHandle == aHandles[i].hHandle)
+ {
+ int fd = _dup(j);
+ if (fd >= 0)
+ {
+ if (_dup2(fd, j) >= 0)
+ {
+ aHandles[j].hHandle = (HANDLE)_get_osfhandle(j);
+ assert(aHandles[j].hHandle != aHandles[i].hHandle);
+ assert(aHandles[j].hHandle == GetStdHandle(aHandles[j].uStdHandle));
+ }
+ else
+ ONNNS(fatal, NILF, "_dup2(%d, %d) failed: %u (%s)", fd, j, errno, strerror(errno));
+ if (fd != j)
+ close(fd);
+ }
+ else
+ ONNS(fatal, NILF, "_dup(%d) failed: %u (%s)", j, errno, strerror(errno));
+ }
+ }
+ }
+#endif
+}
+
+/**
+ * Create or open a job object for this make instance and its children.
+ *
+ * Depending on the --job-object=mode value, we typically create/open a job
+ * object here if we're the root make instance. The job object is then
+ * typically configured to kill all remaining processes when the root make
+ * terminates, so that there aren't any stuck processes around messing up
+ * subsequent builds. This is very handy on build servers.
+ *
+ * If we're it no-kill mode, the job object is pretty pointless for manual
+ * cleanup as the job object becomes invisible (or something) when the last
+ * handle to it closes, i.e. g_hJob. On windows 8 and later it looks
+ * like any orphaned children are immediately assigned to the parent job
+ * object. Too bad for kmk_kill and such.
+ *
+ * win_job_object_mode values: login, root, each, none
+ */
+static void mkWinChildInitJobObjectAssociation(void)
+{
+ BOOL fCreate = TRUE;
+ char szJobName[128];
+ const char *pszJobName = win_job_object_name;
+
+ /* Skip if disabled. */
+ if (strcmp(win_job_object_mode, "none") == 0)
+ return;
+
+ /* Skip if not root make instance, unless we're having one job object
+ per make instance. */
+ if ( makelevel != 0
+ && strcmp(win_job_object_mode, "each") != 0)
+ return;
+
+ /* Format the the default job object name if --job-object-name
+ wasn't given. */
+ if (!pszJobName || *pszJobName == '\0')
+ {
+ pszJobName = szJobName;
+ if (strcmp(win_job_object_mode, "login") == 0)
+ {
+ /* Use the AuthenticationId like mspdbsrv.exe does. */
+ HANDLE hToken;
+ if (OpenProcessToken(GetCurrentProcess(), TOKEN_QUERY, &hToken))
+ {
+ TOKEN_STATISTICS TokenStats;
+ DWORD cbRet = 0;
+ memset(&TokenStats, 0, sizeof(TokenStats));
+ if (GetTokenInformation(hToken, TokenStatistics, &TokenStats, sizeof(TokenStats), &cbRet))
+ snprintf(szJobName, sizeof(szJobName), "kmk-job-obj-login-%08x.%08x",
+ (unsigned)TokenStats.AuthenticationId.HighPart, (unsigned)TokenStats.AuthenticationId.LowPart);
+ else
+ {
+ ONN(message, 0, _("GetTokenInformation failed: %u (cbRet=%u)"), GetLastError(), cbRet);
+ return;
+ }
+ CloseHandle(hToken);
+ }
+ else
+ {
+ ON(message, 0, _("OpenProcessToken failed: %u"), GetLastError());
+ return;
+ }
+ }
+ else
+ {
+ SYSTEMTIME Now = {0};
+ GetSystemTime(&Now);
+ snprintf(szJobName, sizeof(szJobName), "kmk-job-obj-%04u-%02u-%02uT%02u-%02u-%02uZ%u",
+ Now.wYear, Now.wMonth, Now.wDay, Now.wHour, Now.wMinute, Now.wSecond, getpid());
+ }
+ }
+
+ /* In login mode and when given a job object name, we try open it first. */
+ if ( win_job_object_name
+ || strcmp(win_job_object_mode, "login") == 0)
+ {
+ g_hJob = OpenJobObjectA(JOB_OBJECT_ASSIGN_PROCESS, win_job_object_no_kill /*bInheritHandle*/, pszJobName);
+ if (g_hJob)
+ fCreate = FALSE;
+ else
+ {
+ DWORD dwErr = GetLastError();
+ if (dwErr != ERROR_PATH_NOT_FOUND && dwErr != ERROR_FILE_NOT_FOUND)
+ {
+ OSN(message, 0, _("OpenJobObjectA(,,%s) failed: %u"), pszJobName, GetLastError());
+ return;
+ }
+ }
+ }
+
+ if (fCreate)
+ {
+ SECURITY_ATTRIBUTES SecAttr = { sizeof(SecAttr), NULL, TRUE /*bInheritHandle*/ };
+ g_hJob = CreateJobObjectA(win_job_object_no_kill ? &SecAttr : NULL, pszJobName);
+ if (g_hJob)
+ {
+ /* We need to set the BREAKAWAY_OK flag, as we don't want make CreateProcess
+ fail if someone tries to break way. Also set KILL_ON_JOB_CLOSE unless
+ --job-object-no-kill is given. */
+ JOBOBJECT_EXTENDED_LIMIT_INFORMATION Info;
+ DWORD cbActual = 0;
+ memset(&Info, 0, sizeof(Info));
+ if (QueryInformationJobObject(g_hJob, JobObjectExtendedLimitInformation, &Info, sizeof(Info), &cbActual))
+ {
+ Info.BasicLimitInformation.LimitFlags |= JOB_OBJECT_LIMIT_BREAKAWAY_OK;
+ if (!win_job_object_no_kill)
+ Info.BasicLimitInformation.LimitFlags |= JOB_OBJECT_LIMIT_KILL_ON_JOB_CLOSE;
+ else
+ Info.BasicLimitInformation.LimitFlags &= ~JOB_OBJECT_LIMIT_KILL_ON_JOB_CLOSE;
+ if (!SetInformationJobObject(g_hJob, JobObjectExtendedLimitInformation, &Info, sizeof(Info)))
+ OSSN(message, 0, _("SetInformationJobObject(%s,JobObjectExtendedLimitInformation,{%s},) failed: %u"),
+ pszJobName, win_job_object_mode, GetLastError());
+ }
+ else
+ OSN(message, 0, _("QueryInformationJobObject(%s,JobObjectExtendedLimitInformation,,,) failed: %u"),
+ pszJobName, GetLastError());
+ }
+ else
+ {
+ OSN(message, 0, _("CreateJobObjectA(NULL,%s) failed: %u"), pszJobName, GetLastError());
+ return;
+ }
+ }
+
+ /* Make it our job object. */
+ if (!(AssignProcessToJobObject(g_hJob, GetCurrentProcess())))
+ OSN(message, 0, _("AssignProcessToJobObject(%s, me) failed: %u"), pszJobName, GetLastError());
+}
+
+/**
+ * Used by mkWinChildcareWorkerThread() and MkWinChildWait() to get the head
+ * child from a lifo (g_pTailCompletedChildren, pTailTodoChildren).
+ *
+ * @returns Head child.
+ * @param ppTail Pointer to the child variable.
+ * @param pChild Tail child.
+ */
+static PWINCHILD mkWinChildDequeFromLifo(PWINCHILD volatile *ppTail, PWINCHILD pChild)
+{
+ if (pChild->pNext)
+ {
+ PWINCHILD pPrev;
+ do
+ {
+ pPrev = pChild;
+ pChild = pChild->pNext;
+ } while (pChild->pNext);
+ pPrev->pNext = NULL;
+ }
+ else
+ {
+ PWINCHILD const pWantedChild = pChild;
+ pChild = _InterlockedCompareExchangePointer(ppTail, NULL, pWantedChild);
+ if (pChild != pWantedChild)
+ {
+ PWINCHILD pPrev;
+ do
+ {
+ pPrev = pChild;
+ pChild = pChild->pNext;
+ } while (pChild->pNext);
+ pPrev->pNext = NULL;
+ assert(pChild == pWantedChild);
+ }
+ }
+ return pChild;
+}
+
+/**
+ * Output error message while running on a worker thread.
+ *
+ * @returns -1
+ * @param pWorker The calling worker. Mainly for getting the
+ * current child and its stderr output unit. Pass
+ * NULL if the output should go thru the child
+ * stderr buffering.
+ * @param iType The error type:
+ * - 0: more of a info directly to stdout,
+ * - 1: child related error,
+ * - 2: child related error for immedate release.
+ * @param pszFormat The message format string.
+ * @param ... Argument for the message.
+ */
+static int MkWinChildError(PWINCHILDCAREWORKER pWorker, int iType, const char *pszFormat, ...)
+{
+ /*
+ * Format the message into stack buffer.
+ */
+ char szMsg[4096];
+ int cchMsg;
+ int cchPrefix;
+ va_list va;
+
+ /* Compose the prefix, being paranoid about it not exceeding the buffer in any way. */
+ const char *pszInfix = iType == 0 ? "info: " : "error: ";
+ const char *pszProgram = program;
+ if (strlen(pszProgram) > 80)
+ {
+#ifdef KMK
+ pszProgram = "kmk";
+#else
+ pszProgram = "gnumake";
+#endif
+ }
+ if (makelevel == 0)
+ cchPrefix = snprintf(szMsg, sizeof(szMsg) / 2, "%s: %s", pszProgram, pszInfix);
+ else
+ cchPrefix = snprintf(szMsg, sizeof(szMsg) / 2, "%s[%u]: %s", pszProgram, makelevel, pszInfix);
+ assert(cchPrefix < sizeof(szMsg) / 2 && cchPrefix > 0);
+
+ /* Format the user specified message. */
+ va_start(va, pszFormat);
+ cchMsg = vsnprintf(&szMsg[cchPrefix], sizeof(szMsg) - 2 - cchPrefix, pszFormat, va);
+ va_end(va);
+ szMsg[sizeof(szMsg) - 2] = '\0';
+ cchMsg = strlen(szMsg);
+
+ /* Make sure there's a newline at the end of it (we reserved space for that). */
+ if (cchMsg <= 0 || szMsg[cchMsg - 1] != '\n')
+ {
+ szMsg[cchMsg++] = '\n';
+ szMsg[cchMsg] = '\0';
+ }
+
+#ifdef CONFIG_WITH_OUTPUT_IN_MEMORY
+ /*
+ * Try use the stderr of the current child of the worker.
+ */
+ if ( iType != 0
+ && iType != 3
+ && pWorker)
+ {
+ PWINCHILD pChild = pWorker->pCurChild;
+ if (pChild)
+ {
+ struct child *pMkChild = pChild->pMkChild;
+ if (pMkChild)
+ {
+ output_write_text(&pMkChild->output, 1 /*is_err*/, szMsg, cchMsg);
+ return -1;
+ }
+ }
+ }
+#endif
+
+ /*
+ * Fallback to writing directly to stderr.
+ */
+ maybe_con_fwrite(szMsg, cchMsg, 1, iType == 0 ? stdout : stderr);
+ return -1;
+}
+
+/**
+ * Duplicates the given UTF-16 string.
+ *
+ * @returns 0
+ * @param pwszSrc The UTF-16 string to duplicate.
+ * @param cwcSrc Length, may include the terminator.
+ * @param ppwszDst Where to return the duplicate.
+ */
+static int mkWinChildDuplicateUtf16String(const WCHAR *pwszSrc, size_t cwcSrc, WCHAR **ppwszDst)
+{
+ size_t cb = sizeof(WCHAR) * cwcSrc;
+ if (cwcSrc > 0 && pwszSrc[cwcSrc - 1] == L'\0')
+ *ppwszDst = (WCHAR *)memcpy(xmalloc(cb), pwszSrc, cb);
+ else
+ {
+ WCHAR *pwszDst = (WCHAR *)xmalloc(cb + sizeof(WCHAR));
+ memcpy(pwszDst, pwszSrc, cb);
+ pwszDst[cwcSrc] = L'\0';
+ *ppwszDst = pwszDst;
+ }
+ return 0;
+}
+
+
+/**
+ * Used to flush data we're read but not yet written at the termination of a
+ * process.
+ *
+ * @param pChild The child.
+ * @param pPipe The pipe.
+ */
+static void mkWinChildcareWorkerFlushUnwritten(PWINCHILD pChild, PWINCCWPIPE pPipe)
+{
+ DWORD cbUnwritten = pPipe->offPendingRead - pPipe->cbWritten;
+ assert(pPipe->cbWritten <= pPipe->cbBuffer - 16);
+ assert(pPipe->offPendingRead <= pPipe->cbBuffer - 16);
+ if (cbUnwritten)
+ {
+#ifdef CONFIG_WITH_OUTPUT_IN_MEMORY
+ if (pChild && pChild->pMkChild)
+ {
+ output_write_bin(&pChild->pMkChild->output, pPipe->iWhich == 2, &pPipe->pbBuffer[pPipe->cbWritten], cbUnwritten);
+ pPipe->cbWritten += cbUnwritten;
+ }
+ else
+#endif
+ {
+ DWORD cbWritten = 0;
+ if (WriteFile(GetStdHandle(pPipe->iWhich == 1 ? STD_OUTPUT_HANDLE : STD_ERROR_HANDLE),
+ &pPipe->pbBuffer[pPipe->cbWritten], cbUnwritten, &cbWritten, NULL))
+ pPipe->cbWritten += cbWritten <= cbUnwritten ? cbWritten : cbUnwritten; /* paranoia */
+ }
+ pPipe->fHaveWrittenOut = TRUE;
+ }
+}
+
+/**
+ * This logic mirrors kwSandboxConsoleFlushAll.
+ *
+ * @returns TRUE if it looks like a CL.EXE source line, otherwise FALSE.
+ * @param pPipe The pipe.
+ * @param offStart The start of the output in the pipe buffer.
+ * @param offEnd The end of the output in the pipe buffer.
+ */
+static BOOL mkWinChildcareWorkerIsClExeSourceLine(PWINCCWPIPE pPipe, DWORD offStart, DWORD offEnd)
+{
+ if (offEnd < offStart + 2)
+ return FALSE;
+ if (offEnd - offStart > 80)
+ return FALSE;
+
+ if ( pPipe->pbBuffer[offEnd - 2] != '\r'
+ || pPipe->pbBuffer[offEnd - 1] != '\n')
+ return FALSE;
+
+ offEnd -= 2;
+ while (offEnd-- > offStart)
+ {
+ char ch = pPipe->pbBuffer[offEnd];
+ if (isalnum(ch) || ch == '.' || ch == ' ' || ch == '_' || ch == '-')
+ { /* likely */ }
+ else
+ return FALSE;
+ }
+
+ return TRUE;
+}
+
+/**
+ * Adds output to the given standard output for the child.
+ *
+ * There is no pending read when this function is called, so we're free to
+ * reshuffle the buffer if desirable.
+ *
+ * @param pChild The child. Optional (kSubmit).
+ * @param iWhich Which standard descriptor number.
+ * @param cbNewData How much more output was caught.
+ */
+static void mkWinChildcareWorkerCaughtMoreOutput(PWINCHILD pChild, PWINCCWPIPE pPipe, DWORD cbNewData)
+{
+ DWORD offStart = pPipe->cbWritten;
+ assert(offStart <= pPipe->offPendingRead);
+ assert(offStart <= pPipe->cbBuffer - 16);
+ assert(pPipe->offPendingRead <= pPipe->cbBuffer - 16);
+ if (cbNewData > 0)
+ {
+ DWORD offRest;
+
+ /* Move offPendingRead ahead by cbRead. */
+ pPipe->offPendingRead += cbNewData;
+ assert(pPipe->offPendingRead <= pPipe->cbBuffer);
+ if (pPipe->offPendingRead > pPipe->cbBuffer)
+ pPipe->offPendingRead = pPipe->cbBuffer;
+
+ /* Locate the last newline in the buffer. */
+ offRest = pPipe->offPendingRead;
+ while (offRest > offStart && pPipe->pbBuffer[offRest - 1] != '\n')
+ offRest--;
+
+ /* If none were found and we've less than 16 bytes left in the buffer, try
+ find a word boundrary to flush on instead. */
+ if ( offRest <= offStart
+ && pPipe->cbBuffer - pPipe->offPendingRead + offStart < 16)
+ {
+ offRest = pPipe->offPendingRead;
+ while ( offRest > offStart
+ && isalnum(pPipe->pbBuffer[offRest - 1]))
+ offRest--;
+ if (offRest == offStart)
+ offRest = pPipe->offPendingRead;
+ }
+ /* If this is a potential CL.EXE process, we will keep the source
+ filename unflushed and maybe discard it at the end. */
+ else if ( pChild
+ && pChild->fProbableClExe
+ && pPipe->iWhich == 1
+ && offRest == pPipe->offPendingRead
+ && mkWinChildcareWorkerIsClExeSourceLine(pPipe, offStart, offRest))
+ offRest = offStart;
+
+ if (offRest > offStart)
+ {
+ /* Write out offStart..offRest. */
+ DWORD cbToWrite = offRest - offStart;
+#ifdef CONFIG_WITH_OUTPUT_IN_MEMORY
+ if (pChild && pChild->pMkChild)
+ {
+ output_write_bin(&pChild->pMkChild->output, pPipe->iWhich == 2, &pPipe->pbBuffer[offStart], cbToWrite);
+ offStart += cbToWrite;
+ pPipe->cbWritten = offStart;
+ }
+ else
+#endif
+ {
+ DWORD cbWritten = 0;
+ if (WriteFile(GetStdHandle(pPipe->iWhich == 1 ? STD_OUTPUT_HANDLE : STD_ERROR_HANDLE),
+ &pPipe->pbBuffer[offStart], cbToWrite, &cbWritten, NULL))
+ {
+ offStart += cbWritten <= cbToWrite ? cbWritten : cbToWrite; /* paranoia */
+ pPipe->cbWritten = offStart;
+ }
+ }
+ pPipe->fHaveWrittenOut = TRUE;
+ }
+ }
+
+ /* Shuffle the data to the front of the buffer. */
+ if (offStart > 0)
+ {
+ DWORD cbUnwritten = pPipe->offPendingRead - offStart;
+ if (cbUnwritten > 0)
+ memmove(pPipe->pbBuffer, &pPipe->pbBuffer[offStart], cbUnwritten);
+ pPipe->offPendingRead -= pPipe->cbWritten;
+ pPipe->cbWritten = 0;
+ }
+}
+
+/**
+ * Catches output from the given pipe.
+ *
+ * @param pChild The child. Optional (kSubmit).
+ * @param pPipe The pipe.
+ * @param fDraining Set if we're draining the pipe after the process
+ * terminated.
+ */
+static void mkWinChildcareWorkerCatchOutput(PWINCHILD pChild, PWINCCWPIPE pPipe, BOOL fDraining)
+{
+ /*
+ * Deal with already pending read.
+ */
+ if (pPipe->fReadPending)
+ {
+ DWORD cbRead = 0;
+ if (GetOverlappedResult(pPipe->hPipeMine, &pPipe->Overlapped, &cbRead, !fDraining))
+ {
+ mkWinChildcareWorkerCaughtMoreOutput(pChild, pPipe, cbRead);
+ pPipe->fReadPending = FALSE;
+ }
+ else if (fDraining && GetLastError() == ERROR_IO_INCOMPLETE)
+ return;
+ else
+ {
+ MkWinChildError(pChild ? pChild->pWorker : NULL, 2, "GetOverlappedResult failed: %u\n", GetLastError());
+ pPipe->fReadPending = FALSE;
+ if (fDraining)
+ return;
+ }
+ }
+
+ /*
+ * Read data till one becomes pending.
+ */
+ for (;;)
+ {
+ DWORD cbRead;
+
+ memset(&pPipe->Overlapped, 0, sizeof(pPipe->Overlapped));
+ pPipe->Overlapped.hEvent = pPipe->hEvent;
+ ResetEvent(pPipe->hEvent);
+
+ assert(pPipe->offPendingRead < pPipe->cbBuffer);
+ SetLastError(0);
+ cbRead = 0;
+ if (!ReadFile(pPipe->hPipeMine, &pPipe->pbBuffer[pPipe->offPendingRead],
+ pPipe->cbBuffer - pPipe->offPendingRead, &cbRead, &pPipe->Overlapped))
+ {
+ DWORD dwErr = GetLastError();
+ if (dwErr == ERROR_IO_PENDING)
+ pPipe->fReadPending = TRUE;
+ else
+ MkWinChildError(pChild ? pChild->pWorker : NULL, 2,
+ "ReadFile failed on standard %s: %u\n",
+ pPipe->iWhich == 1 ? "output" : "error", GetLastError());
+ return;
+ }
+
+ mkWinChildcareWorkerCaughtMoreOutput(pChild, pPipe, cbRead);
+ }
+}
+
+/**
+ * Makes sure the output pipes are drained and pushed to output.
+ *
+ * @param pChild The child. Optional (kSubmit).
+ * @param pStdOut The standard output pipe structure.
+ * @param pStdErr The standard error pipe structure.
+ */
+void MkWinChildcareWorkerDrainPipes(PWINCHILD pChild, PWINCCWPIPE pStdOut, PWINCCWPIPE pStdErr)
+{
+ mkWinChildcareWorkerCatchOutput(pChild, pStdOut, TRUE /*fDraining*/);
+ mkWinChildcareWorkerCatchOutput(pChild, pStdErr, TRUE /*fDraining*/);
+
+ /* Drop lone 'source.c' line from CL.exe, but only if no other output at all. */
+ if ( pChild
+ && pChild->fProbableClExe
+ && !pStdOut->fHaveWrittenOut
+ && !pStdErr->fHaveWrittenOut
+ && pStdErr->cbWritten == pStdErr->offPendingRead
+ && pStdOut->cbWritten < pStdOut->offPendingRead
+ && mkWinChildcareWorkerIsClExeSourceLine(pStdOut, pStdOut->cbWritten, pStdOut->offPendingRead))
+ {
+ if (!pStdOut->fReadPending)
+ pStdOut->cbWritten = pStdOut->offPendingRead = 0;
+ else
+ pStdOut->cbWritten = pStdOut->offPendingRead;
+ }
+ else
+ {
+ mkWinChildcareWorkerFlushUnwritten(pChild, pStdOut);
+ mkWinChildcareWorkerFlushUnwritten(pChild, pStdErr);
+ }
+}
+
+/**
+ * Commmon worker for waiting on a child process and retrieving the exit code.
+ *
+ * @returns Child exit code.
+ * @param pWorker The worker.
+ * @param pChild The child.
+ * @param hProcess The process handle.
+ * @param pwszJob The job name.
+ * @param fCatchOutput Set if we need to work the output pipes
+ * associated with the worker.
+ */
+static int mkWinChildcareWorkerWaitForProcess(PWINCHILDCAREWORKER pWorker, PWINCHILD pChild, HANDLE hProcess,
+ WCHAR const *pwszJob, BOOL fCatchOutput)
+{
+ DWORD const msStart = GetTickCount();
+ DWORD msNextMsg = msStart + 15000;
+
+ /* Reset the written indicators on the pipes before we start loop. */
+ pWorker->pStdOut->fHaveWrittenOut = FALSE;
+ pWorker->pStdErr->fHaveWrittenOut = FALSE;
+
+ for (;;)
+ {
+ /*
+ * Do the waiting and output catching.
+ */
+ DWORD dwStatus;
+ if (!fCatchOutput)
+ dwStatus = WaitForSingleObject(hProcess, 15001 /*ms*/);
+ else
+ {
+ HANDLE ahHandles[3] = { hProcess, pWorker->pStdOut->hEvent, pWorker->pStdErr->hEvent };
+ dwStatus = WaitForMultipleObjects(3, ahHandles, FALSE /*fWaitAll*/, 1000 /*ms*/);
+ if (dwStatus == WAIT_OBJECT_0 + 1)
+ mkWinChildcareWorkerCatchOutput(pChild, pWorker->pStdOut, FALSE /*fDraining*/);
+ else if (dwStatus == WAIT_OBJECT_0 + 2)
+ mkWinChildcareWorkerCatchOutput(pChild, pWorker->pStdErr, FALSE /*fDraining*/);
+ }
+ assert(dwStatus != WAIT_FAILED);
+
+ /*
+ * Get the exit code and return if the process was signalled as done.
+ */
+ if (dwStatus == WAIT_OBJECT_0)
+ {
+ DWORD dwExitCode = -42;
+ if (GetExitCodeProcess(hProcess, &dwExitCode))
+ {
+ pChild->iExitCode = (int)dwExitCode;
+ if (fCatchOutput)
+ MkWinChildcareWorkerDrainPipes(pChild, pWorker->pStdOut, pWorker->pStdErr);
+ return dwExitCode;
+ }
+ }
+ /*
+ * Loop again if just a timeout or pending output?
+ * Put out a message every 15 or 30 seconds if the job takes a while.
+ */
+ else if ( dwStatus == WAIT_TIMEOUT
+ || dwStatus == WAIT_OBJECT_0 + 1
+ || dwStatus == WAIT_OBJECT_0 + 2
+ || dwStatus == WAIT_IO_COMPLETION)
+ {
+ DWORD msNow = GetTickCount();
+ if (msNow >= msNextMsg)
+ {
+ if ( !pChild->pMkChild
+ || !pChild->pMkChild->recursive) /* ignore make recursions */
+ {
+ if ( !pChild->pMkChild
+ || !pChild->pMkChild->file
+ || !pChild->pMkChild->file->name)
+ MkWinChildError(NULL, 0, "Pid %u ('%ls') still running after %u seconds\n",
+ GetProcessId(hProcess), pwszJob, (msNow - msStart) / 1000);
+ else
+ MkWinChildError(NULL, 0, "Target '%s' (pid %u) still running after %u seconds\n",
+ pChild->pMkChild->file->name, GetProcessId(hProcess), (msNow - msStart) / 1000);
+ }
+
+ /* After 15s, 30s, 60s, 120s, 180s, ... */
+ if (msNextMsg == msStart + 15000)
+ msNextMsg += 15000;
+ else
+ msNextMsg += 30000;
+ }
+ continue;
+ }
+
+ /* Something failed. */
+ pChild->iExitCode = GetLastError();
+ if (pChild->iExitCode == 0)
+ pChild->iExitCode = -4242;
+ return pChild->iExitCode;
+ }
+}
+
+
+/**
+ * Closes standard handles that need closing before destruction.
+ *
+ * @param pChild The child (WINCHILDTYPE_PROCESS).
+ */
+static void mkWinChildcareWorkerCloseStandardHandles(PWINCHILD pChild)
+{
+ if ( pChild->u.Process.fCloseStdOut
+ && pChild->u.Process.hStdOut != INVALID_HANDLE_VALUE)
+ {
+ CloseHandle(pChild->u.Process.hStdOut);
+ pChild->u.Process.hStdOut = INVALID_HANDLE_VALUE;
+ pChild->u.Process.fCloseStdOut = FALSE;
+ }
+ if ( pChild->u.Process.fCloseStdErr
+ && pChild->u.Process.hStdErr != INVALID_HANDLE_VALUE)
+ {
+ CloseHandle(pChild->u.Process.hStdErr);
+ pChild->u.Process.hStdErr = INVALID_HANDLE_VALUE;
+ pChild->u.Process.fCloseStdErr = FALSE;
+ }
+}
+
+
+/**
+ * Does the actual process creation.
+ *
+ * @returns 0 if there is anything to wait on, otherwise non-zero windows error.
+ * @param pWorker The childcare worker.
+ * @param pChild The child.
+ * @param pwszImageName The image path.
+ * @param pwszCommandLine The command line.
+ * @param pwszzEnvironment The enviornment block.
+ */
+static int mkWinChildcareWorkerCreateProcess(PWINCHILDCAREWORKER pWorker, WCHAR const *pwszImageName,
+ WCHAR const *pwszCommandLine, WCHAR const *pwszzEnvironment, WCHAR const *pwszCwd,
+ BOOL pafReplace[3], HANDLE pahChild[3], BOOL fCatchOutput, HANDLE *phProcess)
+{
+ PROCESS_INFORMATION ProcInfo;
+ STARTUPINFOW StartupInfo;
+ DWORD fFlags = CREATE_UNICODE_ENVIRONMENT;
+ BOOL const fHaveHandles = pafReplace[0] | pafReplace[1] | pafReplace[2];
+ BOOL fRet;
+ DWORD dwErr;
+#ifdef KMK
+ extern int process_priority;
+#endif
+
+ /*
+ * Populate startup info.
+ *
+ * Turns out we can get away without passing TRUE for the inherit handles
+ * parameter to CreateProcess when we're not using STARTF_USESTDHANDLES.
+ * At least on NT, which is all worth caring about at this point + context IMO.
+ *
+ * Not inherting the handles is a good thing because it means we won't
+ * accidentally end up with a pipe handle or such intended for a different
+ * child process, potentially causing the EOF/HUP event to be delayed.
+ *
+ * Since the present handle inhertiance requirements only involves standard
+ * output and error, we'll never set the inherit handles flag and instead
+ * do manual handle duplication and planting.
+ */
+ memset(&StartupInfo, 0, sizeof(StartupInfo));
+ StartupInfo.cb = sizeof(StartupInfo);
+ GetStartupInfoW(&StartupInfo);
+ StartupInfo.lpReserved2 = 0; /* No CRT file handle + descriptor info possible, sorry. */
+ StartupInfo.cbReserved2 = 0;
+ if ( !fHaveHandles
+ && !fCatchOutput)
+ StartupInfo.dwFlags &= ~STARTF_USESTDHANDLES;
+ else
+ {
+ fFlags |= CREATE_SUSPENDED;
+ StartupInfo.dwFlags &= ~STARTF_USESTDHANDLES;
+ }
+
+ /*
+ * Flags.
+ */
+#ifdef KMK
+ switch (process_priority)
+ {
+ case 1: fFlags |= CREATE_SUSPENDED | IDLE_PRIORITY_CLASS; break;
+ case 2: fFlags |= CREATE_SUSPENDED | BELOW_NORMAL_PRIORITY_CLASS; break;
+ case 3: fFlags |= CREATE_SUSPENDED | NORMAL_PRIORITY_CLASS; break;
+ case 4: fFlags |= CREATE_SUSPENDED | HIGH_PRIORITY_CLASS; break;
+ case 5: fFlags |= CREATE_SUSPENDED | REALTIME_PRIORITY_CLASS; break;
+ }
+#endif
+ if (g_cProcessorGroups > 1)
+ fFlags |= CREATE_SUSPENDED;
+
+ /*
+ * Try create the process.
+ */
+ DB(DB_JOBS, ("CreateProcessW(%ls, %ls,,, TRUE, %#x...)\n", pwszImageName, pwszCommandLine, fFlags));
+ memset(&ProcInfo, 0, sizeof(ProcInfo));
+#ifdef WITH_RW_LOCK
+ AcquireSRWLockShared(&g_RWLock);
+#endif
+
+ fRet = CreateProcessW((WCHAR *)pwszImageName, (WCHAR *)pwszCommandLine, NULL /*pProcSecAttr*/, NULL /*pThreadSecAttr*/,
+ FALSE /*fInheritHandles*/, fFlags, (WCHAR *)pwszzEnvironment, pwszCwd, &StartupInfo, &ProcInfo);
+ dwErr = GetLastError();
+
+#ifdef WITH_RW_LOCK
+ ReleaseSRWLockShared(&g_RWLock);
+#endif
+ if (fRet)
+ *phProcess = ProcInfo.hProcess;
+ else
+ {
+ MkWinChildError(pWorker, 1, "CreateProcess(%ls) failed: %u\n", pwszImageName, dwErr);
+ return (int)dwErr;
+ }
+
+ /*
+ * If the child is suspended, we've got some adjustment work to be done.
+ */
+ dwErr = ERROR_SUCCESS;
+ if (fFlags & CREATE_SUSPENDED)
+ {
+ /*
+ * First do handle inhertiance as that's the most complicated.
+ */
+ if (fHaveHandles || fCatchOutput)
+ {
+ char szErrMsg[128];
+ if (fCatchOutput)
+ {
+ if (!pafReplace[1])
+ {
+ pafReplace[1] = TRUE;
+ pahChild[1] = pWorker->pStdOut->hPipeChild;
+ }
+ if (!pafReplace[2])
+ {
+ pafReplace[2] = TRUE;
+ pahChild[2] = pWorker->pStdErr->hPipeChild;
+ }
+ }
+ dwErr = nt_child_inject_standard_handles(ProcInfo.hProcess, pafReplace, pahChild, szErrMsg, sizeof(szErrMsg));
+ if (dwErr != 0)
+ MkWinChildError(pWorker, 1, "%s\n", szErrMsg);
+ }
+
+ /*
+ * Assign processor group (ignore failure).
+ */
+#ifdef MKWINCHILD_DO_SET_PROCESSOR_GROUP
+ if (g_cProcessorGroups > 1)
+ {
+ GROUP_AFFINITY Affinity = { 0 /* == all active apparently */, pWorker->iProcessorGroup, { 0, 0, 0 } };
+ fRet = g_pfnSetThreadGroupAffinity(ProcInfo.hThread, &Affinity, NULL);
+ assert(fRet);
+ }
+#endif
+
+#ifdef KMK
+ /*
+ * Set priority (ignore failure).
+ */
+ switch (process_priority)
+ {
+ case 1: fRet = SetThreadPriority(ProcInfo.hThread, THREAD_PRIORITY_IDLE); break;
+ case 2: fRet = SetThreadPriority(ProcInfo.hThread, THREAD_PRIORITY_BELOW_NORMAL); break;
+ case 3: fRet = SetThreadPriority(ProcInfo.hThread, THREAD_PRIORITY_NORMAL); break;
+ case 4: fRet = SetThreadPriority(ProcInfo.hThread, THREAD_PRIORITY_HIGHEST); break;
+ case 5: fRet = SetThreadPriority(ProcInfo.hThread, THREAD_PRIORITY_TIME_CRITICAL); break;
+ default: fRet = TRUE;
+ }
+ assert(fRet);
+#endif
+
+ /*
+ * Inject the job object if we're in a non-killing mode, to postpone
+ * the closing of the job object and maybe make it more useful.
+ */
+ if (win_job_object_no_kill && g_hJob)
+ {
+ HANDLE hWhatever = INVALID_HANDLE_VALUE;
+ DuplicateHandle(GetCurrentProcess(), g_hJob, ProcInfo.hProcess, &hWhatever, GENERIC_ALL,
+ TRUE /*bInheritHandle*/, DUPLICATE_SAME_ACCESS);
+ }
+
+ /*
+ * Resume the thread if the adjustments succeeded, otherwise kill it.
+ */
+ if (dwErr == ERROR_SUCCESS)
+ {
+ fRet = ResumeThread(ProcInfo.hThread);
+ assert(fRet);
+ if (!fRet)
+ {
+ dwErr = GetLastError();
+ MkWinChildError(pWorker, 1, "ResumeThread failed on child process: %u\n", dwErr);
+ }
+ }
+ if (dwErr != ERROR_SUCCESS)
+ TerminateProcess(ProcInfo.hProcess, dwErr);
+ }
+
+ /*
+ * Close unnecessary handles and cache the image.
+ */
+ CloseHandle(ProcInfo.hThread);
+ kmk_cache_exec_image_w(pwszImageName);
+ return 0;
+}
+
+/**
+ * Converts a argument vector that has already been quoted correctly.
+ *
+ * The argument vector is typically the result of quote_argv().
+ *
+ * @returns 0 on success, non-zero on failure.
+ * @param pWorker The childcare worker.
+ * @param papszArgs The argument vector to convert.
+ * @param ppwszCommandLine Where to return the command line.
+ */
+static int mkWinChildcareWorkerConvertQuotedArgvToCommandline(PWINCHILDCAREWORKER pWorker, char **papszArgs,
+ WCHAR **ppwszCommandLine)
+{
+ WCHAR *pwszCmdLine;
+ WCHAR *pwszDst;
+
+ /*
+ * Calc length the converted length.
+ */
+ unsigned cwcNeeded = 1;
+ unsigned i = 0;
+ const char *pszSrc;
+ while ((pszSrc = papszArgs[i]) != NULL)
+ {
+ int cwcThis = MultiByteToWideChar(CP_ACP, 0 /*fFlags*/, pszSrc, -1, NULL, 0);
+ if (cwcThis > 0 || *pszSrc == '\0')
+ cwcNeeded += cwcThis + 1;
+ else
+ {
+ DWORD dwErr = GetLastError();
+ MkWinChildError(pWorker, 1, _("MultiByteToWideChar failed to convert argv[%u] (%s): %u\n"), i, pszSrc, dwErr);
+ return dwErr;
+ }
+ i++;
+ }
+
+ /*
+ * Allocate and do the conversion.
+ */
+ pwszCmdLine = pwszDst = (WCHAR *)xmalloc(cwcNeeded * sizeof(WCHAR));
+ i = 0;
+ while ((pszSrc = papszArgs[i]) != NULL)
+ {
+ int cwcThis;
+ if (i > 0)
+ {
+ *pwszDst++ = ' ';
+ cwcNeeded--;
+ }
+
+ cwcThis = MultiByteToWideChar(CP_ACP, 0 /*fFlags*/, pszSrc, -1, pwszDst, cwcNeeded);
+ if (!cwcThis && *pszSrc != '\0')
+ {
+ DWORD dwErr = GetLastError();
+ MkWinChildError(pWorker, 1, _("MultiByteToWideChar failed to convert argv[%u] (%s): %u\n"), i, pszSrc, dwErr);
+ free(pwszCmdLine);
+ return dwErr;
+ }
+ if (cwcThis > 0 && pwszDst[cwcThis - 1] == '\0')
+ cwcThis--;
+ pwszDst += cwcThis;
+ cwcNeeded -= cwcThis;
+ i++;
+ }
+ *pwszDst++ = '\0';
+
+ *ppwszCommandLine = pwszCmdLine;
+ return 0;
+}
+
+
+#define MKWCCWCMD_F_CYGWIN_SHELL 1
+#define MKWCCWCMD_F_MKS_SHELL 2
+#define MKWCCWCMD_F_HAVE_SH 4
+#define MKWCCWCMD_F_HAVE_KASH_C 8 /**< kmk_ash -c "..." */
+
+/*
+ * @param pWorker The childcare worker if on one, otherwise NULL.
+ */
+static int mkWinChildcareWorkerConvertCommandline(PWINCHILDCAREWORKER pWorker, char **papszArgs, unsigned fFlags,
+ WCHAR **ppwszCommandLine)
+{
+ struct ARGINFO
+ {
+ size_t cchSrc;
+ size_t cwcDst; /**< converted size w/o terminator. */
+ size_t cwcDstExtra : 24; /**< Only set with fSlowly. */
+ size_t fSlowly : 1;
+ size_t fQuoteIt : 1;
+ size_t fEndSlashes : 1; /**< if escapes needed for trailing backslashes. */
+ size_t fExtraSpace : 1; /**< if kash -c "" needs an extra space before the quote. */
+ } *paArgInfo;
+ size_t cArgs;
+ size_t i;
+ size_t cwcNeeded;
+ WCHAR *pwszDst;
+ WCHAR *pwszCmdLine;
+
+ /*
+ * Count them first so we can allocate an info array of the stack.
+ */
+ cArgs = 0;
+ while (papszArgs[cArgs] != NULL)
+ cArgs++;
+ paArgInfo = (struct ARGINFO *)alloca(sizeof(paArgInfo[0]) * cArgs);
+
+ /*
+ * Preprocess them and calculate the exact command line length.
+ */
+ cwcNeeded = 1;
+ for (i = 0; i < cArgs; i++)
+ {
+ char *pszSrc = papszArgs[i];
+ size_t cchSrc = strlen(pszSrc);
+ paArgInfo[i].cchSrc = cchSrc;
+ if (cchSrc == 0)
+ {
+ /* empty needs quoting. */
+ paArgInfo[i].cwcDst = 2;
+ paArgInfo[i].cwcDstExtra = 0;
+ paArgInfo[i].fSlowly = 0;
+ paArgInfo[i].fQuoteIt = 1;
+ paArgInfo[i].fExtraSpace = 0;
+ paArgInfo[i].fEndSlashes = 0;
+ }
+ else
+ {
+ const char *pszSpace = memchr(pszSrc, ' ', cchSrc);
+ const char *pszTab = memchr(pszSrc, '\t', cchSrc);
+ const char *pszDQuote = memchr(pszSrc, '"', cchSrc);
+ const char *pszEscape = memchr(pszSrc, '\\', cchSrc);
+ int cwcDst = MultiByteToWideChar(CP_ACP, 0 /*fFlags*/, pszSrc, cchSrc + 1, NULL, 0);
+ if (cwcDst >= 0)
+ --cwcDst;
+ else
+ {
+ DWORD dwErr = GetLastError();
+ MkWinChildError(pWorker, 1, _("MultiByteToWideChar failed to convert argv[%u] (%s): %u\n"), i, pszSrc, dwErr);
+ return dwErr;
+ }
+#if 0
+ if (!pszSpace && !pszTab && !pszDQuote && !pszEscape)
+ {
+ /* no special handling needed. */
+ paArgInfo[i].cwcDst = cwcDst;
+ paArgInfo[i].cwcDstExtra = 0;
+ paArgInfo[i].fSlowly = 0;
+ paArgInfo[i].fQuoteIt = 0;
+ paArgInfo[i].fExtraSpace = 0;
+ paArgInfo[i].fEndSlashes = 0;
+ }
+ else if (!pszDQuote && !pszEscape)
+ {
+ /* Just double quote it. */
+ paArgInfo[i].cwcDst = cwcDst + 2;
+ paArgInfo[i].cwcDstExtra = 0;
+ paArgInfo[i].fSlowly = 0;
+ paArgInfo[i].fQuoteIt = 1;
+ paArgInfo[i].fExtraSpace = 0;
+ paArgInfo[i].fEndSlashes = 0;
+ }
+ else
+#endif
+ {
+ /* Complicated, need to scan the string to figure out what to do. */
+ size_t cwcDstExtra;
+ int cBackslashes;
+ char ch;
+
+ paArgInfo[i].fQuoteIt = 0;
+ paArgInfo[i].fSlowly = 1;
+ paArgInfo[i].fExtraSpace = 0;
+ paArgInfo[i].fEndSlashes = 0;
+
+ cwcDstExtra = 0;
+ cBackslashes = 0;
+ while ((ch = *pszSrc++) != '\0')
+ {
+ switch (ch)
+ {
+ default:
+ cBackslashes = 0;
+ break;
+
+ case '\\':
+ cBackslashes++;
+ break;
+
+ case '"':
+ if (fFlags & (MKWCCWCMD_F_CYGWIN_SHELL | MKWCCWCMD_F_MKS_SHELL))
+ cwcDstExtra += 1; /* just an extra '"' */
+ else
+ cwcDstExtra += 1 + cBackslashes; /* extra '\\' for the '"' and for each preceeding slash. */
+ cBackslashes = 0;
+ break;
+
+ case ' ':
+ case '\t':
+ if (!paArgInfo[i].fQuoteIt)
+ {
+ paArgInfo[i].fQuoteIt = 1;
+ cwcDstExtra += 2;
+ }
+ cBackslashes = 0;
+ break;
+ }
+ }
+
+ /* If we're quoting the argument and it ends with trailing '\\', it/they must be escaped. */
+ if ( cBackslashes > 0
+ && paArgInfo[i].fQuoteIt
+ && !(fFlags & (MKWCCWCMD_F_CYGWIN_SHELL | MKWCCWCMD_F_MKS_SHELL)))
+ {
+ cwcDstExtra += cBackslashes;
+ paArgInfo[i].fEndSlashes = 1;
+ }
+
+ paArgInfo[i].cwcDst = cwcDst + cwcDstExtra;
+ paArgInfo[i].cwcDstExtra = cwcDstExtra;
+ }
+ }
+
+ if ( (fFlags & MKWCCWCMD_F_HAVE_KASH_C)
+ && paArgInfo[i].fQuoteIt)
+ {
+ paArgInfo[i].fExtraSpace = 1;
+ paArgInfo[i].cwcDst++;
+ paArgInfo[i].cwcDstExtra++;
+ }
+
+ cwcNeeded += (i != 0) + paArgInfo[i].cwcDst;
+ }
+
+ /*
+ * Allocate the result buffer and do the actual conversion.
+ */
+ pwszDst = pwszCmdLine = (WCHAR *)xmalloc(sizeof(WCHAR) * cwcNeeded);
+ for (i = 0; i < cArgs; i++)
+ {
+ char *pszSrc = papszArgs[i];
+ size_t cwcDst = paArgInfo[i].cwcDst;
+
+ if (i != 0)
+ *pwszDst++ = L' ';
+
+ if (paArgInfo[i].fQuoteIt)
+ {
+ *pwszDst++ = L'"';
+ cwcDst -= 2;
+ }
+
+ if (!paArgInfo[i].fSlowly)
+ {
+ int cwcDst2 = MultiByteToWideChar(CP_ACP, 0 /*fFlags*/, pszSrc, paArgInfo[i].cchSrc, pwszDst, cwcDst + 1);
+ assert(cwcDst2 >= 0);
+ pwszDst += cwcDst;
+ }
+ else
+ {
+ /* Do the conversion into the end of the output buffer, then move
+ it up to where it should be char by char. */
+ int cBackslashes;
+ size_t cwcLeft = paArgInfo[i].cwcDst - paArgInfo[i].cwcDstExtra;
+ WCHAR volatile *pwchSlowSrc = pwszDst + paArgInfo[i].cwcDstExtra;
+ WCHAR volatile *pwchSlowDst = pwszDst;
+ int cwcDst2 = MultiByteToWideChar(CP_ACP, 0 /*fFlags*/, pszSrc, paArgInfo[i].cchSrc,
+ (WCHAR *)pwchSlowSrc, cwcLeft + 1);
+ assert(cwcDst2 >= 0);
+
+ cBackslashes = 0;
+ while (cwcLeft-- > 0)
+ {
+ WCHAR wcSrc = *pwchSlowSrc++;
+ if (wcSrc != L'\\' && wcSrc != L'"')
+ cBackslashes = 0;
+ else if (wcSrc == L'\\')
+ cBackslashes++;
+ else if ( (fFlags & (MKWCCWCMD_F_CYGWIN_SHELL | MKWCCWCMD_F_HAVE_SH))
+ == (MKWCCWCMD_F_CYGWIN_SHELL | MKWCCWCMD_F_HAVE_SH))
+ {
+ *pwchSlowDst++ = L'"'; /* cygwin: '"' instead of '\\', no escaped slashes. */
+ cBackslashes = 0;
+ }
+ else
+ {
+ if (!(fFlags & (MKWCCWCMD_F_CYGWIN_SHELL | MKWCCWCMD_F_MKS_SHELL)))
+ cBackslashes += 1; /* one extra escape the '"' and one for each preceeding slash. */
+ while (cBackslashes > 0)
+ {
+ *pwchSlowDst++ = L'\\';
+ cBackslashes--;
+ }
+ }
+ *pwchSlowDst++ = wcSrc;
+ assert((uintptr_t)pwchSlowDst <= (uintptr_t)pwchSlowSrc);
+ }
+
+ if (paArgInfo[i].fEndSlashes)
+ while (cBackslashes-- > 0)
+ *pwchSlowDst++ = L'\\';
+
+ pwszDst += cwcDst;
+ assert(pwszDst == (WCHAR *)pwchSlowDst);
+ }
+
+ if (paArgInfo[i].fExtraSpace)
+ *pwszDst++ = L' ';
+ if (paArgInfo[i].fQuoteIt)
+ *pwszDst++ = L'"';
+ }
+ *pwszDst = L'\0';
+ *ppwszCommandLine = pwszCmdLine;
+ return 0;
+}
+
+static int mkWinChildcareWorkerConvertCommandlineWithShell(PWINCHILDCAREWORKER pWorker, const WCHAR *pwszShell, char **papszArgs,
+ WCHAR **ppwszCommandLine)
+{
+ MkWinChildError(pWorker, 1, "%s: not found!\n", papszArgs[0]);
+//__debugbreak();
+ return ERROR_FILE_NOT_FOUND;
+}
+
+/**
+ * Searches the environment block for the PATH variable.
+ *
+ * @returns Pointer to the path in the block or "." in pwszPathFallback.
+ * @param pwszzEnv The UTF-16 environment block to search.
+ * @param pwszPathFallback Fallback.
+ */
+static const WCHAR *mkWinChildcareWorkerFindPathValue(const WCHAR *pwszzEnv, WCHAR pwszPathFallback[4])
+{
+ while (*pwszzEnv)
+ {
+ size_t cwcVar = wcslen(pwszzEnv);
+ if (!IS_PATH_ENV_VAR(cwcVar, pwszzEnv))
+ pwszzEnv += cwcVar + 1;
+ else if (cwcVar > 5)
+ return &pwszzEnv[5];
+ else
+ break;
+ }
+ pwszPathFallback[0] = L'.';
+ pwszPathFallback[1] = L'\0';
+ return pwszPathFallback;
+}
+
+/**
+ * Checks if we need to had this executable file to the shell.
+ *
+ * @returns TRUE if it's shell fooder, FALSE if we think windows can handle it.
+ * @param hFile Handle to the file in question
+ */
+static BOOL mkWinChildcareWorkerCheckIfNeedShell(HANDLE hFile)
+{
+ /*
+ * Read the first 512 bytes and check for an executable image header.
+ */
+ union
+ {
+ DWORD dwSignature;
+ WORD wSignature;
+ BYTE ab[128];
+ } uBuf;
+ DWORD cbRead;
+ uBuf.dwSignature = 0;
+ if ( ReadFile(hFile, &uBuf, sizeof(uBuf), &cbRead, NULL /*pOverlapped*/)
+ && cbRead == sizeof(uBuf))
+ {
+ if (uBuf.wSignature == IMAGE_DOS_SIGNATURE)
+ return FALSE;
+ if (uBuf.dwSignature == IMAGE_NT_SIGNATURE)
+ return FALSE;
+ if ( uBuf.wSignature == IMAGE_OS2_SIGNATURE /* NE */
+ || uBuf.wSignature == 0x5d4c /* LX */
+ || uBuf.wSignature == IMAGE_OS2_SIGNATURE_LE /* LE */)
+ return FALSE;
+ }
+ return TRUE;
+}
+
+/**
+ * Checks if the image path looks like microsoft CL.exe.
+ *
+ * @returns TRUE / FALSE.
+ * @param pwszImagePath The executable image path to evalutate.
+ * @param cwcImagePath The length of the image path.
+ */
+static BOOL mkWinChildIsProbableClExe(WCHAR const *pwszImagePath, size_t cwcImagePath)
+{
+ assert(pwszImagePath[cwcImagePath] == '\0');
+ return cwcImagePath > 7
+ && (pwszImagePath[cwcImagePath - 7] == L'/' || pwszImagePath[cwcImagePath - 7] == L'\\')
+ && (pwszImagePath[cwcImagePath - 6] == L'c' || pwszImagePath[cwcImagePath - 6] == L'C')
+ && (pwszImagePath[cwcImagePath - 5] == L'l' || pwszImagePath[cwcImagePath - 5] == L'L')
+ && pwszImagePath[cwcImagePath - 4] == L'.'
+ && (pwszImagePath[cwcImagePath - 3] == L'e' || pwszImagePath[cwcImagePath - 3] == L'E')
+ && (pwszImagePath[cwcImagePath - 2] == L'x' || pwszImagePath[cwcImagePath - 2] == L'X')
+ && (pwszImagePath[cwcImagePath - 1] == L'e' || pwszImagePath[cwcImagePath - 1] == L'E');
+}
+
+/**
+ * Temporary workaround for seemingly buggy kFsCache.c / dir-nt-bird.c.
+ *
+ * Something is not invalidated / updated correctly!
+ */
+static BOOL mkWinChildcareWorkerIsRegularFileW(PWINCHILDCAREWORKER pWorker, wchar_t const *pwszPath)
+{
+ BOOL fRet = FALSE;
+#ifdef KMK
+ if (utf16_regular_file_p(pwszPath))
+ fRet = TRUE;
+ else
+#endif
+ {
+ /* Don't believe the cache. */
+ DWORD dwAttr = GetFileAttributesW(pwszPath);
+ if (dwAttr != INVALID_FILE_ATTRIBUTES)
+ {
+ if (!(dwAttr & FILE_ATTRIBUTE_DIRECTORY))
+ {
+#ifdef KMK
+ extern void dir_cache_invalid_volatile(void);
+ dir_cache_invalid_volatile();
+ if (utf16_regular_file_p(pwszPath))
+ MkWinChildError(pWorker, 1, "kFsCache was out of sync! pwszPath=%S\n", pwszPath);
+ else
+ {
+ dir_cache_invalid_all();
+ if (utf16_regular_file_p(pwszPath))
+ MkWinChildError(pWorker, 1, "kFsCache was really out of sync! pwszPath=%S\n", pwszPath);
+ else
+ MkWinChildError(pWorker, 1, "kFsCache is really out of sync!! pwszPath=%S\n", pwszPath);
+ }
+#endif
+ fRet = TRUE;
+ }
+ }
+ }
+ return fRet;
+}
+
+
+/**
+ * Tries to locate the image file, searching the path and maybe falling back on
+ * the shell in case it knows more (think cygwin with its own view of the file
+ * system).
+ *
+ * This will also check for shell script, falling back on the shell too to
+ * handle those.
+ *
+ * @returns 0 on success, windows error code on failure.
+ * @param pWorker The childcare worker.
+ * @param pszArg0 The first argument.
+ * @param pwszSearchPath In case mkWinChildcareWorkerConvertEnvironment
+ * had a chance of locating the search path already.
+ * @param pwszzEnv The environment block, in case we need to look for
+ * the path.
+ * @param pszShell The shell.
+ * @param ppwszImagePath Where to return the pointer to the image path. This
+ * could be the shell.
+ * @param pfNeedShell Where to return shell vs direct execution indicator.
+ * @param pfProbableClExe Where to return an indicator of probably CL.EXE.
+ */
+static int mkWinChildcareWorkerFindImage(PWINCHILDCAREWORKER pWorker, char const *pszArg0, WCHAR *pwszSearchPath,
+ WCHAR const *pwszzEnv, const char *pszShell,
+ WCHAR **ppwszImagePath, BOOL *pfNeedShell, BOOL *pfProbableClExe)
+{
+ /** @todo Slap a cache on this code. We usually end up executing the same
+ * stuff over and over again (e.g. compilers, linkers, etc).
+ * Hitting the file system is slow on windows. */
+
+ /*
+ * Convert pszArg0 to unicode so we can work directly on that.
+ */
+ WCHAR wszArg0[MKWINCHILD_MAX_PATH + 4]; /* +4 for painless '.exe' appending */
+ DWORD dwErr;
+ size_t cbArg0 = strlen(pszArg0) + 1;
+ int const cwcArg0 = MultiByteToWideChar(CP_ACP, 0 /*fFlags*/, pszArg0, cbArg0, wszArg0, MKWINCHILD_MAX_PATH);
+ if (cwcArg0 > 0)
+ {
+ HANDLE hFile = INVALID_HANDLE_VALUE;
+ WCHAR wszPathBuf[MKWINCHILD_MAX_PATH + 4]; /* +4 for painless '.exe' appending */
+ int cwc;
+
+ /*
+ * If there isn't an .exe suffix, we may have to add one.
+ * Also we ASSUME that .exe suffixes means no hash bang detection needed.
+ */
+ int const fHasExeSuffix = cwcArg0 > CSTRLEN(".exe")
+ && wszArg0[cwcArg0 - 4] == '.'
+ && (wszArg0[cwcArg0 - 3] == L'e' || wszArg0[cwcArg0 - 3] == L'E')
+ && (wszArg0[cwcArg0 - 2] == L'x' || wszArg0[cwcArg0 - 2] == L'X')
+ && (wszArg0[cwcArg0 - 1] == L'e' || wszArg0[cwcArg0 - 1] == L'E');
+
+ /*
+ * If there isn't any path specified, we need to search the PATH env.var.
+ */
+ int const fHasPath = wszArg0[1] == L':'
+ || wszArg0[0] == L'\\'
+ || wszArg0[0] == L'/'
+ || wmemchr(wszArg0, L'/', cwcArg0)
+ || wmemchr(wszArg0, L'\\', cwcArg0);
+
+ /* Before we do anything, flip UNIX slashes to DOS ones. */
+ WCHAR *pwc = wszArg0;
+ while ((pwc = wcschr(pwc, L'/')) != NULL)
+ *pwc++ = L'\\';
+
+ /* Don't need to set these all the time... */
+ *pfNeedShell = FALSE;
+ *pfProbableClExe = FALSE;
+
+ /*
+ * If any kind of path is specified in arg0, we will not search the
+ * PATH env.var and can limit ourselves to maybe slapping a .exe on to it.
+ */
+ if (fHasPath)
+ {
+ /*
+ * If relative to a CWD, turn it into an absolute one.
+ */
+ unsigned cwcPath = cwcArg0;
+ WCHAR *pwszPath = wszArg0;
+ if ( *pwszPath != L'\\'
+ && (pwszPath[1] != ':' || pwszPath[2] != L'\\') )
+ {
+ DWORD cwcAbsPath = GetFullPathNameW(wszArg0, MKWINCHILD_MAX_PATH, wszPathBuf, NULL);
+ if (cwcAbsPath > 0)
+ {
+ cwcPath = cwcAbsPath + 1; /* include terminator, like MultiByteToWideChar does. */
+ pwszPath = wszPathBuf;
+ }
+ }
+
+ /*
+ * Check with .exe suffix first.
+ * We don't open .exe files and look for hash bang stuff, we just
+ * assume they are executable images that CreateProcess can deal with.
+ */
+ if (!fHasExeSuffix)
+ {
+ pwszPath[cwcPath - 1] = L'.';
+ pwszPath[cwcPath ] = L'e';
+ pwszPath[cwcPath + 1] = L'x';
+ pwszPath[cwcPath + 2] = L'e';
+ pwszPath[cwcPath + 3] = L'\0';
+ }
+
+ if (mkWinChildcareWorkerIsRegularFileW(pWorker, pwszPath))
+ {
+ *pfProbableClExe = mkWinChildIsProbableClExe(pwszPath, cwcPath + 4 - 1);
+ return mkWinChildDuplicateUtf16String(pwszPath, cwcPath + 4, ppwszImagePath);
+ }
+
+ /*
+ * If no suffix was specified, try without .exe too, but now we need
+ * to see if it's for the shell or CreateProcess.
+ */
+ if (!fHasExeSuffix)
+ {
+ pwszPath[cwcPath - 1] = L'\0';
+#ifdef KMK
+ if (mkWinChildcareWorkerIsRegularFileW(pWorker, pwszPath))
+#endif
+ {
+ hFile = CreateFileW(pwszPath, GENERIC_READ, FILE_SHARE_READ | FILE_SHARE_DELETE | FILE_SHARE_WRITE,
+ NULL /*pSecAttr*/, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
+ if (hFile != INVALID_HANDLE_VALUE)
+ {
+ *pfNeedShell = mkWinChildcareWorkerCheckIfNeedShell(hFile);
+ CloseHandle(hFile);
+ if (!*pfNeedShell)
+ {
+ *pfProbableClExe = mkWinChildIsProbableClExe(pwszPath, cwcPath - 1);
+ return mkWinChildDuplicateUtf16String(pwszPath, cwcPath, ppwszImagePath);
+ }
+ }
+ }
+ }
+ }
+ /*
+ * No path, need to search the PATH env.var. for the executable, maybe
+ * adding an .exe suffix while do so if that is missing.
+ */
+ else
+ {
+ BOOL fSearchedCwd = FALSE;
+ WCHAR wszPathFallback[4];
+ if (!pwszSearchPath)
+ pwszSearchPath = (WCHAR *)mkWinChildcareWorkerFindPathValue(pwszzEnv, wszPathFallback);
+
+ for (;;)
+ {
+ size_t cwcCombined;
+
+ /*
+ * Find the end of the current PATH component.
+ */
+ size_t cwcSkip;
+ WCHAR wcEnd;
+ size_t cwcComponent = 0;
+ WCHAR wc;
+ while ((wc = pwszSearchPath[cwcComponent]) != L'\0')
+ {
+ if (wc != ';' && wc != ':')
+ { /* likely */ }
+ else if (wc == ';')
+ break;
+ else if (cwcComponent != (pwszSearchPath[cwcComponent] != L'"' ? 1 : 2))
+ break;
+ cwcComponent++;
+ }
+ wcEnd = wc;
+
+ /* Trim leading spaces and double quotes. */
+ while ( cwcComponent > 0
+ && ((wc = *pwszSearchPath) == L'"' || wc == L' ' || wc == L'\t'))
+ {
+ pwszSearchPath++;
+ cwcComponent--;
+ }
+ cwcSkip = cwcComponent;
+
+ /* Trim trailing spaces & double quotes. */
+ while ( cwcComponent > 0
+ && ((wc = pwszSearchPath[cwcComponent - 1]) == L'"' || wc == L' ' || wc == L'\t'))
+ cwcComponent--;
+
+ /*
+ * Skip empty components. Join the component and the filename, making sure to
+ * resolve any CWD relative stuff first.
+ */
+ cwcCombined = cwcComponent + 1 + cwcArg0;
+ if (cwcComponent > 0 && cwcCombined <= MKWINCHILD_MAX_PATH)
+ {
+ /* Copy the component into wszPathBuf, maybe abspath'ing it. */
+ DWORD cwcAbsPath = 0;
+ if ( *pwszSearchPath != L'\\'
+ && (pwszSearchPath[1] != ':' || pwszSearchPath[2] != L'\\') )
+ {
+ /* To save an extra buffer + copying, we'll temporarily modify the PATH
+ value in our converted UTF-16 environment block. */
+ WCHAR const wcSaved = pwszSearchPath[cwcComponent];
+ pwszSearchPath[cwcComponent] = L'\0';
+ cwcAbsPath = GetFullPathNameW(pwszSearchPath, MKWINCHILD_MAX_PATH, wszPathBuf, NULL);
+ pwszSearchPath[cwcComponent] = wcSaved;
+ if (cwcAbsPath > 0 && cwcAbsPath + 1 + cwcArg0 <= MKWINCHILD_MAX_PATH)
+ cwcCombined = cwcAbsPath + 1 + cwcArg0;
+ else
+ cwcAbsPath = 0;
+ }
+ if (cwcAbsPath == 0)
+ {
+ memcpy(wszPathBuf, pwszSearchPath, cwcComponent * sizeof(WCHAR));
+ cwcAbsPath = cwcComponent;
+ }
+
+ /* Append the filename. */
+ if ((wc = wszPathBuf[cwcAbsPath - 1]) == L'\\' || wc == L'/' || wc == L':')
+ {
+ memcpy(&wszPathBuf[cwcAbsPath], wszArg0, cwcArg0 * sizeof(WCHAR));
+ cwcCombined--;
+ }
+ else
+ {
+ wszPathBuf[cwcAbsPath] = L'\\';
+ memcpy(&wszPathBuf[cwcAbsPath + 1], wszArg0, cwcArg0 * sizeof(WCHAR));
+ }
+ assert(wszPathBuf[cwcCombined - 1] == L'\0');
+
+ /* DOS slash conversion */
+ pwc = wszPathBuf;
+ while ((pwc = wcschr(pwc, L'/')) != NULL)
+ *pwc++ = L'\\';
+
+ /*
+ * Search with exe suffix first.
+ */
+ if (!fHasExeSuffix)
+ {
+ wszPathBuf[cwcCombined - 1] = L'.';
+ wszPathBuf[cwcCombined ] = L'e';
+ wszPathBuf[cwcCombined + 1] = L'x';
+ wszPathBuf[cwcCombined + 2] = L'e';
+ wszPathBuf[cwcCombined + 3] = L'\0';
+ }
+ if (mkWinChildcareWorkerIsRegularFileW(pWorker, wszPathBuf))
+ {
+ *pfProbableClExe = mkWinChildIsProbableClExe(wszPathBuf, cwcCombined + (fHasExeSuffix ? 0 : 4) - 1);
+ return mkWinChildDuplicateUtf16String(wszPathBuf, cwcCombined + (fHasExeSuffix ? 0 : 4), ppwszImagePath);
+ }
+ if (!fHasExeSuffix)
+ {
+ wszPathBuf[cwcCombined - 1] = L'\0';
+#ifdef KMK
+ if (mkWinChildcareWorkerIsRegularFileW(pWorker, wszPathBuf))
+#endif
+ {
+ /*
+ * Check if the file exists w/o the added '.exe' suffix. If it does,
+ * we need to check if we can pass it to CreateProcess or need the shell.
+ */
+ hFile = CreateFileW(wszPathBuf, GENERIC_READ, FILE_SHARE_READ | FILE_SHARE_DELETE | FILE_SHARE_WRITE,
+ NULL /*pSecAttr*/, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
+ if (hFile != INVALID_HANDLE_VALUE)
+ {
+ *pfNeedShell = mkWinChildcareWorkerCheckIfNeedShell(hFile);
+ CloseHandle(hFile);
+ if (!*pfNeedShell)
+ {
+ *pfProbableClExe = mkWinChildIsProbableClExe(wszPathBuf, cwcCombined - 1);
+ return mkWinChildDuplicateUtf16String(wszPathBuf, cwcCombined, ppwszImagePath);
+ }
+ break;
+ }
+ }
+ }
+ }
+
+ /*
+ * Advance to the next component.
+ */
+ if (wcEnd != '\0')
+ pwszSearchPath += cwcSkip + 1;
+ else if (fSearchedCwd)
+ break;
+ else
+ {
+ fSearchedCwd = TRUE;
+ wszPathFallback[0] = L'.';
+ wszPathFallback[1] = L'\0';
+ pwszSearchPath = wszPathFallback;
+ }
+ }
+ }
+
+ /*
+ * We need the shell. It will take care of finding/reporting missing
+ * image files and such.
+ */
+ *pfNeedShell = TRUE;
+ if (pszShell)
+ {
+ cwc = MultiByteToWideChar(CP_ACP, 0 /*fFlags*/, pszShell, strlen(pszShell) + 1, wszPathBuf, MKWINCHILD_MAX_PATH);
+ if (cwc > 0)
+ return mkWinChildDuplicateUtf16String(wszPathBuf, cwc, ppwszImagePath);
+ dwErr = GetLastError();
+ MkWinChildError(pWorker, 1, _("MultiByteToWideChar failed to convert shell (%s): %u\n"), pszShell, dwErr);
+ }
+ else
+ {
+ MkWinChildError(pWorker, 1, "%s: not found!\n", pszArg0);
+ dwErr = ERROR_FILE_NOT_FOUND;
+ }
+ }
+ else
+ {
+ dwErr = GetLastError();
+ MkWinChildError(pWorker, 1, _("MultiByteToWideChar failed to convert argv[0] (%s): %u\n"), pszArg0, dwErr);
+ }
+ return dwErr == ERROR_INSUFFICIENT_BUFFER ? ERROR_FILENAME_EXCED_RANGE : dwErr;
+}
+
+/**
+ * Creates the environment block.
+ *
+ * @returns 0 on success, windows error code on failure.
+ * @param pWorker The childcare worker if on one, otherwise NULL.
+ * @param papszEnv The environment vector to convert.
+ * @param cbEnvStrings The size of the environment strings, iff they are
+ * sequential in a block. Otherwise, zero.
+ * @param ppwszEnv Where to return the pointer to the environment
+ * block.
+ * @param ppwszSearchPath Where to return the pointer to the path value
+ * within the environment block. This will not be set
+ * if cbEnvStrings is non-zero, more efficient to let
+ * mkWinChildcareWorkerFindImage() search when needed.
+ */
+static int mkWinChildcareWorkerConvertEnvironment(PWINCHILDCAREWORKER pWorker, char **papszEnv, size_t cbEnvStrings,
+ WCHAR **ppwszEnv, WCHAR const **ppwszSearchPath)
+{
+ DWORD dwErr;
+ int cwcRc;
+ int cwcDst;
+ WCHAR *pwszzDst;
+
+ *ppwszSearchPath = NULL;
+
+ /*
+ * We've got a little optimization here with help from mkWinChildCopyStringArray.
+ */
+ if (cbEnvStrings)
+ {
+ cwcDst = cbEnvStrings + 32;
+ pwszzDst = (WCHAR *)xmalloc(cwcDst * sizeof(WCHAR));
+ cwcRc = MultiByteToWideChar(CP_ACP, 0 /*fFlags*/, papszEnv[0], cbEnvStrings, pwszzDst, cwcDst);
+ if (cwcRc != 0)
+ {
+ *ppwszEnv = pwszzDst;
+ return 0;
+ }
+
+ /* Resize the allocation and try again. */
+ dwErr = GetLastError();
+ if (dwErr == ERROR_INSUFFICIENT_BUFFER)
+ {
+ cwcRc = MultiByteToWideChar(CP_ACP, 0 /*fFlags*/, papszEnv[0], cbEnvStrings, NULL, 0);
+ if (cwcRc > 0)
+ cwcDst = cwcRc + 32;
+ else
+ cwcDst *= 2;
+ pwszzDst = (WCHAR *)xrealloc(pwszzDst, cwcDst);
+ cwcRc = MultiByteToWideChar(CP_ACP, 0 /*fFlags*/, papszEnv[0], cbEnvStrings, pwszzDst, cwcDst);
+ if (cwcRc != 0)
+ {
+ *ppwszEnv = pwszzDst;
+ return 0;
+ }
+ dwErr = GetLastError();
+ }
+ MkWinChildError(pWorker, 1, _("MultiByteToWideChar failed to convert environment block: %u\n"), dwErr);
+ }
+ /*
+ * Need to convert it string by string.
+ */
+ else
+ {
+ size_t offPathValue = ~(size_t)0;
+ size_t offDst;
+
+ /*
+ * Estimate the size first.
+ */
+ size_t cEnvVars;
+ size_t cwcDst = 32;
+ size_t iVar = 0;
+ const char *pszSrc;
+ while ((pszSrc = papszEnv[iVar]) != NULL)
+ {
+ cwcDst += strlen(pszSrc) + 1;
+ iVar++;
+ }
+ cEnvVars = iVar;
+
+ /* Allocate estimated WCHARs and convert the variables one by one, reallocating
+ the block as needed. */
+ pwszzDst = (WCHAR *)xmalloc(cwcDst * sizeof(WCHAR));
+ cwcDst--; /* save one wchar for the terminating empty string. */
+ offDst = 0;
+ for (iVar = 0; iVar < cEnvVars; iVar++)
+ {
+ size_t cwcLeft = cwcDst - offDst;
+ size_t const cbSrc = strlen(pszSrc = papszEnv[iVar]) + 1;
+ assert(cwcDst >= offDst);
+
+
+ cwcRc = MultiByteToWideChar(CP_ACP, 0 /*fFlags*/, pszSrc, cbSrc, &pwszzDst[offDst], cwcLeft);
+ if (cwcRc > 0)
+ { /* likely */ }
+ else
+ {
+ dwErr = GetLastError();
+ if (dwErr == ERROR_INSUFFICIENT_BUFFER)
+ {
+ /* Need more space. So, calc exacly how much and resize the block accordingly. */
+ size_t cbSrc2 = cbSrc;
+ size_t iVar2 = iVar;
+ cwcLeft = 1;
+ for (;;)
+ {
+ size_t cwcRc2 = MultiByteToWideChar(CP_ACP, 0 /*fFlags*/, pszSrc, cbSrc, NULL, 0);
+ if (cwcRc2 > 0)
+ cwcLeft += cwcRc2;
+ else
+ cwcLeft += cbSrc * 4;
+
+ /* advance */
+ iVar2++;
+ if (iVar2 >= cEnvVars)
+ break;
+ pszSrc = papszEnv[iVar2];
+ cbSrc2 = strlen(pszSrc) + 1;
+ }
+ pszSrc = papszEnv[iVar];
+
+ /* Grow the allocation and repeat the conversion. */
+ if (offDst + cwcLeft > cwcDst + 1)
+ {
+ cwcDst = offDst + cwcLeft;
+ pwszzDst = (WCHAR *)xrealloc(pwszzDst, cwcDst * sizeof(WCHAR));
+ cwcDst--; /* save one wchar for the terminating empty string. */
+ cwcRc = MultiByteToWideChar(CP_ACP, 0 /*fFlags*/, pszSrc, cbSrc, &pwszzDst[offDst], cwcLeft - 1);
+ if (cwcRc <= 0)
+ dwErr = GetLastError();
+ }
+ }
+ if (cwcRc <= 0)
+ {
+ MkWinChildError(pWorker, 1, _("MultiByteToWideChar failed to convert environment string #%u (%s): %u\n"),
+ iVar, pszSrc, dwErr);
+ free(pwszzDst);
+ return dwErr;
+ }
+ }
+
+ /* Look for the PATH. */
+ if ( offPathValue == ~(size_t)0
+ && IS_PATH_ENV_VAR(cwcRc, &pwszzDst[offDst]) )
+ offPathValue = offDst + 4 + 1;
+
+ /* Advance. */
+ offDst += cwcRc;
+ }
+ pwszzDst[offDst++] = '\0';
+
+ if (offPathValue != ~(size_t)0)
+ *ppwszSearchPath = &pwszzDst[offPathValue];
+ *ppwszEnv = pwszzDst;
+ return 0;
+ }
+ free(pwszzDst);
+ return dwErr;
+}
+
+/**
+ * Childcare worker: handle regular process.
+ *
+ * @param pWorker The worker.
+ * @param pChild The kSubmit child.
+ */
+static void mkWinChildcareWorkerThreadHandleProcess(PWINCHILDCAREWORKER pWorker, PWINCHILD pChild)
+{
+ WCHAR *pwszSearchPath = NULL;
+ WCHAR *pwszzEnvironment = NULL;
+ WCHAR *pwszCommandLine = NULL;
+ WCHAR *pwszImageName = NULL;
+ BOOL fNeedShell = FALSE;
+ BOOL fProbableClExe = FALSE;
+ int rc;
+
+ /*
+ * First we convert the environment so we get the PATH we need to
+ * search for the executable.
+ */
+ rc = mkWinChildcareWorkerConvertEnvironment(pWorker, pChild->u.Process.papszEnv ? pChild->u.Process.papszEnv : environ,
+ pChild->u.Process.cbEnvStrings,
+ &pwszzEnvironment, &pwszSearchPath);
+ /*
+ * Find the executable and maybe checking if it's a shell script, then
+ * convert it to a command line.
+ */
+ if (rc == 0)
+ rc = mkWinChildcareWorkerFindImage(pWorker, pChild->u.Process.papszArgs[0], pwszSearchPath, pwszzEnvironment,
+ pChild->u.Process.pszShell, &pwszImageName, &fNeedShell, &pChild->fProbableClExe);
+ if (rc == 0)
+ {
+ if (!fNeedShell)
+ rc = mkWinChildcareWorkerConvertCommandline(pWorker, pChild->u.Process.papszArgs, 0 /*fFlags*/, &pwszCommandLine);
+ else
+ rc = mkWinChildcareWorkerConvertCommandlineWithShell(pWorker, pwszImageName, pChild->u.Process.papszArgs,
+ &pwszCommandLine);
+
+ /*
+ * Create the child process.
+ */
+ if (rc == 0)
+ {
+ BOOL afReplace[3] = { FALSE, pChild->u.Process.hStdOut != INVALID_HANDLE_VALUE, pChild->u.Process.hStdErr != INVALID_HANDLE_VALUE };
+ HANDLE ahChild[3] = { INVALID_HANDLE_VALUE, pChild->u.Process.hStdOut, pChild->u.Process.hStdErr };
+ rc = mkWinChildcareWorkerCreateProcess(pWorker, pwszImageName, pwszCommandLine, pwszzEnvironment,
+ NULL /*pwszCwd*/, afReplace, ahChild, pChild->u.Process.fCatchOutput,
+ &pChild->u.Process.hProcess);
+ mkWinChildcareWorkerCloseStandardHandles(pChild);
+ if (rc == 0)
+ {
+ /*
+ * Wait for the child to complete.
+ */
+ mkWinChildcareWorkerWaitForProcess(pWorker, pChild, pChild->u.Process.hProcess, pwszImageName,
+ pChild->u.Process.fCatchOutput);
+ }
+ else
+ pChild->iExitCode = rc;
+ }
+ else
+ pChild->iExitCode = rc;
+ }
+ else
+ pChild->iExitCode = rc;
+ free(pwszCommandLine);
+ free(pwszImageName);
+ free(pwszzEnvironment);
+
+ /* In case we failed, we must make sure the child end of pipes
+ used by $(shell no_such_command.exe) are closed, otherwise
+ the main thread will be stuck reading the parent end. */
+ mkWinChildcareWorkerCloseStandardHandles(pChild);
+}
+
+#ifdef KMK
+
+/**
+ * Childcare worker: handle builtin command.
+ *
+ * @param pWorker The worker.
+ * @param pChild The kSubmit child.
+ */
+static void mkWinChildcareWorkerThreadHandleBuiltIn(PWINCHILDCAREWORKER pWorker, PWINCHILD pChild)
+{
+ PCKMKBUILTINENTRY pBuiltIn = pChild->u.BuiltIn.pBuiltIn;
+ KMKBUILTINCTX Ctx =
+ {
+ pBuiltIn->uName.s.sz,
+ pChild->pMkChild ? &pChild->pMkChild->output : NULL,
+ pWorker,
+ };
+ if (pBuiltIn->uFnSignature == FN_SIG_MAIN)
+ pChild->iExitCode = pBuiltIn->u.pfnMain(pChild->u.BuiltIn.cArgs, pChild->u.BuiltIn.papszArgs,
+ pChild->u.BuiltIn.papszEnv, &Ctx);
+ else if (pBuiltIn->uFnSignature == FN_SIG_MAIN_SPAWNS)
+ pChild->iExitCode = pBuiltIn->u.pfnMainSpawns(pChild->u.BuiltIn.cArgs, pChild->u.BuiltIn.papszArgs,
+ pChild->u.BuiltIn.papszEnv, &Ctx, pChild->pMkChild, NULL /*pPid*/);
+ else
+ {
+ assert(0);
+ pChild->iExitCode = 98;
+ }
+}
+
+/**
+ * Childcare worker: handle append write-out.
+ *
+ * @param pWorker The worker.
+ * @param pChild The kSubmit child.
+ */
+static void mkWinChildcareWorkerThreadHandleAppend(PWINCHILDCAREWORKER pWorker, PWINCHILD pChild)
+{
+ int fd = open(pChild->u.Append.pszFilename,
+ pChild->u.Append.fTruncate
+ ? O_WRONLY | O_TRUNC | O_CREAT | _O_NOINHERIT | _O_BINARY
+ : O_WRONLY | O_APPEND | O_CREAT | _O_NOINHERIT | _O_BINARY,
+ 0666);
+ if (fd >= 0)
+ {
+ ssize_t cbWritten = write(fd, pChild->u.Append.pszAppend, pChild->u.Append.cbAppend);
+ if (cbWritten == (ssize_t)pChild->u.Append.cbAppend)
+ {
+ if (close(fd) >= 0)
+ {
+ pChild->iExitCode = 0;
+ return;
+ }
+ MkWinChildError(pWorker, 1, "kmk_builtin_append: close failed on '%s': %u (%s)\n",
+ pChild->u.Append.pszFilename, errno, strerror(errno));
+ }
+ else
+ MkWinChildError(pWorker, 1, "kmk_builtin_append: error writing %lu bytes to on '%s': %u (%s)\n",
+ pChild->u.Append.cbAppend, pChild->u.Append.pszFilename, errno, strerror(errno));
+ close(fd);
+ }
+ else
+ MkWinChildError(pWorker, 1, "kmk_builtin_append: error opening '%s': %u (%s)\n",
+ pChild->u.Append.pszFilename, errno, strerror(errno));
+ pChild->iExitCode = 1;
+}
+
+/**
+ * Childcare worker: handle kSubmit job.
+ *
+ * @param pWorker The worker.
+ * @param pChild The kSubmit child.
+ */
+static void mkWinChildcareWorkerThreadHandleSubmit(PWINCHILDCAREWORKER pWorker, PWINCHILD pChild)
+{
+ void *pvSubmitWorker = pChild->u.Submit.pvSubmitWorker;
+
+ /*
+ * Prep the wait handles.
+ */
+ HANDLE ahHandles[3] = { pChild->u.Submit.hEvent, NULL, NULL };
+ DWORD cHandles = 1;
+ if (pChild->u.Submit.pStdOut)
+ {
+ assert(pChild->u.Submit.pStdErr);
+ pChild->u.Submit.pStdOut->fHaveWrittenOut = FALSE;
+ ahHandles[cHandles++] = pChild->u.Submit.pStdOut->hEvent;
+ pChild->u.Submit.pStdErr->fHaveWrittenOut = FALSE;
+ ahHandles[cHandles++] = pChild->u.Submit.pStdErr->hEvent;
+ }
+
+ /*
+ * Wait loop.
+ */
+ for (;;)
+ {
+ int iExitCode = -42;
+ int iSignal = -1;
+ DWORD dwStatus;
+ if (cHandles == 1)
+ dwStatus = WaitForSingleObject(ahHandles[0], INFINITE);
+ else
+ {
+ dwStatus = WaitForMultipleObjects(cHandles, ahHandles, FALSE /*fWaitAll*/, INFINITE);
+ assert(dwStatus != WAIT_FAILED);
+ if (dwStatus == WAIT_OBJECT_0 + 1)
+ mkWinChildcareWorkerCatchOutput(pChild, pChild->u.Submit.pStdOut, FALSE /*fDraining*/);
+ else if (dwStatus == WAIT_OBJECT_0 + 2)
+ mkWinChildcareWorkerCatchOutput(pChild, pChild->u.Submit.pStdErr, FALSE /*fDraining*/);
+ }
+ if (kSubmitSubProcGetResult((intptr_t)pvSubmitWorker, dwStatus == WAIT_OBJECT_0 /*fBlock*/, &iExitCode, &iSignal) == 0)
+ {
+ if (pChild->u.Submit.pStdOut)
+ MkWinChildcareWorkerDrainPipes(pChild, pChild->u.Submit.pStdOut, pChild->u.Submit.pStdErr);
+
+ pChild->iExitCode = iExitCode;
+ pChild->iSignal = iSignal;
+ /* Cleanup must be done on the main thread. */
+ return;
+ }
+
+ if (pChild->iSignal != 0)
+ kSubmitSubProcKill((intptr_t)pvSubmitWorker, pChild->iSignal);
+ }
+}
+
+/**
+ * Childcare worker: handle kmk_redirect process.
+ *
+ * @param pWorker The worker.
+ * @param pChild The redirect child.
+ */
+static void mkWinChildcareWorkerThreadHandleRedirect(PWINCHILDCAREWORKER pWorker, PWINCHILD pChild)
+{
+ mkWinChildcareWorkerWaitForProcess(pWorker, pChild, pChild->u.Redirect.hProcess, L"kmk_redirect", FALSE /*fCatchOutput*/);
+}
+
+#endif /* KMK */
+
+/**
+ * Childcare worker thread.
+ *
+ * @returns 0
+ * @param pvUser The worker instance.
+ */
+static unsigned int __stdcall mkWinChildcareWorkerThread(void *pvUser)
+{
+ PWINCHILDCAREWORKER pWorker = (PWINCHILDCAREWORKER)pvUser;
+ assert(pWorker->uMagic == WINCHILDCAREWORKER_MAGIC);
+
+#ifdef MKWINCHILD_DO_SET_PROCESSOR_GROUP
+ /*
+ * Adjust process group if necessary.
+ *
+ * Note! It seems that setting the mask to zero means that we select all
+ * active processors. Couldn't find any alternative API for getting
+ * the correct active processor mask.
+ */
+ if (g_cProcessorGroups > 1)
+ {
+ GROUP_AFFINITY Affinity = { 0 /* == all active apparently */, pWorker->iProcessorGroup, { 0, 0, 0 } };
+ BOOL fRet = g_pfnSetThreadGroupAffinity(GetCurrentThread(), &Affinity, NULL);
+ assert(fRet); (void)fRet;
+# ifndef NDEBUG
+ {
+ GROUP_AFFINITY ActualAffinity = { 0xbeefbeefU, 0xbeef, { 0xbeef, 0xbeef, 0xbeef } };
+ fRet = GetThreadGroupAffinity(GetCurrentThread(), &ActualAffinity);
+ assert(fRet); (void)fRet;
+ assert(ActualAffinity.Group == pWorker->iProcessorGroup);
+ }
+# endif
+ }
+#endif
+
+ /*
+ * Work loop.
+ */
+ while (!g_fShutdown)
+ {
+ /*
+ * Try go idle.
+ */
+ PWINCHILD pChild = pWorker->pTailTodoChildren;
+ if (!pChild)
+ {
+ _InterlockedExchange(&pWorker->fIdle, TRUE);
+ pChild = pWorker->pTailTodoChildren;
+ if (!pChild)
+ {
+ DWORD dwStatus;
+
+ _InterlockedIncrement((long *)&g_cIdleChildcareWorkers);
+ _InterlockedExchange((long *)&g_idxLastChildcareWorker, pWorker->idxWorker);
+ dwStatus = WaitForSingleObject(pWorker->hEvtIdle, INFINITE);
+ _InterlockedExchange(&pWorker->fIdle, FALSE);
+ _InterlockedDecrement((long *)&g_cIdleChildcareWorkers);
+
+ assert(dwStatus != WAIT_FAILED);
+ if (dwStatus == WAIT_FAILED)
+ Sleep(20);
+
+ pChild = pWorker->pTailTodoChildren;
+ }
+ else
+ _InterlockedExchange(&pWorker->fIdle, FALSE);
+ }
+ if (pChild)
+ {
+ /*
+ * We got work to do. First job is to deque the job.
+ */
+ pChild = mkWinChildDequeFromLifo(&pWorker->pTailTodoChildren, pChild);
+ assert(pChild);
+ if (pChild)
+ {
+ PWINCHILD pTailExpect;
+
+ pChild->pWorker = pWorker;
+ pWorker->pCurChild = pChild;
+ switch (pChild->enmType)
+ {
+ case WINCHILDTYPE_PROCESS:
+ mkWinChildcareWorkerThreadHandleProcess(pWorker, pChild);
+ break;
+#ifdef KMK
+ case WINCHILDTYPE_BUILT_IN:
+ mkWinChildcareWorkerThreadHandleBuiltIn(pWorker, pChild);
+ break;
+ case WINCHILDTYPE_APPEND:
+ mkWinChildcareWorkerThreadHandleAppend(pWorker, pChild);
+ break;
+ case WINCHILDTYPE_SUBMIT:
+ mkWinChildcareWorkerThreadHandleSubmit(pWorker, pChild);
+ break;
+ case WINCHILDTYPE_REDIRECT:
+ mkWinChildcareWorkerThreadHandleRedirect(pWorker, pChild);
+ break;
+#endif
+ default:
+ assert(0);
+ }
+ pWorker->pCurChild = NULL;
+ pChild->pWorker = NULL;
+
+ /*
+ * Move the child to the completed list.
+ */
+ pTailExpect = NULL;
+ for (;;)
+ {
+ PWINCHILD pTailActual;
+ pChild->pNext = pTailExpect;
+ pTailActual = _InterlockedCompareExchangePointer(&g_pTailCompletedChildren, pChild, pTailExpect);
+ if (pTailActual != pTailExpect)
+ pTailExpect = pTailActual;
+ else
+ {
+ _InterlockedDecrement(&g_cPendingChildren);
+ if (pTailExpect)
+ break;
+ if (SetEvent(g_hEvtWaitChildren))
+ break;
+ MkWinChildError(pWorker, 1, "SetEvent(g_hEvtWaitChildren=%p) failed: %u\n",
+ g_hEvtWaitChildren, GetLastError());
+ break;
+ }
+ }
+ }
+ }
+ }
+
+ _endthreadex(0);
+ return 0;
+}
+
+/**
+ * Creates a pipe for catching child output.
+ *
+ * This is a custom CreatePipe implementation that allows for overlapped I/O on
+ * our end of the pipe. Silly that they don't offer an API that does this.
+ *
+ * @returns The pipe that was created. NULL on failure.
+ * @param pPipe The structure for the pipe.
+ * @param iWhich Which standard descriptor this is a pipe for.
+ * @param idxWorker The worker index.
+ */
+PWINCCWPIPE MkWinChildcareCreateWorkerPipe(unsigned iWhich, unsigned int idxWorker)
+{
+ /*
+ * We try generate a reasonably unique name from the get go, so this retry
+ * loop shouldn't really ever be needed. But you never know.
+ */
+ static unsigned s_iSeqNo = 0;
+ DWORD const cMaxInstances = 1;
+ DWORD const cbPipe = 4096;
+ DWORD const cMsTimeout = 0;
+ unsigned cTries = 256;
+ while (cTries-- > 0)
+ {
+ /* Create the pipe (our end). */
+ HANDLE hPipeRead;
+ DWORD fOpenMode = PIPE_ACCESS_INBOUND | FILE_FLAG_OVERLAPPED | FILE_FLAG_FIRST_PIPE_INSTANCE;
+ DWORD fPipeMode = PIPE_TYPE_BYTE | PIPE_READMODE_BYTE | PIPE_WAIT | PIPE_REJECT_REMOTE_CLIENTS;
+ WCHAR wszName[MAX_PATH];
+ s_iSeqNo++;
+ _snwprintf(wszName, MAX_PATH, L"\\\\.\\pipe\\kmk-winchildren-%u-%u-%u-%s-%u-%u",
+ GetCurrentProcessId(), GetCurrentThreadId(), idxWorker, iWhich == 1 ? L"out" : L"err", s_iSeqNo, GetTickCount());
+ hPipeRead = CreateNamedPipeW(wszName, fOpenMode, fPipeMode, cMaxInstances, cbPipe, cbPipe, cMsTimeout, NULL /*pSecAttr*/);
+ if (hPipeRead == INVALID_HANDLE_VALUE && GetLastError() == ERROR_INVALID_PARAMETER)
+ {
+ fOpenMode &= ~FILE_FLAG_FIRST_PIPE_INSTANCE;
+ fPipeMode &= ~PIPE_REJECT_REMOTE_CLIENTS;
+ hPipeRead = CreateNamedPipeW(wszName, fOpenMode, fPipeMode, cMaxInstances, cbPipe, cbPipe, cMsTimeout, NULL /*pSecAttr*/);
+ }
+ if (hPipeRead != INVALID_HANDLE_VALUE)
+ {
+ /* Connect the other end. */
+ HANDLE hPipeWrite = CreateFileW(wszName, GENERIC_WRITE | FILE_READ_ATTRIBUTES, 0 /*fShareMode*/, NULL /*pSecAttr*/,
+ OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL /*hTemplateFile*/);
+ if (hPipeWrite != INVALID_HANDLE_VALUE)
+ {
+ /*
+ * Create the event object and we're done.
+ *
+ * It starts in signalled stated so we don't need special code
+ * for handing when we start waiting.
+ */
+ HANDLE hEvent = CreateEventW(NULL /*pSecAttr*/, TRUE /*fManualReset*/, TRUE /*fInitialState*/, NULL /*pwszName*/);
+ if (hEvent != NULL)
+ {
+ PWINCCWPIPE pPipe = (PWINCCWPIPE)xcalloc(sizeof(*pPipe));
+ pPipe->hPipeMine = hPipeRead;
+ pPipe->hPipeChild = hPipeWrite;
+ pPipe->hEvent = hEvent;
+ pPipe->iWhich = iWhich;
+ pPipe->fReadPending = FALSE;
+ pPipe->cbBuffer = cbPipe;
+ pPipe->pbBuffer = xcalloc(cbPipe);
+ return pPipe;
+ }
+
+ CloseHandle(hPipeWrite);
+ CloseHandle(hPipeRead);
+ return NULL;
+ }
+ CloseHandle(hPipeRead);
+ }
+ }
+ return NULL;
+}
+
+/**
+ * Destroys a childcare worker pipe.
+ *
+ * @param pPipe The pipe.
+ */
+void MkWinChildcareDeleteWorkerPipe(PWINCCWPIPE pPipe)
+{
+ if (pPipe->hPipeChild != NULL)
+ {
+ CloseHandle(pPipe->hPipeChild);
+ pPipe->hPipeChild = NULL;
+ }
+
+ if (pPipe->hPipeMine != NULL)
+ {
+ if (pPipe->fReadPending)
+ if (!CancelIo(pPipe->hPipeMine))
+ WaitForSingleObject(pPipe->hEvent, INFINITE);
+ CloseHandle(pPipe->hPipeMine);
+ pPipe->hPipeMine = NULL;
+ }
+
+ if (pPipe->hEvent != NULL)
+ {
+ CloseHandle(pPipe->hEvent);
+ pPipe->hEvent = NULL;
+ }
+
+ if (pPipe->pbBuffer)
+ {
+ free(pPipe->pbBuffer);
+ pPipe->pbBuffer = NULL;
+ }
+}
+
+/**
+ * Initializes the processor group allocator.
+ *
+ * @param pState The allocator to initialize.
+ */
+void MkWinChildInitCpuGroupAllocator(PMKWINCHILDCPUGROUPALLOCSTATE pState)
+{
+ /* We shift the starting group with the make nesting level as part of
+ our very simple distribution strategy. */
+ pState->idxGroup = makelevel;
+ pState->idxProcessorInGroup = 0;
+}
+
+/**
+ * Allocate CPU group for the next child process.
+ *
+ * @returns CPU group.
+ * @param pState The allocator state. Must be initialized by
+ * MkWinChildInitCpuGroupAllocator().
+ */
+unsigned int MkWinChildAllocateCpuGroup(PMKWINCHILDCPUGROUPALLOCSTATE pState)
+{
+ unsigned int iGroup = 0;
+ if (g_cProcessorGroups > 1)
+ {
+ unsigned int cMaxInGroup;
+ unsigned int cInGroup;
+
+ iGroup = pState->idxGroup % g_cProcessorGroups;
+
+ /* Advance. We employ a very simple strategy that does 50% in
+ each group for each group cycle. Odd processor counts are
+ caught in odd group cycles. The init function selects the
+ starting group based on make nesting level to avoid stressing
+ out the first group. */
+ cInGroup = ++pState->idxProcessorInGroup;
+ cMaxInGroup = g_pacProcessorsInGroup[iGroup];
+ if ( !(cMaxInGroup & 1)
+ || !((pState->idxGroup / g_cProcessorGroups) & 1))
+ cMaxInGroup /= 2;
+ else
+ cMaxInGroup = cMaxInGroup / 2 + 1;
+ if (cInGroup >= cMaxInGroup)
+ {
+ pState->idxProcessorInGroup = 0;
+ pState->idxGroup++;
+ }
+ }
+ return iGroup;
+}
+
+/**
+ * Creates another childcare worker.
+ *
+ * @returns The new worker, if we succeeded.
+ */
+static PWINCHILDCAREWORKER mkWinChildcareCreateWorker(void)
+{
+ PWINCHILDCAREWORKER pWorker = (PWINCHILDCAREWORKER)xcalloc(sizeof(*pWorker));
+ pWorker->uMagic = WINCHILDCAREWORKER_MAGIC;
+ pWorker->idxWorker = g_cChildCareworkers;
+ pWorker->hEvtIdle = CreateEventW(NULL, FALSE /*fManualReset*/, FALSE /*fInitialState*/, NULL /*pwszName*/);
+ if (pWorker->hEvtIdle)
+ {
+ pWorker->pStdOut = MkWinChildcareCreateWorkerPipe(1, pWorker->idxWorker);
+ if (pWorker->pStdOut)
+ {
+ pWorker->pStdErr = MkWinChildcareCreateWorkerPipe(2, pWorker->idxWorker);
+ if (pWorker->pStdErr)
+ {
+ /* Before we start the thread, assign it to a processor group. */
+ pWorker->iProcessorGroup = MkWinChildAllocateCpuGroup(&g_ProcessorGroupAllocator);
+
+ /* Try start the thread. */
+ pWorker->hThread = (HANDLE)_beginthreadex(NULL, 0 /*cbStack*/, mkWinChildcareWorkerThread, pWorker,
+ 0, &pWorker->tid);
+ if (pWorker->hThread != NULL)
+ {
+ pWorker->idxWorker = g_cChildCareworkers++; /* paranoia */
+ g_papChildCareworkers[pWorker->idxWorker] = pWorker;
+ return pWorker;
+ }
+
+ /* Bail out! */
+ ONS (error, NILF, "_beginthreadex failed: %u (%s)\n", errno, strerror(errno));
+ MkWinChildcareDeleteWorkerPipe(pWorker->pStdErr);
+ }
+ else
+ ON (error, NILF, "Failed to create stderr pipe: %u\n", GetLastError());
+ MkWinChildcareDeleteWorkerPipe(pWorker->pStdOut);
+ }
+ else
+ ON (error, NILF, "Failed to create stdout pipe: %u\n", GetLastError());
+ CloseHandle(pWorker->hEvtIdle);
+ }
+ else
+ ON (error, NILF, "CreateEvent failed: %u\n", GetLastError());
+ pWorker->uMagic = ~WINCHILDCAREWORKER_MAGIC;
+ free(pWorker);
+ return NULL;
+}
+
+/**
+ * Helper for copying argument and environment vectors.
+ *
+ * @returns Single alloc block copy.
+ * @param papszSrc The source vector.
+ * @param pcbStrings Where to return the size of the strings & terminator.
+ */
+static char **mkWinChildCopyStringArray(char **papszSrc, size_t *pcbStrings)
+{
+ const char *psz;
+ char **papszDstArray;
+ char *pszDstStr;
+ size_t i;
+
+ /* Calc sizes first. */
+ size_t cbStrings = 1; /* (one extra for terminator string) */
+ size_t cStrings = 0;
+ while ((psz = papszSrc[cStrings]) != NULL)
+ {
+ cbStrings += strlen(psz) + 1;
+ cStrings++;
+ }
+ *pcbStrings = cbStrings;
+
+ /* Allocate destination. */
+ papszDstArray = (char **)xmalloc(cbStrings + (cStrings + 1) * sizeof(papszDstArray[0]));
+ pszDstStr = (char *)&papszDstArray[cStrings + 1];
+
+ /* Copy it. */
+ for (i = 0; i < cStrings; i++)
+ {
+ const char *pszSource = papszSrc[i];
+ size_t cchString = strlen(pszSource);
+ papszDstArray[i] = pszDstStr;
+ memcpy(pszDstStr, pszSource, cchString);
+ pszDstStr += cchString;
+ *pszDstStr++ = '\0';
+ }
+ *pszDstStr = '\0';
+ assert(&pszDstStr[1] - papszDstArray[0] == cbStrings);
+ papszDstArray[i] = NULL;
+ return papszDstArray;
+}
+
+/**
+ * Allocate and init a WINCHILD.
+ *
+ * @returns The new windows child structure.
+ * @param enmType The child type.
+ */
+static PWINCHILD mkWinChildNew(WINCHILDTYPE enmType)
+{
+ PWINCHILD pChild = xcalloc(sizeof(*pChild));
+ pChild->enmType = enmType;
+ pChild->fCoreDumped = 0;
+ pChild->iSignal = 0;
+ pChild->iExitCode = 222222;
+ pChild->uMagic = WINCHILD_MAGIC;
+ pChild->pid = (intptr_t)pChild;
+ return pChild;
+}
+
+/**
+ * Destructor for WINCHILD.
+ *
+ * @param pChild The child structure to destroy.
+ */
+static void mkWinChildDelete(PWINCHILD pChild)
+{
+ assert(pChild->uMagic == WINCHILD_MAGIC);
+ pChild->uMagic = ~WINCHILD_MAGIC;
+
+ switch (pChild->enmType)
+ {
+ case WINCHILDTYPE_PROCESS:
+ {
+ if (pChild->u.Process.papszArgs)
+ {
+ free(pChild->u.Process.papszArgs);
+ pChild->u.Process.papszArgs = NULL;
+ }
+ if (pChild->u.Process.cbEnvStrings && pChild->u.Process.papszEnv)
+ {
+ free(pChild->u.Process.papszEnv);
+ pChild->u.Process.papszEnv = NULL;
+ }
+ if (pChild->u.Process.pszShell)
+ {
+ free(pChild->u.Process.pszShell);
+ pChild->u.Process.pszShell = NULL;
+ }
+ if (pChild->u.Process.hProcess)
+ {
+ CloseHandle(pChild->u.Process.hProcess);
+ pChild->u.Process.hProcess = NULL;
+ }
+ mkWinChildcareWorkerCloseStandardHandles(pChild);
+ break;
+ }
+
+#ifdef KMK
+ case WINCHILDTYPE_BUILT_IN:
+ if (pChild->u.BuiltIn.papszArgs)
+ {
+ free(pChild->u.BuiltIn.papszArgs);
+ pChild->u.BuiltIn.papszArgs = NULL;
+ }
+ if (pChild->u.BuiltIn.papszEnv)
+ {
+ free(pChild->u.BuiltIn.papszEnv);
+ pChild->u.BuiltIn.papszEnv = NULL;
+ }
+ break;
+
+ case WINCHILDTYPE_APPEND:
+ if (pChild->u.Append.pszFilename)
+ {
+ free(pChild->u.Append.pszFilename);
+ pChild->u.Append.pszFilename = NULL;
+ }
+ if (pChild->u.Append.pszAppend)
+ {
+ free(pChild->u.Append.pszAppend);
+ pChild->u.Append.pszAppend = NULL;
+ }
+ break;
+
+ case WINCHILDTYPE_SUBMIT:
+ if (pChild->u.Submit.pvSubmitWorker)
+ {
+ kSubmitSubProcCleanup((intptr_t)pChild->u.Submit.pvSubmitWorker);
+ pChild->u.Submit.pvSubmitWorker = NULL;
+ }
+ break;
+
+ case WINCHILDTYPE_REDIRECT:
+ if (pChild->u.Redirect.hProcess)
+ {
+ CloseHandle(pChild->u.Redirect.hProcess);
+ pChild->u.Redirect.hProcess = NULL;
+ }
+ break;
+#endif /* KMK */
+
+ default:
+ assert(0);
+ }
+
+ free(pChild);
+}
+
+/**
+ * Queues the child with a worker, creating new workers if necessary.
+ *
+ * @returns 0 on success, windows error code on failure (child destroyed).
+ * @param pChild The child.
+ * @param pPid Where to return the PID (optional).
+ */
+static int mkWinChildPushToCareWorker(PWINCHILD pChild, pid_t *pPid)
+{
+ PWINCHILDCAREWORKER pWorker = NULL;
+ PWINCHILD pOldChild;
+ PWINCHILD pCurChild;
+
+ /*
+ * There are usually idle workers around, except for at the start.
+ */
+ if (g_cIdleChildcareWorkers > 0)
+ {
+ /*
+ * Try the idle hint first and move forward from it.
+ */
+ unsigned int const cWorkers = g_cChildCareworkers;
+ unsigned int iHint = g_idxLastChildcareWorker;
+ unsigned int i;
+ for (i = iHint; i < cWorkers; i++)
+ {
+ PWINCHILDCAREWORKER pPossibleWorker = g_papChildCareworkers[i];
+ if (pPossibleWorker->fIdle)
+ {
+ pWorker = pPossibleWorker;
+ break;
+ }
+ }
+ if (!pWorker)
+ {
+ /* Scan from the start. */
+ if (iHint > cWorkers)
+ iHint = cWorkers;
+ for (i = 0; i < iHint; i++)
+ {
+ PWINCHILDCAREWORKER pPossibleWorker = g_papChildCareworkers[i];
+ if (pPossibleWorker->fIdle)
+ {
+ pWorker = pPossibleWorker;
+ break;
+ }
+ }
+ }
+ }
+ if (!pWorker)
+ {
+ /*
+ * Try create more workers if we haven't reached the max yet.
+ */
+ if (g_cChildCareworkers < g_cChildCareworkersMax)
+ pWorker = mkWinChildcareCreateWorker();
+
+ /*
+ * Queue it with an existing worker. Look for one without anthing extra scheduled.
+ */
+ if (!pWorker)
+ {
+ unsigned int i = g_cChildCareworkers;
+ if (i == 0)
+ fatal(NILF, 0, _("Failed to create worker threads for managing child processes!\n"));
+ pWorker = g_papChildCareworkers[--i];
+ if (pWorker->pTailTodoChildren)
+ while (i-- > 0)
+ {
+ PWINCHILDCAREWORKER pPossibleWorker = g_papChildCareworkers[i];
+ if (!pPossibleWorker->pTailTodoChildren)
+ {
+ pWorker = pPossibleWorker;
+ break;
+ }
+ }
+ }
+ }
+
+ /*
+ * Do the queueing.
+ */
+ pOldChild = NULL;
+ for (;;)
+ {
+ pChild->pNext = pOldChild;
+ pCurChild = _InterlockedCompareExchangePointer((void **)&pWorker->pTailTodoChildren, pChild, pOldChild);
+ if (pCurChild == pOldChild)
+ {
+ DWORD volatile dwErr;
+ _InterlockedIncrement(&g_cPendingChildren);
+ if ( !pWorker->fIdle
+ || SetEvent(pWorker->hEvtIdle))
+ {
+ *pPid = pChild->pid;
+ return 0;
+ }
+
+ _InterlockedDecrement(&g_cPendingChildren);
+ dwErr = GetLastError();
+ assert(0);
+ mkWinChildDelete(pChild);
+ return dwErr ? dwErr : -20;
+ }
+ pOldChild = pCurChild;
+ }
+}
+
+/**
+ * Creates a regular child process (job.c).
+ *
+ * Will copy the information and push it to a childcare thread that does the
+ * actual process creation.
+ *
+ * @returns 0 on success, windows status code on failure.
+ * @param papszArgs The arguments.
+ * @param papszEnv The environment (optional).
+ * @param pszShell The SHELL variable value (optional).
+ * @param pMkChild The make child structure (optional).
+ * @param pPid Where to return the pid.
+ */
+int MkWinChildCreate(char **papszArgs, char **papszEnv, const char *pszShell, struct child *pMkChild, pid_t *pPid)
+{
+ PWINCHILD pChild = mkWinChildNew(WINCHILDTYPE_PROCESS);
+ pChild->pMkChild = pMkChild;
+
+ pChild->u.Process.papszArgs = mkWinChildCopyStringArray(papszArgs, &pChild->u.Process.cbArgsStrings);
+ if ( !papszEnv
+ || !pMkChild
+ || pMkChild->environment == papszEnv)
+ {
+ pChild->u.Process.cbEnvStrings = 0;
+ pChild->u.Process.papszEnv = papszEnv;
+ }
+ else
+ pChild->u.Process.papszEnv = mkWinChildCopyStringArray(papszEnv, &pChild->u.Process.cbEnvStrings);
+ if (pszShell)
+ pChild->u.Process.pszShell = xstrdup(pszShell);
+ pChild->u.Process.hStdOut = INVALID_HANDLE_VALUE;
+ pChild->u.Process.hStdErr = INVALID_HANDLE_VALUE;
+
+ /* We always catch the output in order to prevent character soups courtesy
+ of the microsoft CRT and/or linkers writing character by character to the
+ console. Always try write whole lines, even when --output-sync is none. */
+ pChild->u.Process.fCatchOutput = TRUE;
+
+ return mkWinChildPushToCareWorker(pChild, pPid);
+}
+
+/**
+ * Creates a chile process with a pipe hooked up to stdout.
+ *
+ * @returns 0 on success, non-zero on failure.
+ * @param papszArgs The argument vector.
+ * @param papszEnv The environment vector (optional).
+ * @param fdErr File descriptor to hook up to stderr.
+ * @param pPid Where to return the pid.
+ * @param pfdReadPipe Where to return the read end of the pipe.
+ */
+int MkWinChildCreateWithStdOutPipe(char **papszArgs, char **papszEnv, int fdErr, pid_t *pPid, int *pfdReadPipe)
+{
+ /*
+ * Create the pipe.
+ */
+ HANDLE hReadPipe;
+ HANDLE hWritePipe;
+ if (CreatePipe(&hReadPipe, &hWritePipe, NULL, 0 /* default size */))
+ {
+ //if (SetHandleInformation(hWritePipe, HANDLE_FLAG_INHERIT /* clear */ , HANDLE_FLAG_INHERIT /*set*/))
+ {
+ int fdReadPipe = _open_osfhandle((intptr_t)hReadPipe, O_RDONLY);
+ if (fdReadPipe >= 0)
+ {
+ PWINCHILD pChild;
+ int rc;
+
+ /*
+ * Get a handle for fdErr. Ignore failure.
+ */
+ HANDLE hStdErr = INVALID_HANDLE_VALUE;
+ if (fdErr >= 0)
+ {
+ HANDLE hNative = (HANDLE)_get_osfhandle(fdErr);
+ if (!DuplicateHandle(GetCurrentProcess(), hNative, GetCurrentProcess(),
+ &hStdErr, 0 /*DesiredAccess*/, TRUE /*fInherit*/, DUPLICATE_SAME_ACCESS))
+ {
+ ONN(error, NILF, _("DuplicateHandle failed on stderr descriptor (%u): %u\n"), fdErr, GetLastError());
+ hStdErr = INVALID_HANDLE_VALUE;
+ }
+ }
+
+ /*
+ * Push it off to the worker thread.
+ */
+ pChild = mkWinChildNew(WINCHILDTYPE_PROCESS);
+ pChild->u.Process.papszArgs = mkWinChildCopyStringArray(papszArgs, &pChild->u.Process.cbArgsStrings);
+ pChild->u.Process.papszEnv = mkWinChildCopyStringArray(papszEnv ? papszEnv : environ,
+ &pChild->u.Process.cbEnvStrings);
+ //if (pszShell)
+ // pChild->u.Process.pszShell = xstrdup(pszShell);
+ pChild->u.Process.hStdOut = hWritePipe;
+ pChild->u.Process.hStdErr = hStdErr;
+ pChild->u.Process.fCloseStdErr = TRUE;
+ pChild->u.Process.fCloseStdOut = TRUE;
+
+ rc = mkWinChildPushToCareWorker(pChild, pPid);
+ if (rc == 0)
+ *pfdReadPipe = fdReadPipe;
+ else
+ {
+ ON(error, NILF, _("mkWinChildPushToCareWorker failed on pipe: %d\n"), rc);
+ close(fdReadPipe);
+ *pfdReadPipe = -1;
+ *pPid = -1;
+ }
+ return rc;
+ }
+
+ ON(error, NILF, _("_open_osfhandle failed on pipe: %u\n"), errno);
+ }
+ //else
+ // ON(error, NILF, _("SetHandleInformation failed on pipe: %u\n"), GetLastError());
+ if (hReadPipe != INVALID_HANDLE_VALUE)
+ CloseHandle(hReadPipe);
+ CloseHandle(hWritePipe);
+ }
+ else
+ ON(error, NILF, _("CreatePipe failed: %u\n"), GetLastError());
+ *pfdReadPipe = -1;
+ *pPid = -1;
+ return -1;
+}
+
+#ifdef KMK
+
+/**
+ * Interface used by kmkbuiltin.c for executing builtin commands on threads.
+ *
+ * @returns 0 on success, windows status code on failure.
+ * @param pBuiltIn The kmk built-in command entry.
+ * @param cArgs The number of arguments in papszArgs.
+ * @param papszArgs The argument vector.
+ * @param papszEnv The environment vector, optional.
+ * @param pMkChild The make child structure.
+ * @param pPid Where to return the pid.
+ */
+int MkWinChildCreateBuiltIn(PCKMKBUILTINENTRY pBuiltIn, int cArgs, char **papszArgs, char **papszEnv,
+ struct child *pMkChild, pid_t *pPid)
+{
+ size_t cbIgnored;
+ PWINCHILD pChild = mkWinChildNew(WINCHILDTYPE_BUILT_IN);
+ pChild->pMkChild = pMkChild;
+ pChild->u.BuiltIn.pBuiltIn = pBuiltIn;
+ pChild->u.BuiltIn.cArgs = cArgs;
+ pChild->u.BuiltIn.papszArgs = mkWinChildCopyStringArray(papszArgs, &cbIgnored);
+ pChild->u.BuiltIn.papszEnv = papszEnv ? mkWinChildCopyStringArray(papszEnv, &cbIgnored) : NULL;
+ return mkWinChildPushToCareWorker(pChild, pPid);
+}
+
+/**
+ * Interface used by append.c for do the slow file system part.
+ *
+ * This will append the given buffer to the specified file and free the buffer.
+ *
+ * @returns 0 on success, windows status code on failure.
+ *
+ * @param pszFilename The name of the file to append to.
+ * @param ppszAppend What to append. The pointer pointed to is set to
+ * NULL once we've taken ownership of the buffer and
+ * promise to free it.
+ * @param cbAppend How much to append.
+ * @param fTruncate Whether to truncate the file before appending to it.
+ * @param pMkChild The make child structure.
+ * @param pPid Where to return the pid.
+ */
+int MkWinChildCreateAppend(const char *pszFilename, char **ppszAppend, size_t cbAppend, int fTruncate,
+ struct child *pMkChild, pid_t *pPid)
+{
+ size_t cbFilename = strlen(pszFilename) + 1;
+ PWINCHILD pChild = mkWinChildNew(WINCHILDTYPE_APPEND);
+ pChild->pMkChild = pMkChild;
+ pChild->u.Append.fTruncate = fTruncate;
+ pChild->u.Append.pszAppend = *ppszAppend;
+ pChild->u.Append.cbAppend = cbAppend;
+ pChild->u.Append.pszFilename = (char *)memcpy(xmalloc(cbFilename), pszFilename, cbFilename);
+ *ppszAppend = NULL;
+ return mkWinChildPushToCareWorker(pChild, pPid);
+}
+
+/**
+ * Interface used by kSubmit.c for registering stuff to wait on.
+ *
+ * @returns 0 on success, windows status code on failure.
+ * @param hEvent The event object handle to wait on.
+ * @param pvSubmitWorker The argument to pass back to kSubmit to clean up.
+ * @param pStdOut Standard output pipe for the worker. Optional.
+ * @param pStdErr Standard error pipe for the worker. Optional.
+ * @param pMkChild The make child structure.
+ * @param pPid Where to return the pid.
+ */
+int MkWinChildCreateSubmit(intptr_t hEvent, void *pvSubmitWorker, PWINCCWPIPE pStdOut, PWINCCWPIPE pStdErr,
+ struct child *pMkChild, pid_t *pPid)
+{
+ PWINCHILD pChild = mkWinChildNew(WINCHILDTYPE_SUBMIT);
+ pChild->pMkChild = pMkChild;
+ pChild->u.Submit.hEvent = (HANDLE)hEvent;
+ pChild->u.Submit.pvSubmitWorker = pvSubmitWorker;
+ pChild->u.Submit.pStdOut = pStdOut;
+ pChild->u.Submit.pStdErr = pStdErr;
+ return mkWinChildPushToCareWorker(pChild, pPid);
+}
+
+/**
+ * Interface used by redirect.c for registering stuff to wait on.
+ *
+ * @returns 0 on success, windows status code on failure.
+ * @param hProcess The process object to wait on.
+ * @param pPid Where to return the pid.
+ */
+int MkWinChildCreateRedirect(intptr_t hProcess, pid_t *pPid)
+{
+ PWINCHILD pChild = mkWinChildNew(WINCHILDTYPE_REDIRECT);
+ pChild->u.Redirect.hProcess = (HANDLE)hProcess;
+ return mkWinChildPushToCareWorker(pChild, pPid);
+}
+
+
+/**
+ * New interface used by redirect.c for spawning and waitin on a child.
+ *
+ * This interface is only used when kmk_builtin_redirect is already running on
+ * a worker thread.
+ *
+ * @returns exit status.
+ * @param pvWorker The worker instance.
+ * @param pszExecutable The executable image to run.
+ * @param papszArgs Argument vector.
+ * @param fQuotedArgv Whether the argument vector is already quoted and
+ * just need some space to be turned into a command
+ * line.
+ * @param papszEnvVars Environment vector.
+ * @param pszCwd The working directory of the child. Optional.
+ * @param pafReplace Which standard handles to replace. Maybe modified!
+ * @param pahReplace The replacement handles. Maybe modified!
+ *
+ */
+int MkWinChildBuiltInExecChild(void *pvWorker, const char *pszExecutable, char **papszArgs, BOOL fQuotedArgv,
+ char **papszEnvVars, const char *pszCwd, BOOL pafReplace[3], HANDLE pahReplace[3])
+{
+ PWINCHILDCAREWORKER pWorker = (PWINCHILDCAREWORKER)pvWorker;
+ WCHAR *pwszSearchPath = NULL;
+ WCHAR *pwszzEnvironment = NULL;
+ WCHAR *pwszCommandLine = NULL;
+ WCHAR *pwszImageName = NULL;
+ WCHAR *pwszCwd = NULL;
+ BOOL fNeedShell = FALSE;
+ PWINCHILD pChild;
+ int rc;
+ assert(pWorker->uMagic == WINCHILDCAREWORKER_MAGIC);
+ pChild = pWorker->pCurChild;
+ assert(pChild != NULL && pChild->uMagic == WINCHILD_MAGIC);
+
+ /*
+ * Convert the CWD first since it's optional and we don't need to clean
+ * up anything here if it fails.
+ */
+ if (pszCwd)
+ {
+ size_t cchCwd = strlen(pszCwd);
+ int cwcCwd = MultiByteToWideChar(CP_ACP, 0 /*fFlags*/, pszCwd, cchCwd + 1, NULL, 0);
+ pwszCwd = xmalloc((cwcCwd + 1) * sizeof(WCHAR)); /* (+1 in case cwcCwd is 0) */
+ cwcCwd = MultiByteToWideChar(CP_ACP, 0 /*fFlags*/, pszCwd, cchCwd + 1, pwszCwd, cwcCwd + 1);
+ if (!cwcCwd)
+ {
+ rc = GetLastError();
+ MkWinChildError(pWorker, 1, _("MultiByteToWideChar failed to convert CWD (%s): %u\n"), pszCwd, (unsigned)rc);
+ return rc;
+ }
+ }
+
+ /*
+ * Before we search for the image, we convert the environment so we don't
+ * have to traverse it twice to find the PATH.
+ */
+ rc = mkWinChildcareWorkerConvertEnvironment(pWorker, papszEnvVars ? papszEnvVars : environ, 0/*cbEnvStrings*/,
+ &pwszzEnvironment, &pwszSearchPath);
+ /*
+ * Find the executable and maybe checking if it's a shell script, then
+ * convert it to a command line.
+ */
+ if (rc == 0)
+ rc = mkWinChildcareWorkerFindImage(pWorker, pszExecutable, pwszSearchPath, pwszzEnvironment, NULL /*pszShell*/,
+ &pwszImageName, &fNeedShell, &pChild->fProbableClExe);
+ if (rc == 0)
+ {
+ assert(!fNeedShell);
+ if (!fQuotedArgv)
+ rc = mkWinChildcareWorkerConvertCommandline(pWorker, papszArgs, 0 /*fFlags*/, &pwszCommandLine);
+ else
+ rc = mkWinChildcareWorkerConvertQuotedArgvToCommandline(pWorker, papszArgs, &pwszCommandLine);
+
+ /*
+ * Create the child process.
+ */
+ if (rc == 0)
+ {
+ HANDLE hProcess;
+ rc = mkWinChildcareWorkerCreateProcess(pWorker, pwszImageName, pwszCommandLine, pwszzEnvironment,
+ pwszCwd, pafReplace, pahReplace, TRUE /*fCatchOutput*/, &hProcess);
+ if (rc == 0)
+ {
+ /*
+ * Wait for the child to complete.
+ */
+ rc = mkWinChildcareWorkerWaitForProcess(pWorker, pChild, hProcess, pwszImageName, TRUE /*fCatchOutput*/);
+ CloseHandle(hProcess);
+ }
+ }
+ }
+
+ free(pwszCwd);
+ free(pwszCommandLine);
+ free(pwszImageName);
+ free(pwszzEnvironment);
+
+ return rc;
+}
+
+#endif /* CONFIG_NEW_WIN_CHILDREN */
+
+/**
+ * Interface used to kill process when processing Ctrl-C and fatal errors.
+ *
+ * @returns 0 on success, -1 & errno on error.
+ * @param pid The process to kill (PWINCHILD).
+ * @param iSignal What to kill it with.
+ * @param pMkChild The make child structure for validation.
+ */
+int MkWinChildKill(pid_t pid, int iSignal, struct child *pMkChild)
+{
+ PWINCHILD pChild = (PWINCHILD)pid;
+ if (pChild)
+ {
+ assert(pChild->uMagic == WINCHILD_MAGIC);
+ if (pChild->uMagic == WINCHILD_MAGIC)
+ {
+ switch (pChild->enmType)
+ {
+ case WINCHILDTYPE_PROCESS:
+ assert(pChild->pMkChild == pMkChild);
+ TerminateProcess(pChild->u.Process.hProcess, DBG_TERMINATE_PROCESS);
+ pChild->iSignal = iSignal;
+ break;
+#ifdef KMK
+ case WINCHILDTYPE_SUBMIT:
+ {
+ pChild->iSignal = iSignal;
+ SetEvent(pChild->u.Submit.hEvent);
+ break;
+ }
+
+ case WINCHILDTYPE_REDIRECT:
+ TerminateProcess(pChild->u.Redirect.hProcess, DBG_TERMINATE_PROCESS);
+ pChild->iSignal = iSignal;
+ break;
+
+ case WINCHILDTYPE_BUILT_IN:
+ break;
+
+#endif /* KMK */
+ default:
+ assert(0);
+ }
+ }
+ }
+ return -1;
+}
+
+/**
+ * Wait for a child process to complete
+ *
+ * @returns 0 on success, windows error code on failure.
+ * @param fBlock Whether to block.
+ * @param pPid Where to return the pid if a child process
+ * completed. This is set to zero if none.
+ * @param piExitCode Where to return the exit code.
+ * @param piSignal Where to return the exit signal number.
+ * @param pfCoreDumped Where to return the core dumped indicator.
+ * @param ppMkChild Where to return the associated struct child pointer.
+ */
+int MkWinChildWait(int fBlock, pid_t *pPid, int *piExitCode, int *piSignal, int *pfCoreDumped, struct child **ppMkChild)
+{
+ PWINCHILD pChild;
+
+ *pPid = 0;
+ *piExitCode = -222222;
+ *pfCoreDumped = 0;
+ *ppMkChild = NULL;
+
+ /*
+ * Wait if necessary.
+ */
+ if (fBlock && !g_pTailCompletedChildren && g_cPendingChildren > 0)
+ {
+ DWORD dwStatus = WaitForSingleObject(g_hEvtWaitChildren, INFINITE);
+ if (dwStatus == WAIT_FAILED)
+ return (int)GetLastError();
+ }
+
+ /*
+ * Try unlink the last child in the LIFO.
+ */
+ pChild = g_pTailCompletedChildren;
+ if (!pChild)
+ return 0;
+ pChild = mkWinChildDequeFromLifo(&g_pTailCompletedChildren, pChild);
+ assert(pChild);
+
+ /*
+ * Set return values and ditch the child structure.
+ */
+ *pPid = pChild->pid;
+ *piExitCode = pChild->iExitCode;
+ *pfCoreDumped = pChild->fCoreDumped;
+ *ppMkChild = pChild->pMkChild;
+ switch (pChild->enmType)
+ {
+ case WINCHILDTYPE_PROCESS:
+ break;
+#ifdef KMK
+ case WINCHILDTYPE_BUILT_IN:
+ case WINCHILDTYPE_APPEND:
+ case WINCHILDTYPE_SUBMIT:
+ case WINCHILDTYPE_REDIRECT:
+ break;
+#endif /* KMK */
+ default:
+ assert(0);
+ }
+ mkWinChildDelete(pChild);
+
+#ifdef KMK
+ /* Flush the volatile directory cache. */
+ dir_cache_invalid_after_job();
+#endif
+ return 0;
+}
+
+/**
+ * Get the child completed event handle.
+ *
+ * Needed when w32os.c is waiting for a job token to become available, given
+ * that completed children is the typical source of these tokens (esp. for kmk).
+ *
+ * @returns Zero if no active children, event handle if waiting is required.
+ */
+intptr_t MkWinChildGetCompleteEventHandle(void)
+{
+ /* We don't return the handle if we've got completed children. This
+ is a safe guard against being called twice in a row without any
+ MkWinChildWait call inbetween. */
+ if (!g_pTailCompletedChildren)
+ return (intptr_t)g_hEvtWaitChildren;
+ return 0;
+}
+
+/**
+ * Emulate execv() for restarting kmk after one or more makefiles has been made.
+ *
+ * Does not return.
+ *
+ * @param papszArgs The arguments.
+ * @param papszEnv The environment.
+ */
+void MkWinChildReExecMake(char **papszArgs, char **papszEnv)
+{
+ PROCESS_INFORMATION ProcInfo;
+ STARTUPINFOW StartupInfo;
+ WCHAR *pwszCommandLine;
+ WCHAR *pwszzEnvironment;
+ WCHAR *pwszPathIgnored;
+ int rc;
+
+ /*
+ * Get the executable name.
+ */
+ WCHAR wszImageName[MKWINCHILD_MAX_PATH];
+ DWORD cwcImageName = GetModuleFileNameW(GetModuleHandle(NULL), wszImageName, MKWINCHILD_MAX_PATH);
+ if (cwcImageName == 0)
+ ON(fatal, NILF, _("MkWinChildReExecMake: GetModuleFileName failed: %u\n"), GetLastError());
+
+ /*
+ * Create the command line and environment.
+ */
+ rc = mkWinChildcareWorkerConvertCommandline(NULL, papszArgs, 0 /*fFlags*/, &pwszCommandLine);
+ if (rc != 0)
+ ON(fatal, NILF, _("MkWinChildReExecMake: mkWinChildcareWorkerConvertCommandline failed: %u\n"), rc);
+
+ rc = mkWinChildcareWorkerConvertEnvironment(NULL, papszEnv ? papszEnv : environ, 0 /*cbEnvStrings*/,
+ &pwszzEnvironment, &pwszPathIgnored);
+ if (rc != 0)
+ ON(fatal, NILF, _("MkWinChildReExecMake: mkWinChildcareWorkerConvertEnvironment failed: %u\n"), rc);
+
+#ifdef KMK
+ /*
+ * Flush the file system cache to avoid messing up tools fetching
+ * going on in the "exec'ed" make by keeping directories open.
+ */
+ dir_cache_invalid_all_and_close_dirs(1);
+#endif
+
+ /*
+ * Fill out the startup info and try create the process.
+ */
+ memset(&ProcInfo, 0, sizeof(ProcInfo));
+ memset(&StartupInfo, 0, sizeof(StartupInfo));
+ StartupInfo.cb = sizeof(StartupInfo);
+ GetStartupInfoW(&StartupInfo);
+ if (!CreateProcessW(wszImageName, pwszCommandLine, NULL /*pProcSecAttr*/, NULL /*pThreadSecAttr*/,
+ TRUE /*fInheritHandles*/, CREATE_UNICODE_ENVIRONMENT, pwszzEnvironment, NULL /*pwsz*/,
+ &StartupInfo, &ProcInfo))
+ ON(fatal, NILF, _("MkWinChildReExecMake: CreateProcessW failed: %u\n"), GetLastError());
+ CloseHandle(ProcInfo.hThread);
+
+ /*
+ * Wait for it to complete and forward the status code to our parent.
+ */
+ for (;;)
+ {
+ DWORD dwExitCode = -2222;
+ DWORD dwStatus = WaitForSingleObject(ProcInfo.hProcess, INFINITE);
+ if ( dwStatus == WAIT_IO_COMPLETION
+ || dwStatus == WAIT_TIMEOUT /* whatever */)
+ continue; /* however unlikely, these aren't fatal. */
+
+ /* Get the status code and terminate. */
+ if (dwStatus == WAIT_OBJECT_0)
+ {
+ if (!GetExitCodeProcess(ProcInfo.hProcess, &dwExitCode))
+ {
+ ON(fatal, NILF, _("MkWinChildReExecMake: GetExitCodeProcess failed: %u\n"), GetLastError());
+ dwExitCode = -2222;
+ }
+ }
+ else if (dwStatus)
+ dwExitCode = dwStatus;
+
+ CloseHandle(ProcInfo.hProcess);
+ for (;;)
+ exit(dwExitCode);
+ }
+}
+
+#ifdef WITH_RW_LOCK
+/** Serialization with kmkbuiltin_redirect. */
+void MkWinChildExclusiveAcquire(void)
+{
+ AcquireSRWLockExclusive(&g_RWLock);
+}
+
+/** Serialization with kmkbuiltin_redirect. */
+void MkWinChildExclusiveRelease(void)
+{
+ ReleaseSRWLockExclusive(&g_RWLock);
+}
+#endif /* WITH_RW_LOCK */
+
+/**
+ * Implementation of the CLOSE_ON_EXEC macro.
+ *
+ * @returns errno value.
+ * @param fd The file descriptor to hide from children.
+ */
+int MkWinChildUnrelatedCloseOnExec(int fd)
+{
+ if (fd >= 0)
+ {
+ HANDLE hNative = (HANDLE)_get_osfhandle(fd);
+ if (hNative != INVALID_HANDLE_VALUE && hNative != NULL)
+ {
+ if (SetHandleInformation(hNative, HANDLE_FLAG_INHERIT /*clear*/ , 0 /*set*/))
+ return 0;
+ }
+ return errno;
+ }
+ return EINVAL;
+}
+
diff --git a/src/kmk/w32/winchildren.h b/src/kmk/w32/winchildren.h
new file mode 100644
index 0000000..e70bd1e
--- /dev/null
+++ b/src/kmk/w32/winchildren.h
@@ -0,0 +1,115 @@
+/* $Id: winchildren.h 3313 2020-03-16 02:31:38Z bird $ */
+/** @file
+ * Child process creation and management for kmk.
+ */
+
+/*
+ * Copyright (c) 2018 knut st. osmundsen <bird-kBuild-spamx@anduin.net>
+ *
+ * This file is part of kBuild.
+ *
+ * kBuild 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 3 of the License, or
+ * (at your option) any later version.
+ *
+ * kBuild 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 kBuild. If not, see <http://www.gnu.org/licenses/>
+ *
+ */
+
+#ifndef INCLUDED_WINCHILDREN_H
+#define INCLUDED_WINCHILDREN_H
+
+/** Child processor group allocator state. */
+typedef struct MKWINCHILDCPUGROUPALLOCSTATE
+{
+ /** The group index for the worker allocator.
+ * This is ever increasing and must be modded by g_cProcessorGroups. */
+ unsigned int idxGroup;
+ /** The processor in group index for the worker allocator. */
+ unsigned int idxProcessorInGroup;
+} MKWINCHILDCPUGROUPALLOCSTATE;
+/** Pointer to a CPU group allocator state. */
+typedef MKWINCHILDCPUGROUPALLOCSTATE *PMKWINCHILDCPUGROUPALLOCSTATE;
+
+#ifdef DECLARE_HANDLE
+/**
+ * A childcare worker pipe.
+ */
+typedef struct WINCCWPIPE
+{
+ /** My end of the pipe. */
+ HANDLE hPipeMine;
+ /** The child end of the pipe. */
+ HANDLE hPipeChild;
+ /** The event for asynchronous reading. */
+ HANDLE hEvent;
+ /** Which pipe this is (1 == stdout, 2 == stderr). */
+ unsigned char iWhich;
+ /** Set if we've got a read pending already. */
+ BOOL fReadPending;
+ /** Indicator that we've written out something. This is cleared before
+ * we start catching output from a new child and use in the CL.exe
+ * supression heuristics. */
+ BOOL fHaveWrittenOut;
+ /** Number of bytes at the start of the buffer that we've already
+ * written out. We try write out whole lines. */
+ DWORD cbWritten;
+ /** The buffer offset of the read currently pending. */
+ DWORD offPendingRead;
+ /** Read buffer size. */
+ DWORD cbBuffer;
+ /** The read buffer allocation. */
+ unsigned char *pbBuffer;
+ /** Overlapped I/O structure. */
+ OVERLAPPED Overlapped;
+} WINCCWPIPE;
+#endif
+
+typedef struct WINCCWPIPE *PWINCCWPIPE;
+
+void MkWinChildInit(unsigned int cJobSlot);
+void MkWinChildReExecMake(char **papszArgs, char **papszEnv);
+intptr_t MkWinChildGetCompleteEventHandle(void);
+int MkWinChildCreate(char **papszArgs, char **papszEnv, const char *pszShell, struct child *pMkChild, pid_t *pPid);
+int MkWinChildCreateWithStdOutPipe(char **papszArgs, char **papszEnv, int fdErr, pid_t *pPid, int *pfdReadPipe);
+void MkWinChildInitCpuGroupAllocator(PMKWINCHILDCPUGROUPALLOCSTATE pState);
+unsigned int MkWinChildAllocateCpuGroup(PMKWINCHILDCPUGROUPALLOCSTATE pState);
+
+#ifdef KMK
+struct KMKBUILTINENTRY;
+int MkWinChildCreateBuiltIn(struct KMKBUILTINENTRY const *pBuiltIn, int cArgs, char **papszArgs,
+ char **papszEnv, struct child *pMkChild, pid_t *pPid);
+int MkWinChildCreateAppend(const char *pszFilename, char **ppszAppend, size_t cbAppend, int fTruncate,
+ struct child *pMkChild, pid_t *pPid);
+
+int MkWinChildCreateSubmit(intptr_t hEvent, void *pvSubmitWorker, PWINCCWPIPE pStdOut, PWINCCWPIPE pStdErr,
+ struct child *pMkChild, pid_t *pPid);
+PWINCCWPIPE MkWinChildcareCreateWorkerPipe(unsigned iWhich, unsigned int idxWorker);
+void MkWinChildcareWorkerDrainPipes(struct WINCHILD *pChild, PWINCCWPIPE pStdOut, PWINCCWPIPE pStdErr);
+void MkWinChildcareDeleteWorkerPipe(PWINCCWPIPE pPipe);
+
+int MkWinChildCreateRedirect(intptr_t hProcess, pid_t *pPid);
+# ifdef DECLARE_HANDLE
+int MkWinChildBuiltInExecChild(void *pvWorker, const char *pszExecutable, char **papszArgs, BOOL fQuotedArgv,
+ char **papszEnvVars, const char *pszCwd, BOOL pafReplace[3], HANDLE pahReplace[3]);
+# endif
+#endif /* KMK */
+int MkWinChildKill(pid_t pid, int iSignal, struct child *pMkChild);
+int MkWinChildWait(int fBlock, pid_t *pPid, int *piExitCode, int *piSignal, int *pfCoreDumped, struct child **ppMkChild);
+void MkWinChildExclusiveAcquire(void);
+void MkWinChildExclusiveRelease(void);
+
+#undef CLOSE_ON_EXEC
+#define CLOSE_ON_EXEC(a_fd) MkWinChildUnrelatedCloseOnExec(a_fd)
+int MkWinChildUnrelatedCloseOnExec(int fd);
+
+
+#endif
+