7457b4de3f
[INTERPRETER] * BUG: Fix Task.Wait() so that it restores SIGCHLD handler for GTK+ libraries, like Process.Wait(). * NEW: Task.Wait() now can take an optional timeout argument.
763 lines
13 KiB
C
763 lines
13 KiB
C
/***************************************************************************
|
|
|
|
gbx_c_task.c
|
|
|
|
(c) 2000-2017 Benoît Minisini <g4mba5@gmail.com>
|
|
|
|
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 2, 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, write to the Free Software
|
|
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
|
|
MA 02110-1301, USA.
|
|
|
|
***************************************************************************/
|
|
|
|
#define __GBX_C_TASK_C
|
|
|
|
#include "gbx_info.h"
|
|
|
|
#ifndef GBX_INFO
|
|
|
|
#include <errno.h>
|
|
#include <sys/wait.h>
|
|
#include <sys/ioctl.h>
|
|
#include <sys/time.h>
|
|
|
|
#include "gb_common.h"
|
|
#include "gb_list.h"
|
|
#include "gb_file.h"
|
|
#include "gb_error.h"
|
|
#include "gbx_api.h"
|
|
#include "gbx_exec.h"
|
|
#include "gbx_string.h"
|
|
#include "gbx_signal.h"
|
|
#include "gbx_event.h"
|
|
#include "gbx_component.h"
|
|
#include "gbx_watch.h"
|
|
|
|
#include "gbx_c_file.h"
|
|
#include "gbx_c_task.h"
|
|
|
|
//#define DEBUG_ME 1
|
|
#ifdef __CYGWIN__
|
|
#define FIONREAD TIOCINQ
|
|
#endif
|
|
|
|
DECLARE_EVENT(EVENT_Read);
|
|
DECLARE_EVENT(EVENT_Error);
|
|
DECLARE_EVENT(EVENT_Kill);
|
|
|
|
enum {
|
|
CHILD_OK = 0,
|
|
CHILD_ERROR = 1,
|
|
CHILD_STDOUT = 2,
|
|
CHILD_STDERR = 3,
|
|
CHILD_RETURN = 4
|
|
};
|
|
|
|
static SIGNAL_CALLBACK *_signal_handler = NULL;
|
|
static CTASK *_task_list = NULL;
|
|
static int _task_count = 0;
|
|
|
|
//-------------------------------------------------------------------------
|
|
|
|
static void cleanup_task(CTASK *_object);
|
|
static void stop_task(CTASK *_object);
|
|
static void close_fd(int *pfd);
|
|
|
|
static void has_forked(void)
|
|
{
|
|
CTASK *task;
|
|
pid_t pid = getpid();
|
|
STREAM *stream;
|
|
|
|
FILE_init();
|
|
EXEC_debug = FALSE;
|
|
EXEC_task = TRUE;
|
|
if (EXEC_profile)
|
|
DEBUG.Profile.Cancel();
|
|
EXEC_profile = FALSE;
|
|
EXEC_profile_instr = FALSE;
|
|
|
|
EXEC_Hook.loop = NULL;
|
|
EXEC_Hook.wait = NULL;
|
|
EXEC_Hook.timer = NULL;
|
|
EXEC_Hook.watch = NULL;
|
|
EXEC_Hook.post = NULL;
|
|
//EXEC_Hook.quit = NULL;
|
|
|
|
stream = CSTREAM_TO_STREAM(CFILE_get_standard_stream(CFILE_OUT));
|
|
|
|
stream->common.eol = 0;
|
|
STREAM_blocking(stream, TRUE);
|
|
|
|
SIGNAL_has_forked();
|
|
|
|
task = _task_list;
|
|
while (task)
|
|
{
|
|
if (task->pid != pid)
|
|
{
|
|
close_fd(&task->fd_out);
|
|
close_fd(&task->fd_err);
|
|
}
|
|
|
|
task = task->list.next;
|
|
}
|
|
}
|
|
|
|
static void callback_child(int signum, intptr_t data)
|
|
{
|
|
CTASK *_object, *next;
|
|
int status;
|
|
|
|
#if DEBUG_ME
|
|
fprintf(stderr, ">> callback_child\n");
|
|
#endif
|
|
|
|
_object = _task_list;
|
|
while (_object)
|
|
{
|
|
next = THIS->list.next;
|
|
if (wait4(THIS->pid, &status, WNOHANG, NULL) == THIS->pid)
|
|
{
|
|
THIS->status = status;
|
|
stop_task(THIS);
|
|
}
|
|
_object = next;
|
|
}
|
|
|
|
#if DEBUG_ME
|
|
fprintf(stderr, "<< callback_child\n");
|
|
#endif
|
|
}
|
|
|
|
static int get_readable(int fd)
|
|
{
|
|
int len;
|
|
|
|
if (ioctl(fd, FIONREAD, &len) < 0 || len <= 0)
|
|
return 0;
|
|
else
|
|
return len;
|
|
}
|
|
|
|
static bool callback_write(int fd, int type, CTASK *_object)
|
|
{
|
|
int len;
|
|
char *data;
|
|
char *p;
|
|
int n;
|
|
|
|
//fprintf(stderr, "callback_write: %d %p\n", fd, THIS);
|
|
|
|
len = get_readable(fd);
|
|
if (len == 0)
|
|
return TRUE;
|
|
|
|
data = STRING_new(NULL, len);
|
|
p = data;
|
|
|
|
while (len > 0)
|
|
{
|
|
n = read(fd, p, len);
|
|
if (n < 0)
|
|
{
|
|
if (errno == EINTR)
|
|
continue;
|
|
else
|
|
break;
|
|
}
|
|
len -= n;
|
|
p += n;
|
|
}
|
|
|
|
GB_Raise(THIS, EVENT_Read, 1, GB_T_STRING, data, STRING_length(data) - len);
|
|
|
|
STRING_free(&data);
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
static int callback_error(int fd, int type, CTASK *_object)
|
|
{
|
|
char buffer[256];
|
|
int n;
|
|
|
|
//fprintf(stderr, "callback_error: %d %p\n", fd, THIS);
|
|
|
|
n = read(fd, buffer, sizeof(buffer));
|
|
|
|
if (n <= 0)
|
|
return TRUE;
|
|
|
|
GB_Raise(THIS, EVENT_Error, 1, GB_T_STRING, buffer, n);
|
|
return FALSE;
|
|
}
|
|
|
|
static bool create_return_directory(void)
|
|
{
|
|
static bool mkdir_done = FALSE;
|
|
|
|
char buf[PATH_MAX];
|
|
|
|
if (mkdir_done)
|
|
return FALSE;
|
|
|
|
sprintf(buf, RETURN_DIR_PATTERN, getuid(), getpid());
|
|
|
|
if (mkdir(buf, S_IRWXU) != 0)
|
|
{
|
|
GB_Error("Cannot create task return directory");
|
|
return TRUE;
|
|
}
|
|
|
|
mkdir_done = TRUE;
|
|
return FALSE;
|
|
}
|
|
|
|
static void init_task(void)
|
|
{
|
|
_task_count++;
|
|
|
|
if (_task_count <= 1)
|
|
_signal_handler = SIGNAL_register(SIGCHLD, callback_child, 0);
|
|
|
|
SIGNAL_check(SIGCHLD);
|
|
}
|
|
|
|
static void exit_task(void)
|
|
{
|
|
_task_count--;
|
|
|
|
if (_task_count > 0)
|
|
return;
|
|
|
|
SIGNAL_unregister(SIGCHLD, _signal_handler);
|
|
_signal_handler = NULL;
|
|
}
|
|
|
|
static void prepare_task(CTASK *_object)
|
|
{
|
|
THIS->fd_out = -1;
|
|
THIS->fd_err = -1;
|
|
}
|
|
|
|
static void exit_child(int ret)
|
|
{
|
|
FILE_exit();
|
|
_exit(ret);
|
|
}
|
|
|
|
static bool start_task(CTASK *_object)
|
|
{
|
|
const char *err = NULL;
|
|
pid_t pid;
|
|
sigset_t sig, old;
|
|
GB_FUNCTION func;
|
|
int fd_out[2], fd_err[2];
|
|
bool has_read, has_error;
|
|
GB_VALUE *ret;
|
|
char buf[PATH_MAX];
|
|
FILE *f;
|
|
|
|
if (EXEC_task)
|
|
return TRUE;
|
|
|
|
if (THIS->stopped)
|
|
{
|
|
cleanup_task(THIS);
|
|
return TRUE;
|
|
}
|
|
|
|
init_task();
|
|
|
|
LIST_insert(&_task_list, THIS, &THIS->list);
|
|
|
|
// Create pipes
|
|
|
|
has_read = GB_CanRaise(THIS, EVENT_Read);
|
|
has_error = GB_CanRaise(THIS, EVENT_Error);
|
|
|
|
if (has_read && pipe(fd_out) != 0)
|
|
goto __ERROR;
|
|
|
|
if (has_error && pipe(fd_err) != 0)
|
|
goto __ERROR;
|
|
|
|
COMPONENT_before_fork();
|
|
|
|
// Block SIGCHLD
|
|
|
|
sigemptyset(&sig);
|
|
|
|
sigaddset(&sig, SIGCHLD);
|
|
sigprocmask(SIG_BLOCK, &sig, &old);
|
|
|
|
pid = fork();
|
|
if (pid == (-1))
|
|
{
|
|
stop_task(THIS);
|
|
sigprocmask(SIG_SETMASK, &old, NULL);
|
|
goto __ERROR;
|
|
}
|
|
|
|
if (pid)
|
|
{
|
|
#if DEBUG_ME
|
|
fprintf(stderr, "start_task: %p %d\n", THIS, pid);
|
|
#endif
|
|
THIS->pid = pid;
|
|
|
|
if (has_read)
|
|
{
|
|
close(fd_out[1]);
|
|
THIS->fd_out = fd_out[0];
|
|
|
|
GB_Watch(THIS->fd_out, GB_WATCH_READ, (void *)callback_write, (intptr_t)THIS);
|
|
}
|
|
|
|
if (has_error)
|
|
{
|
|
close(fd_err[1]);
|
|
THIS->fd_err = fd_err[0];
|
|
|
|
fcntl(THIS->fd_err, F_SETFL, fcntl(THIS->fd_err, F_GETFL) | O_NONBLOCK);
|
|
GB_Watch(THIS->fd_err, GB_WATCH_READ, (void *)callback_error, (intptr_t)THIS);
|
|
}
|
|
|
|
sigprocmask(SIG_SETMASK, &old, NULL);
|
|
}
|
|
else // child task
|
|
{
|
|
THIS->child = TRUE;
|
|
THIS->pid = getpid();
|
|
|
|
sigprocmask(SIG_SETMASK, &old, NULL);
|
|
|
|
if (has_read)
|
|
{
|
|
close(fd_out[0]);
|
|
|
|
if (dup2(fd_out[1], STDOUT_FILENO) == -1)
|
|
exit_child(CHILD_STDOUT);
|
|
|
|
setlinebuf(stdout);
|
|
}
|
|
/*else
|
|
close(CHILD_STDOUT);*/
|
|
|
|
if (has_error)
|
|
{
|
|
close(fd_err[0]);
|
|
|
|
if (dup2(fd_err[1], STDERR_FILENO) == -1)
|
|
exit_child(CHILD_STDERR);
|
|
|
|
setlinebuf(stderr);
|
|
}
|
|
/*else
|
|
close(CHILD_STDERR);*/
|
|
|
|
has_forked(); // After the redirection
|
|
|
|
GB_GetFunction(&func, THIS, "Main", "", NULL);
|
|
|
|
TRY
|
|
{
|
|
ret = GB_Call(&func, 0, FALSE);
|
|
if (ret->type != GB_T_VOID)
|
|
{
|
|
sprintf(buf, RETURN_FILE_PATTERN, getuid(), getppid(), getpid());
|
|
#if DEBUG_ME
|
|
fprintf(stderr, "serialize to: %s\n", buf);
|
|
#endif
|
|
GB_ReturnConvVariant();
|
|
if (GB_Serialize(buf, ret))
|
|
{
|
|
#if DEBUG_ME
|
|
fprintf(stderr, "gb.task: serialization has failed\n");
|
|
#endif
|
|
exit_child(CHILD_RETURN);
|
|
}
|
|
}
|
|
}
|
|
CATCH
|
|
{
|
|
if (ERROR_current->info.code && ERROR_current->info.code != E_ABORT)
|
|
{
|
|
sprintf(buf, RETURN_FILE_PATTERN, getuid(), getppid(), getpid());
|
|
f = fopen(buf, "w+");
|
|
if (f)
|
|
{
|
|
ERROR_print_at(f, FALSE, FALSE);
|
|
fclose(f);
|
|
}
|
|
|
|
exit_child(CHILD_ERROR);
|
|
}
|
|
}
|
|
END_TRY
|
|
|
|
exit_child(CHILD_OK);
|
|
}
|
|
|
|
return FALSE;
|
|
|
|
__ERROR:
|
|
|
|
// TODO: as the routine is posted, nobody will see the error!
|
|
if (!err)
|
|
err = strerror(errno);
|
|
fprintf(stderr, "gb.task: cannot run task: %s\n", err);
|
|
GB_Error("Cannot run task: &1", err);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static void close_fd(int *pfd)
|
|
{
|
|
int fd = *pfd;
|
|
|
|
if (fd >= 0)
|
|
{
|
|
GB_Watch(fd, GB_WATCH_NONE, NULL, 0);
|
|
close(fd);
|
|
*pfd = -1;
|
|
}
|
|
}
|
|
|
|
static bool get_return_value(CTASK *_object, bool cleanup)
|
|
{
|
|
char path[PATH_MAX];
|
|
GB_VALUE value;
|
|
bool fail = FALSE;
|
|
struct stat info;
|
|
char *err = NULL;
|
|
int fd;
|
|
int n;
|
|
|
|
sprintf(path, RETURN_FILE_PATTERN, getuid(), getpid(), THIS->pid);
|
|
|
|
if (!cleanup)
|
|
{
|
|
switch (THIS->status)
|
|
{
|
|
case CHILD_OK:
|
|
|
|
#if DEBUG_ME
|
|
fprintf(stderr,"unserialize from: %s\n", path);
|
|
#endif
|
|
if (!THIS->got_value)
|
|
{
|
|
if (stat(path, &info))
|
|
{
|
|
GB_Error((char *)E_NRETURN);
|
|
return TRUE;
|
|
}
|
|
|
|
fail = GB_UnSerialize(path, &value);
|
|
if (!fail)
|
|
GB_StoreVariant(&value._variant, &THIS->ret);
|
|
THIS->got_value = TRUE;
|
|
}
|
|
break;
|
|
|
|
case CHILD_ERROR:
|
|
|
|
if (stat(path, &info))
|
|
{
|
|
fail = TRUE;
|
|
}
|
|
else
|
|
{
|
|
err = STRING_new_temp(NULL, info.st_size);
|
|
|
|
fd = open(path, O_RDONLY);
|
|
if (fd < 0)
|
|
{
|
|
fail = TRUE;
|
|
break;
|
|
}
|
|
else
|
|
{
|
|
for(;;)
|
|
{
|
|
n = read(fd, err, info.st_size);
|
|
if (n == info.st_size || errno != EINTR)
|
|
break;
|
|
}
|
|
close(fd);
|
|
|
|
if (n == info.st_size)
|
|
GB_Error("Task has failed: &1", err);
|
|
else
|
|
fail = TRUE;
|
|
}
|
|
}
|
|
|
|
if (fail)
|
|
GB_Error("Unable to get task error");
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
unlink(path);
|
|
|
|
return fail;
|
|
}
|
|
|
|
static void cleanup_task(CTASK *_object)
|
|
{
|
|
//printf("cleanup task %p\n", THIS); fflush(stdout);
|
|
|
|
OBJECT_UNREF(_object);
|
|
}
|
|
|
|
static void stop_task(CTASK *_object)
|
|
{
|
|
int len;
|
|
GB_RAISE_HANDLER handler;
|
|
|
|
#if DEBUG_ME
|
|
fprintf(stderr, "stop_task: %p %d\n", THIS, THIS->pid);
|
|
#endif
|
|
|
|
THIS->stopped = TRUE;
|
|
|
|
// Remove task temporary files
|
|
FILE_remove_temp_file_pid(THIS->pid);
|
|
|
|
// Flush standard error
|
|
if (THIS->fd_err >= 0)
|
|
while (callback_error(THIS->fd_err, 0, THIS) == 0);
|
|
|
|
// Flush standard output
|
|
if (THIS->fd_out >= 0)
|
|
{
|
|
for(;;)
|
|
{
|
|
len = get_readable(THIS->fd_out);
|
|
if (len <= 0)
|
|
break;
|
|
|
|
if (callback_write(THIS->fd_out, 0, THIS))
|
|
break;
|
|
}
|
|
}
|
|
|
|
close_fd(&THIS->fd_out);
|
|
close_fd(&THIS->fd_err);
|
|
|
|
LIST_remove(&_task_list, THIS, &THIS->list);
|
|
exit_task();
|
|
|
|
//printf("Kill event...\n"); fflush(stdout);
|
|
|
|
if (GB_CanRaise(THIS, EVENT_Kill))
|
|
{
|
|
handler.callback = (GB_CALLBACK)cleanup_task;
|
|
handler.data = (intptr_t)THIS;
|
|
|
|
GB_RaiseBegin(&handler);
|
|
GB_Raise(THIS, EVENT_Kill, 0);
|
|
GB_RaiseEnd(&handler);
|
|
}
|
|
|
|
cleanup_task(THIS);
|
|
}
|
|
|
|
//-------------------------------------------------------------------------
|
|
|
|
BEGIN_METHOD_VOID(Task_new)
|
|
|
|
GB_FUNCTION func;
|
|
|
|
THIS->ret.type = GB_T_NULL;
|
|
|
|
if (EXEC_task)
|
|
{
|
|
GB_Error("A task cannot create other tasks");
|
|
return;
|
|
}
|
|
|
|
if (create_return_directory())
|
|
return;
|
|
|
|
if (GB_GetFunction(&func, THIS, "Main", "", NULL))
|
|
return;
|
|
|
|
if (_task_count > MAX_TASK)
|
|
{
|
|
GB_Error("Too many tasks");
|
|
return;
|
|
}
|
|
|
|
prepare_task(THIS);
|
|
|
|
OBJECT_REF(THIS);
|
|
EVENT_post((GB_CALLBACK)start_task, (intptr_t)THIS);
|
|
|
|
END_METHOD
|
|
|
|
BEGIN_METHOD_VOID(Task_free)
|
|
|
|
get_return_value(THIS, TRUE);
|
|
GB_StoreVariant(NULL, &THIS->ret);
|
|
|
|
END_METHOD
|
|
|
|
BEGIN_PROPERTY(Task_Handle)
|
|
|
|
GB_ReturnInteger(THIS->pid);
|
|
|
|
END_PROPERTY
|
|
|
|
|
|
BEGIN_METHOD_VOID(Task_Stop)
|
|
|
|
if (THIS->pid > 0)
|
|
kill(THIS->pid, SIGKILL);
|
|
else
|
|
THIS->stopped = TRUE;
|
|
|
|
END_METHOD
|
|
|
|
static void error_Task_Wait(CTASK *task)
|
|
{
|
|
OBJECT_UNREF(task);
|
|
}
|
|
|
|
BEGIN_METHOD(Task_Wait, GB_FLOAT timeout)
|
|
|
|
int ret;
|
|
int sigfd;
|
|
int timeout;
|
|
|
|
timeout = (int)(VARGOPT(timeout, 0.0) * 1000);
|
|
|
|
OBJECT_REF(THIS);
|
|
|
|
//fprintf(stderr, "Task_Wait: %p\n", THIS);
|
|
|
|
ON_ERROR_1(error_Task_Wait, THIS)
|
|
{
|
|
GB_Wait(0);
|
|
|
|
while (!THIS->stopped)
|
|
{
|
|
sigfd = SIGNAL_get_fd();
|
|
SIGNAL_check(SIGCHLD);
|
|
ret = WATCH_process(sigfd, THIS->fd_out, THIS->fd_err, timeout);
|
|
|
|
if (ret & WP_OUTPUT)
|
|
callback_write(THIS->fd_out, GB_WATCH_READ, THIS);
|
|
|
|
if (ret & WP_ERROR)
|
|
callback_error(THIS->fd_err, GB_WATCH_READ, THIS);
|
|
|
|
if (ret & WP_END)
|
|
SIGNAL_raise_callbacks(sigfd, GB_WATCH_READ, 0);
|
|
|
|
if (ret & WP_TIMEOUT)
|
|
break;
|
|
|
|
if (ret == 0)
|
|
usleep(1000);
|
|
}
|
|
}
|
|
END_ERROR
|
|
|
|
OBJECT_UNREF(_object);
|
|
|
|
END_METHOD
|
|
|
|
BEGIN_PROPERTY(Task_Value)
|
|
|
|
if (!THIS->child && THIS->stopped)
|
|
{
|
|
if (WIFEXITED(THIS->status))
|
|
{
|
|
switch (WEXITSTATUS(THIS->status))
|
|
{
|
|
case CHILD_OK:
|
|
|
|
if (get_return_value(THIS, FALSE))
|
|
break;
|
|
GB_ReturnVariant(&THIS->ret);
|
|
return;
|
|
|
|
case CHILD_STDOUT:
|
|
|
|
GB_Error("Unable to redirect task standard output");
|
|
return;
|
|
|
|
case CHILD_STDERR:
|
|
|
|
GB_Error("Unable to redirect task error output");
|
|
return;
|
|
|
|
case CHILD_RETURN:
|
|
|
|
GB_Error("Unable to serialize task return value");
|
|
return;
|
|
|
|
case CHILD_ERROR:
|
|
|
|
get_return_value(THIS, FALSE);
|
|
break;
|
|
}
|
|
}
|
|
else if (WIFSIGNALED(THIS->status))
|
|
{
|
|
GB_Error("Task has aborted: &1", strsignal(WTERMSIG(THIS->status)));
|
|
return;
|
|
}
|
|
}
|
|
|
|
GB_ReturnNull();
|
|
GB_ReturnConvVariant();
|
|
|
|
END_PROPERTY
|
|
|
|
BEGIN_PROPERTY(Task_Running)
|
|
|
|
GB_ReturnBoolean(!THIS->stopped);
|
|
|
|
END_PROPERTY
|
|
|
|
//-------------------------------------------------------------------------
|
|
|
|
#endif
|
|
|
|
GB_DESC TaskDesc[] =
|
|
{
|
|
GB_DECLARE("Task", sizeof(CTASK)), GB_NOT_CREATABLE(),
|
|
|
|
GB_METHOD("_new", NULL, Task_new, NULL),
|
|
GB_METHOD("_free", NULL, Task_free, NULL),
|
|
|
|
GB_PROPERTY_READ("Handle", "i", Task_Handle),
|
|
GB_PROPERTY_READ("Value", "v", Task_Value),
|
|
GB_PROPERTY_READ("Running", "b", Task_Running),
|
|
|
|
GB_METHOD("Stop", NULL, Task_Stop, NULL),
|
|
GB_METHOD("Kill", NULL, Task_Stop, NULL),
|
|
GB_METHOD("Wait", NULL, Task_Wait, "[(Timeout)f]"),
|
|
|
|
GB_EVENT("Read", NULL, "(Data)s", &EVENT_Read),
|
|
GB_EVENT("Error", NULL, "(Data)s", &EVENT_Error),
|
|
GB_EVENT("Kill", NULL, NULL, &EVENT_Kill),
|
|
|
|
GB_END_DECLARE
|
|
};
|