From fa6f44ea7205416837bc72bb9948ba376c2af7c5 Mon Sep 17 00:00:00 2001 From: Nguyễn Thái Ngọc Duy <pclouds@gmail.com> Date: Wed, 14 Apr 2010 00:44:48 +0200 Subject: win32: ash: reimplement waitpid(-1) --- shell/ash.c | 53 +++++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 53 insertions(+) diff --git a/shell/ash.c b/shell/ash.c index 1b448371d..31ad46aa7 100644 --- a/shell/ash.c +++ b/shell/ash.c @@ -3889,6 +3889,53 @@ sprint_status(char *s, int status, int sigonly) return col; } +#if ENABLE_PLATFORM_MINGW32 +/* + * Windows does not know about parent-child relationship + * They don't support waitpid(-1) + */ +static pid_t +waitpid_child(int *status) +{ + HANDLE *pidlist, *pidp; + int pid_nr = 0; + pid_t pid; + DWORD win_status, idx; + struct job *jb; + + #define LOOP(stmt) \ + for (jb = curjob; jb; jb = jb->prev_job) { \ + struct procstat *ps, *psend; \ + if (jb->state == JOBDONE) \ + continue; \ + ps = jb->ps; \ + psend = ps + jb->nprocs; \ + while (ps < psend) { \ + if (ps->ps_pid != -1) { \ + stmt; \ + } \ + ps++; \ + } \ + } + + LOOP(pid_nr++); + pidp = pidlist = ckmalloc(sizeof(*pidlist)*pid_nr); + LOOP(*pidp++ = (HANDLE)ps->ps_pid); + #undef LOOP + + idx = WaitForMultipleObjects(pid_nr, pidlist, FALSE, INFINITE); + if (idx >= pid_nr) { + free(pidlist); + return -1; + } + GetExitCodeProcess(pidlist[idx], &win_status); + pid = (int)pidlist[idx]; + free(pidlist); + *status = (int)win_status; + return pid; +} +#endif + static int dowait(int wait_flags, struct job *job) { @@ -3905,7 +3952,11 @@ dowait(int wait_flags, struct job *job) * NB: _not_ safe_waitpid, we need to detect EINTR */ if (doing_jobctl) wait_flags |= WUNTRACED; +#if ENABLE_PLATFORM_MINGW32 + pid = waitpid_child(&status); +#else pid = waitpid(-1, &status, wait_flags); +#endif TRACE(("wait returns pid=%d, status=0x%x, errno=%d(%s)\n", pid, status, errno, strerror(errno))); if (pid <= 0) @@ -3928,6 +3979,8 @@ dowait(int wait_flags, struct job *job) jobno(jp), pid, ps->ps_status, status)); ps->ps_status = status; thisjob = jp; + if (ENABLE_PLATFORM_MINGW32) + ps->ps_pid = -1; } if (ps->ps_status == -1) state = JOBRUNNING; -- cgit v1.2.3-55-g6feb