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);
154 arg0 = va_arg(alist2, char*);
163 do_spawn(int mode, const char* cmdname, const char* args, const char* envp)
165 STARTUPINFO StartupInfo;
166 PROCESS_INFORMATION ProcessInformation;
174 DPRINT("do_spawn('%s')\n", cmdname);
176 if (mode != _P_NOWAIT && mode != _P_NOWAITO && mode != _P_WAIT && mode != _P_DETACH && mode != _P_OVERLAY)
182 if (0 != _access(cmdname, F_OK))
187 if (0 == _access(cmdname, D_OK))
193 memset (&StartupInfo, 0, sizeof(STARTUPINFO));
194 StartupInfo.cb = sizeof(STARTUPINFO);
196 for (last = i = 0; i < maxfno; i++)
198 if ((void*)-1 != _get_osfhandle(i))
206 StartupInfo.cbReserved2 = sizeof(ULONG) + last * (sizeof(char) + sizeof(HANDLE));
207 StartupInfo.lpReserved2 = malloc(StartupInfo.cbReserved2);
208 if (StartupInfo.lpReserved2 == NULL)
214 *(DWORD*)StartupInfo.lpReserved2 = last;
215 fmode = (char*)(StartupInfo.lpReserved2 + sizeof(ULONG));
216 hFile = (HANDLE*)(StartupInfo.lpReserved2 + sizeof(ULONG) + last * sizeof(char));
217 for (i = 0; i < last; i++)
219 int _mode = __fileno_getmode(i);
220 HANDLE h = _get_osfhandle(i);
221 /* FIXME: The test of console handles (((ULONG)Handle) & 0x10000003) == 0x3)
224 if ((((ULONG)h) & 0x10000003) == 0x3 || _mode & _O_NOINHERIT || (i < 3 && mode == _P_DETACH))
226 *hFile = INVALID_HANDLE_VALUE;
233 bFlag = GetHandleInformation(h, &dwFlags);
234 if (bFlag && (dwFlags & HANDLE_FLAG_INHERIT))
237 *fmode = (_O_ACCMODE & _mode) | (((_O_TEXT | _O_BINARY) & _mode) >> 8);
241 *hFile = INVALID_HANDLE_VALUE;
250 bResult = CreateProcessA((char *)cmdname,
255 mode == _P_DETACH ? DETACHED_PROCESS : 0,
259 &ProcessInformation);
260 if (StartupInfo.lpReserved2)
262 free(StartupInfo.lpReserved2);
267 dwError = GetLastError();
268 DPRINT("%x\n", dwError);
269 __set_errno(dwError);
272 CloseHandle(ProcessInformation.hThread);
278 WaitForSingleObject(ProcessInformation.hProcess, INFINITE);
279 GetExitCodeProcess(ProcessInformation.hProcess, &dwExitCode);
280 CloseHandle(ProcessInformation.hProcess);
281 return (int)dwExitCode;
283 CloseHandle(ProcessInformation.hProcess);
286 return (int)ProcessInformation.hProcess;
289 int _spawnl(int mode, const char *cmdname, const char* arg0, ...)
295 DPRINT("_spawnl('%s')\n", cmdname);
297 va_start(argp, arg0);
298 args = valisttos(arg0, argp, ' ');
302 ret = do_spawn(mode, cmdname, args, NULL);
308 int _spawnv(int mode, const char *cmdname, char* const* argv)
313 DPRINT("_spawnv('%s')\n", cmdname);
315 args = argvtos(argv, ' ');
319 ret = do_spawn(mode, cmdname, args, NULL);
325 int _spawnle(int mode, const char *cmdname, const char* arg0, ... /*, NULL, const char* const* envp*/)
333 DPRINT("_spawnle('%s')\n", cmdname);
335 va_start(argp, arg0);
336 args = valisttos(arg0, argp, ' ');
339 ptr = (char* const*)va_arg(argp, char*);
342 envs = argvtos(ptr, 0);
345 ret = do_spawn(mode, cmdname, args, envs);
356 int _spawnve(int mode, const char *cmdname, char* const* argv, char* const* envp)
362 DPRINT("_spawnve('%s')\n", cmdname);
364 args = argvtos(argv, ' ');
365 envs = argvtos(envp, 0);
369 ret = do_spawn(mode, cmdname, args, envs);
379 int _spawnvp(int mode, const char* cmdname, char* const* argv)
381 char pathname[FILENAME_MAX];
383 DPRINT("_spawnvp('%s')\n", cmdname);
385 return _spawnv(mode, find_exec(cmdname, pathname), argv);
388 int _spawnlpe(int mode, const char* cmdname, const char* arg0, .../*, NULL, const char* const* envp*/)
395 char pathname[FILENAME_MAX];
397 DPRINT("_spawnlpe('%s')\n", cmdname);
399 va_start(argp, arg0);
400 args = valisttos(arg0, argp, ' ');
403 ptr = (char* const*)va_arg(argp, char*);
406 envs = argvtos(ptr, 0);
409 ret = do_spawn(mode, find_exec(cmdname, pathname), args, envs);
419 int _spawnvpe(int mode, const char* cmdname, char* const* argv, char* const* envp)
421 char pathname[FILENAME_MAX];
423 DPRINT("_spawnvpe('%s')\n", cmdname);
425 return _spawnve(mode, find_exec(cmdname, pathname), argv, envp);
428 int _execl(const char* cmdname, const char* arg0, ...)
434 DPRINT("_execl('%s')\n", cmdname);
436 va_start(argp, arg0);
437 args = valisttos(arg0, argp, ' ');
441 ret = do_spawn(P_OVERLAY, cmdname, args, NULL);
447 int _execv(const char* cmdname, char* const* argv)
449 DPRINT("_execv('%s')\n", cmdname);
450 return _spawnv(P_OVERLAY, cmdname, argv);
453 int _execle(const char* cmdname, const char* arg0, ... /*, NULL, char* const* envp */)
461 DPRINT("_execle('%s')\n", cmdname);
463 va_start(argp, arg0);
464 args = valisttos(arg0, argp, ' ');
467 ptr = (char* const*)va_arg(argp, char*);
470 envs = argvtos((char**)ptr, 0);
473 ret = do_spawn(P_OVERLAY, cmdname, args, envs);
483 int _execve(const char* cmdname, char* const* argv, char* const* envp)
485 DPRINT("_execve('%s')\n", cmdname);
486 return _spawnve(P_OVERLAY, cmdname, argv, envp);
489 int _execlp(const char* cmdname, const char* arg0, ...)
494 char pathname[FILENAME_MAX];
496 DPRINT("_execlp('%s')\n", cmdname);
498 va_start(argp, arg0);
499 args = valisttos(arg0, argp, ' ');
503 ret = do_spawn(P_OVERLAY, find_exec(cmdname, pathname), args, NULL);
509 int _execvp(const char* cmdname, char* const* argv)
511 DPRINT("_execvp('%s')\n", cmdname);
512 return _spawnvp(P_OVERLAY, cmdname, argv);
515 int _execlpe(const char* cmdname, const char* arg0, ... /*, NULL, char* const* envp */)
522 char pathname[FILENAME_MAX];
524 DPRINT("_execlpe('%s')\n", cmdname);
526 va_start(argp, arg0);
527 args = valisttos(arg0, argp, ' ');
530 ptr = (char* const*)va_arg(argp, char*);
533 envs = argvtos(ptr, 0);
536 ret = do_spawn(P_OVERLAY, find_exec(cmdname, pathname), args, envs);
546 int _execvpe(const char* cmdname, char* const* argv, char* const* envp)
548 DPRINT("_execvpe('%s')\n", cmdname);
549 return _spawnvpe(P_OVERLAY, cmdname, argv, envp);