elk/lib/unix/wait.c

143 lines
3.5 KiB
C

#ifdef __MACH__
# define _POSIX_SOURCE
#endif
#include "unix.h"
#if defined(WAITPID) || defined(WAIT4)
# define WAIT_PROCESS
#endif
#if defined(WAITPID) || defined(WAIT3) || defined(WAIT4)
# define WAIT_OPTIONS
#endif
#if defined(WAIT3) || defined(WAIT4)
# define WAIT_RUSAGE
# include <sys/time.h>
# include <sys/resource.h>
#endif
#ifdef WAIT_OPTIONS
static SYMDESCR Wait_Flags[] = {
{ "nohang", WNOHANG },
{ "untraced", WUNTRACED },
{ 0, 0 }
};
#endif
#ifndef WEXITSTATUS
# define WEXITSTATUS(stat) ((int)((stat >> 8) & 0xFF))
#endif
#ifndef WTERMSIG
# define WTERMSIG(stat) ((int)(stat & 0x7F))
#endif
#ifndef WSTOPSIG
# define WSTOPSIG(stat) ((int)((stat >> 8) & 0xFF))
#endif
#ifndef WIFSIGNALED
# define WIFSIGNALED(stat) ((int)(stat & 0x7F))
#endif
#ifndef WIFSTOPPED
# define WIFSTOPPED(stat) ((int)(stat & 0x7F) == 0x7F)
#endif
static Object General_Wait(ret, ruret, haspid, pid, options)
Object ret, ruret; int haspid, pid, options; {
int retpid, st, code;
char *status;
#ifdef WAIT_RUSAGE
struct rusage ru;
Object sec;
#endif
Object x;
GC_Node3;
x = Null;
Check_Result_Vector(ret, 5);
Check_Result_Vector(ruret, 2);
if (haspid) {
#ifdef WAIT4
retpid = wait4(pid, &st, options, &ru);
#else
#ifdef WAITPID
retpid = waitpid(pid, &st, options);
#endif
#endif
} else {
#ifdef WAIT3
retpid = wait3(&st, options, &ru);
#else
retpid = wait(&st);
#endif
}
if (retpid == -1 && errno != ECHILD)
Raise_System_Error("~E");
GC_Link3(ret, ruret, x);
x = Make_Integer(retpid); VECTOR(ret)->data[0] = x;
if (retpid == 0 || retpid == -1) {
status = "none";
st = code = 0;
#ifdef WAIT_RUSAGE
bzero((char *)&ru, sizeof(ru));
#endif
} else if (WIFSTOPPED(st)) {
status = "stopped"; code = WSTOPSIG(st);
} else if (WIFSIGNALED(st)) {
status = "signaled"; code = WTERMSIG(st);
} else {
status = "exited"; code = WEXITSTATUS(st);
}
x = Intern(status); VECTOR(ret)->data[1] = x;
x = Make_Integer(code); VECTOR(ret)->data[2] = x;
VECTOR(ret)->data[3] = st & 0200 ? True : False;
#ifdef WAIT_RUSAGE
x = Cons(Null, Make_Unsigned_Long((unsigned long)ru.ru_utime.tv_usec
* 1000));
sec = Make_Unsigned_Long((unsigned long)ru.ru_utime.tv_sec);
Car(x) = sec;
VECTOR(ruret)->data[0] = x;
x = Cons(Null, Make_Unsigned_Long((unsigned long)ru.ru_stime.tv_usec
* 1000));
sec = Make_Unsigned_Long((unsigned long)ru.ru_stime.tv_sec);
Car(x) = sec;
VECTOR(ruret)->data[1] = x;
#endif
GC_Unlink;
return Void;
}
static Object P_Wait(argc, argv) int argc; Object *argv; {
int flags = 0;
if (argc == 3)
#ifdef WAIT_OPTIONS
flags = (int)Symbols_To_Bits(argv[2], 1, Wait_Flags);
#else
Primitive_Error("wait options not supported");
#endif
return General_Wait(argv[0], argv[1], 0, 0, flags);
}
#ifdef WAIT_PROCESS
/* If WAIT_PROCESS is supported, then WAIT_OPTIONS is supported as well,
* because both waitpid() and wait4() accept options.
*/
static Object P_Wait_Process(argc, argv) int argc; Object *argv; {
return General_Wait(argv[0], argv[1], 1, Get_Integer(argv[2]),
argc == 4 ? (int)Symbols_To_Bits(argv[3], 1, Wait_Flags) : 0);
}
#endif
elk_init_unix_wait() {
Def_Prim(P_Wait, "unix-wait-vector-fill!", 2, 3, VARARGS);
#ifdef WAIT_PROCESS
Def_Prim(P_Wait_Process, "unix-wait-process-vector-fill!", 3, 4, VARARGS);
P_Provide(Intern("unix:wait-process"));
#endif
#ifdef WAIT_OPTIONS
P_Provide(Intern("unix:wait-options"));
#endif
}