//
// aegis - project change supervisor
// Copyright (C) 2002-2006, 2008, 2012 Peter Miller
//
// This program 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.
//
// This program 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 .
//
#include
#include
#include
#include
#include
#include
#include
int
os_waitpid(int child, int *status_p)
{
struct ret_ty
{
int pid;
int status;
};
static size_t nret;
static size_t nret_max;
static ret_ty *ret;
int pid;
int status;
int result;
//
// see if we already have it
//
trace(("os_waitpid(child = %d)\n{\n", child));
assert(child > 0);
result = 0;
for (size_t j = 0; j < nret; ++j)
{
if (ret[j].pid != child)
continue;
*status_p = ret[j].status;
ret[j] = ret[--nret];
goto done;
}
//
// new one, go hunting
//
for (;;)
{
//
// block until a child terminates,
// or there are no more children
//
pid = wait(&status);
if (pid == -1)
{
if (errno == EINTR)
continue;
result = -1;
break;
}
//
// stop if this is the child
// we are looking for
//
if (pid == child)
{
*status_p = status;
break;
}
//
// remember and keep going
//
if (nret >= nret_max)
{
nret_max = nret_max * 2 + 8;
ret_ty *new_ret = new ret_ty [nret_max];
for (size_t k = 0; k < nret; ++k)
new_ret[k] = ret[k];
delete [] ret;
ret = new_ret;
}
ret[nret].pid = pid;
ret[nret].status = status;
++nret;
}
//
// here for all exits
//
done:
trace(("return %d;\n", result));
trace(("}\n"));
return result;
}
// vim: set ts=8 sw=4 et :