2 #include <msvcrt/process.h>
3 #include <msvcrt/stdlib.h>
4 #include <msvcrt/string.h>
5 #include <msvcrt/errno.h>
6 #include <msvcrt/internal/file.h>
9 #include <msvcrt/msvcrtdbg.h>
22 const char* find_exec(const char* path, char* rpath)
28 DPRINT("find_exec('%s', %x)\n", path, rpath);
34 if (strlen(path) > FILENAME_MAX - 1)
38 /* copy path in rpath */
39 for (rd = path, rp = rpath; *rd; *rp++ = *rd++);
41 /* try first with the name as is */
42 for (i = 0; i < sizeof(ext) / sizeof(*ext); i++)
46 DPRINT("trying '%s'\n", rpath);
48 if (_access(rpath, F_OK) == 0 && _access(rpath, D_OK) != 0)
56 char* env = getenv("PATH");
64 for (; *ep && (*ep != ';'); *rp++ = *ep++);
68 if (*rp != '/' && *rp != '\\')
74 for (rd=path; *rd; *rp++ = *rd++);
75 for (i = 0; i < sizeof(ext) / sizeof(*ext); i++)
79 DPRINT("trying '%s'\n", rpath);
81 if (_access(rpath, F_OK) == 0 && _access(rpath, D_OK) != 0)
92 return found ? rpath : path;
96 argvtos(char* const* argv, char delim)
104 for (i = 0, len = 0; argv[i]; i++)
106 len += strlen(argv[i]) + 1;
109 str = ptr = (char*) malloc(len + 1);
113 for(i = 0; argv[i]; i++)
115 len = strlen(argv[i]);
116 memcpy(ptr, argv[i], len);
126 valisttos(const char* arg0, va_list alist, char delim)
128 va_list alist2 = alist;
139 len += strlen(ptr) + 1;
140 ptr = va_arg(alist, char*);
144 str = (char*) malloc(len + 1);
152 memcpy(ptr, arg0, len);
155 arg0 = va_arg(alist2, char*);
164 do_spawn(int mode, const char* cmdname, const char* args, const char* envp)
166 STARTUPINFO StartupInfo;
167 PROCESS_INFORMATION ProcessInformation;
175 DPRINT("do_spawn('%s')\n", cmdname);
177 if (mode != _P_NOWAIT && mode != _P_NOWAITO && mode != _P_WAIT && mode != _P_DETACH && mode != _P_OVERLAY)
183 if (0 != _access(cmdname, F_OK))
188 if (0 == _access(cmdname, D_OK))
194 memset (&StartupInfo, 0, sizeof(STARTUPINFO));
195 StartupInfo.cb = sizeof(STARTUPINFO);
197 for (last = i = 0; i < maxfno; i++)
199 if ((void*)-1 != _get_osfhandle(i))
207 StartupInfo.cbReserved2 = sizeof(ULONG) + last * (sizeof(char) + sizeof(HANDLE));
208 StartupInfo.lpReserved2 = malloc(StartupInfo.cbReserved2);
209 if (StartupInfo.lpReserved2 == NULL)
215 *(DWORD*)StartupInfo.lpReserved2 = last;
216 fmode = (char*)(StartupInfo.lpReserved2 + sizeof(ULONG));
217 hFile = (HANDLE*)(StartupInfo.lpReserved2 + sizeof(ULONG) + last * sizeof(char));
218 for (i = 0; i < last; i++)
220 int _mode = __fileno_getmode(i);
221 HANDLE h = _get_osfhandle(i);
222 /* FIXME: The test of console handles (((ULONG)Handle) & 0x10000003) == 0x3)
225 if ((((ULONG)h) & 0x10000003) == 0x3 || _mode & _O_NOINHERIT || (i < 3 && mode == _P_DETACH))
227 *hFile = INVALID_HANDLE_VALUE;
234 bFlag = GetHandleInformation(h, &dwFlags);
235 if (bFlag && (dwFlags & HANDLE_FLAG_INHERIT))
238 *fmode = (_O_ACCMODE & _mode) | (((_O_TEXT | _O_BINARY) & _mode) >> 8);
242 *hFile = INVALID_HANDLE_VALUE;
251 bResult = CreateProcessA((char *)cmdname,
256 mode == _P_DETACH ? DETACHED_PROCESS : 0,
260 &ProcessInformation);
261 if (StartupInfo.lpReserved2)
263 free(StartupInfo.lpReserved2);
268 dwError = GetLastError();
269 DPRINT("%x\n", dwError);
270 __set_errno(dwError);
273 CloseHandle(ProcessInformation.hThread);
279 WaitForSingleObject(ProcessInformation.hProcess, INFINITE);
280 GetExitCodeProcess(ProcessInformation.hProcess, &dwExitCode);
281 CloseHandle(ProcessInformation.hProcess);
282 return (int)dwExitCode;
284 CloseHandle(ProcessInformation.hProcess);
287 return (int)ProcessInformation.hProcess;
290 int _spawnl(int mode, const char *cmdname, const char* arg0, ...)
296 DPRINT("_spawnl('%s')\n", cmdname);
298 va_start(argp, arg0);
299 args = valisttos(arg0, argp, ' ');
303 ret = do_spawn(mode, cmdname, args, NULL);
309 int _spawnv(int mode, const char *cmdname, char* const* argv)
314 DPRINT("_spawnv('%s')\n", cmdname);
316 args = argvtos(argv, ' ');
320 ret = do_spawn(mode, cmdname, args, NULL);
326 int _spawnle(int mode, const char *cmdname, const char* arg0, ... /*, NULL, const char* const* envp*/)
334 DPRINT("_spawnle('%s')\n", cmdname);
336 va_start(argp, arg0);
337 args = valisttos(arg0, argp, ' ');
340 ptr = (char* const*)va_arg(argp, char*);
343 envs = argvtos(ptr, 0);
346 ret = do_spawn(mode, cmdname, args, envs);
357 int _spawnve(int mode, const char *cmdname, char* const* argv, char* const* envp)
363 DPRINT("_spawnve('%s')\n", cmdname);
365 args = argvtos(argv, ' ');
366 envs = argvtos(envp, 0);
370 ret = do_spawn(mode, cmdname, args, envs);
380 int _spawnvp(int mode, const char* cmdname, char* const* argv)
382 char pathname[FILENAME_MAX];
384 DPRINT("_spawnvp('%s')\n", cmdname);
386 return _spawnv(mode, find_exec(cmdname, pathname), argv);
389 int _spawnlp(int mode, const char* cmdname, const char* arg0, .../*, NULL*/)
394 char pathname[FILENAME_MAX];
396 DPRINT("_spawnlp('%s')\n", cmdname);
398 va_start(argp, arg0);
399 args = valisttos(arg0, argp, ' ');
402 ret = do_spawn(mode, find_exec(cmdname, pathname), args, NULL);
409 int _spawnlpe(int mode, const char* cmdname, const char* arg0, .../*, NULL, const char* const* envp*/)
416 char pathname[FILENAME_MAX];
418 DPRINT("_spawnlpe('%s')\n", cmdname);
420 va_start(argp, arg0);
421 args = valisttos(arg0, argp, ' ');
424 ptr = (char* const*)va_arg(argp, char*);
427 envs = argvtos(ptr, 0);
430 ret = do_spawn(mode, find_exec(cmdname, pathname), args, envs);
440 int _spawnvpe(int mode, const char* cmdname, char* const* argv, char* const* envp)
442 char pathname[FILENAME_MAX];
444 DPRINT("_spawnvpe('%s')\n", cmdname);
446 return _spawnve(mode, find_exec(cmdname, pathname), argv, envp);
449 int _execl(const char* cmdname, const char* arg0, ...)
455 DPRINT("_execl('%s')\n", cmdname);
457 va_start(argp, arg0);
458 args = valisttos(arg0, argp, ' ');
462 ret = do_spawn(P_OVERLAY, cmdname, args, NULL);
468 int _execv(const char* cmdname, char* const* argv)
470 DPRINT("_execv('%s')\n", cmdname);
471 return _spawnv(P_OVERLAY, cmdname, argv);
474 int _execle(const char* cmdname, const char* arg0, ... /*, NULL, char* const* envp */)
482 DPRINT("_execle('%s')\n", cmdname);
484 va_start(argp, arg0);
485 args = valisttos(arg0, argp, ' ');
488 ptr = (char* const*)va_arg(argp, char*);
491 envs = argvtos((char**)ptr, 0);
494 ret = do_spawn(P_OVERLAY, cmdname, args, envs);
504 int _execve(const char* cmdname, char* const* argv, char* const* envp)
506 DPRINT("_execve('%s')\n", cmdname);
507 return _spawnve(P_OVERLAY, cmdname, argv, envp);
510 int _execlp(const char* cmdname, const char* arg0, ...)
515 char pathname[FILENAME_MAX];
517 DPRINT("_execlp('%s')\n", cmdname);
519 va_start(argp, arg0);
520 args = valisttos(arg0, argp, ' ');
524 ret = do_spawn(P_OVERLAY, find_exec(cmdname, pathname), args, NULL);
530 int _execvp(const char* cmdname, char* const* argv)
532 DPRINT("_execvp('%s')\n", cmdname);
533 return _spawnvp(P_OVERLAY, cmdname, argv);
536 int _execlpe(const char* cmdname, const char* arg0, ... /*, NULL, char* const* envp */)
543 char pathname[FILENAME_MAX];
545 DPRINT("_execlpe('%s')\n", cmdname);
547 va_start(argp, arg0);
548 args = valisttos(arg0, argp, ' ');
551 ptr = (char* const*)va_arg(argp, char*);
554 envs = argvtos(ptr, 0);
557 ret = do_spawn(P_OVERLAY, find_exec(cmdname, pathname), args, envs);
567 int _execvpe(const char* cmdname, char* const* argv, char* const* envp)
569 DPRINT("_execvpe('%s')\n", cmdname);
570 return _spawnvpe(P_OVERLAY, cmdname, argv, envp);