gambas-source-code/main/gbx/gbx_watch.c
Benoît Minisini 54f62f479e 'WAIT' with no argument now behaves as before when no component specific WAIT handler is defined.
[INTERPRETER]
* BUG: 'WAIT' with no argument now behaves as before when no component specific WAIT handler is defined.
2023-02-02 10:05:38 +01:00

766 lines
15 KiB
C

/***************************************************************************
gbx_watch.c
(c) 2000-2017 Benoît Minisini <benoit.minisini@gambas-basic.org>
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_WATCH_C
#include "gb_common.h"
#include "gb_error.h"
#include <sys/time.h>
#include <sys/types.h>
#include <unistd.h>
#include "gb_array.h"
#include "gbx_exec.h"
#include "gbx_event.h"
#include "gbx_date.h"
#include "gbx_c_timer.h"
#include "gbx_watch.h"
//#define DEBUG_TIMER 1
//#define DEBUG_WATCH 1
static fd_set read_fd;
static fd_set write_fd;
static WATCH_CALLBACK *watch_callback = NULL;
//static short *watch_index = NULL;
static int max_fd = 0;
static WATCH_TIMER *_timers = NULL;
static int _do_not_really_delete_callback = 0;
static bool _must_delete_callback = FALSE;
static struct timeval _last_time = { 0 };
#ifdef DEBUG_TIMER
double _debug_time;
static double time_to_double(const struct timeval *t)
{
return (double)t->tv_sec + t->tv_usec / 1E6 - _debug_time;
}
#endif
static void time_add(struct timeval *t1, const struct timeval *t2)
{
t1->tv_sec += t2->tv_sec;
t1->tv_usec += t2->tv_usec;
if (t1->tv_usec > 1000000)
{
t1->tv_usec -= 1000000;
t1->tv_sec++;
}
}
static void time_sub(struct timeval *t1, const struct timeval *t2)
{
t1->tv_sec -= t2->tv_sec;
t1->tv_usec -= t2->tv_usec;
if (t1->tv_usec < 0)
{
t1->tv_usec += 1000000;
t1->tv_sec--;
}
}
#define time_comp(t1, t2, op) ((t1)->tv_sec op (t2)->tv_sec || ((t1)->tv_sec == (t2)->tv_sec && (t1)->tv_usec op (t2)->tv_usec))
#define time_lower_than(t1, t2) time_comp(t1, t2, <)
static void time_from_ms(struct timeval *t, int ms)
{
t->tv_sec = ms / 1000;
t->tv_usec = (ms % 1000) * 1000;
}
static struct timeval *time_now(void)
{
static struct timeval current;
gettimeofday(&current, NULL);
if (current.tv_usec < 0 || current.tv_usec >= 1000000)
fprintf(stderr, "gbx3: warning: gettimeofday: tv_usec = %ld!\n", current.tv_usec);
return &current;
}
static void add_timer(GB_TIMER *timer, const struct timeval *timeout)
{
int i;
WATCH_TIMER *wt;
for (i = 0; i < ARRAY_count(_timers); i++)
{
//if (_timers[i].timeout > timeout)
if (time_lower_than(timeout, &_timers[i].timeout))
break;
}
wt = ARRAY_insert(&_timers, i);
wt->timer = timer;
wt->timeout = *timeout;
#ifdef DEBUG_TIMER
fprintf(stderr, "add_timer: %p at %i: %g\n", timer, i, time_to_double(timeout));
#endif
}
static int find_timer(GB_TIMER *timer)
{
int i;
for (i = 0; i < ARRAY_count(_timers); i++)
{
if (_timers[i].timer == timer)
return i;
}
return (-1);
}
static void remove_timer(GB_TIMER *timer)
{
int i = find_timer(timer);
if (i >= 0)
{
ARRAY_remove(&_timers, i)
#ifdef DEBUG_TIMER
fprintf(stderr, "remove_timer: %p at %i\n", timer, i);
#endif
}
}
double WATCH_get_timeout(GB_TIMER *timer)
{
int i = find_timer(timer);
if (i < 0)
return 0.0;
else
return DATE_to_double(&_timers[i].timeout, TRUE);
}
static void raise_timers()
{
struct timeval timeout;
struct timeval delay;
struct timeval *now;
GB_TIMER *timer;
now = time_now();
while (ARRAY_count(_timers))
{
timeout = _timers[0].timeout;
if (time_lower_than(now, &timeout))
return;
timer = _timers[0].timer;
#ifdef DEBUG_TIMER
fprintf(stderr, "raise_timers: %p has been triggered! now = %.7f timeout = %.7f\n", timer, time_to_double(now), time_to_double(&timeout));
#endif
time_from_ms(&delay, timer->delay);
time_add(&timeout, &delay);
if (time_lower_than(&timeout, now))
{
timeout = *now;
time_add(&timeout, &delay);
}
remove_timer(timer);
add_timer(timer, &timeout);
CTIMER_raise(timer);
}
}
static bool get_timeout(const struct timeval *wait, struct timeval *tv)
{
struct timeval timeout;
struct timeval *now;
if (ARRAY_count(_timers) == 0 && !wait)
return TRUE;
now = time_now();
#ifdef DEBUG_TIMER
fprintf(stderr, "get_timeout: now = %.7g timeout = %.7g\n", time_to_double(now), time_to_double(&_timers[0].timeout));
#endif
if (ARRAY_count(_timers) == 0)
timeout = *wait;
else
{
timeout = _timers[0].timeout;
if (wait && time_lower_than(wait, &timeout))
timeout = *wait;
}
time_sub(&timeout, now);
// HZ = 100 on Linux. If a timer must be triggered in less than 10 ms, then
// we do a busy wait instead of calling the system.
if (timeout.tv_sec == 0 && timeout.tv_usec < (1000000 / 100))
{
#ifdef DEBUG_TIMER
fprintf(stderr, "busy loop\n");
#endif
// busy loop!
time_add(&timeout, now);
for(;;)
{
now = time_now();
if (!time_lower_than(now, &timeout))
break;
}
timeout.tv_sec = 0;
timeout.tv_usec = 0;
}
else if (timeout.tv_sec < 0)
{
#ifdef DEBUG_TIMER
fprintf(stderr, "timeout < 0: %ld %ld\n", timeout.tv_sec, timeout.tv_usec);
#endif
timeout.tv_sec = 0;
timeout.tv_usec = 0;
}
*tv = timeout;
#ifdef DEBUG_TIMER
fprintf(stderr, "timeout: %.7f %ld %ld\n", timeout.tv_sec + timeout.tv_usec / 1E6, tv->tv_sec, tv->tv_usec);
#endif
return FALSE;
}
void WATCH_init(void)
{
FD_ZERO(&read_fd);
FD_ZERO(&write_fd);
ARRAY_create(&watch_callback);
//ARRAY_create(&watch_index);
ARRAY_create(&_timers);
#ifdef DEBUG_TIMER
DATE_timer(&_debug_time, FALSE);
#endif
}
void WATCH_exit(void)
{
ARRAY_delete(&watch_callback);
//ARRAY_delete(&watch_index);
ARRAY_delete(&_timers);
}
static void watch_fd(int fd, int flag, bool watch)
{
#if DEBUG_WATCH
fprintf(stderr, "watch_fd: %d for %s: %s\n", fd, flag == GB_WATCH_READ ? "read" : "write", watch ? "set" : "clear");
#endif
if (flag == GB_WATCH_READ)
{
if (watch)
FD_SET(fd, &read_fd);
else
FD_CLR(fd, &read_fd);
}
else if (flag == GB_WATCH_WRITE)
{
if (watch)
FD_SET(fd, &write_fd);
else
FD_CLR(fd, &write_fd);
}
}
static int watch_find_callback(int fd)
{
int i;
for (i = 0; i < ARRAY_count(watch_callback); i++)
{
if (watch_callback[i].fd == fd)
return i;
}
return -1;
}
static int find_max_fd(void)
{
int i;
int max = -1;
for (i = 0; i < ARRAY_count(watch_callback); i++)
{
if (watch_callback[i].fd > max)
max = watch_callback[i].fd;
}
return max;
}
static WATCH_CALLBACK *watch_create_callback(int fd)
{
int pos;
WATCH_CALLBACK *wcb;
#if DEBUG_WATCH
fprintf(stderr, "watch_create_callback: %d\n", fd);
#endif
pos = watch_find_callback(fd);
if (pos < 0)
{
pos = ARRAY_count(watch_callback);
wcb = ARRAY_add_void(&watch_callback);
wcb->fd = fd;
}
else
wcb = &watch_callback[pos];
if (fd > max_fd)
max_fd = fd;
#if DEBUG_WATCH
fprintf(stderr, "watch_create_callback: %d -> %d read = %p (%p) write = %p (%p)\n", fd, pos, wcb->callback_read, (void *)wcb->param_read, wcb->callback_write, (void *)wcb->param_write);
#endif
//ensure_watch_index(fd);
//watch_index[fd] = pos;
return wcb;
}
static void watch_delete_callback(int pos)
{
WATCH_CALLBACK *wcb;
int fd;
#if DEBUG_WATCH
fprintf(stderr, "watch_delete_callback: %d (%d)\n", fd, _do_not_really_delete_callback);
#endif
//watch_index[fd] = -1;
wcb = &watch_callback[pos];
fd = wcb->fd,
wcb->fd = -1;
watch_fd(fd, GB_WATCH_READ, FALSE);
watch_fd(fd, GB_WATCH_WRITE, FALSE);
max_fd = find_max_fd();
if (_do_not_really_delete_callback)
{
#if DEBUG_WATCH
fprintf(stderr, "--> do not really delete\n");
#endif
_must_delete_callback = TRUE;
return;
}
ARRAY_remove(&watch_callback, pos);
#if DEBUG_WATCH
fprintf(stderr, "--> deleted\n");
#endif
}
static void watch_delete_callback_from_fd(int fd)
{
int pos = watch_find_callback(fd);
if (pos >= 0)
watch_delete_callback(pos);
}
void WATCH_watch(int fd, int type, void *callback, intptr_t param)
{
WATCH_CALLBACK *wcb;
if (fd < 0 || fd > FD_SETSIZE)
{
if (type != GB_WATCH_NONE)
ERROR_warning("trying to watch fd #%d", fd);
return;
}
if (type == GB_WATCH_NONE)
watch_delete_callback_from_fd(fd);
else
{
wcb = watch_create_callback(fd);
if (type == GB_WATCH_READ)
{
wcb->callback_read = callback;
wcb->param_read = param;
}
else
{
wcb->callback_write = callback;
wcb->param_write = param;
}
#if DEBUG_WATCH
fprintf(stderr, "add watch: %d -> %d read = %p (%p) write = %p (%p)\n", fd, watch_find_callback(fd), wcb->callback_read, (void *)wcb->param_read, wcb->callback_write, (void *)wcb->param_write);
#endif
if (!wcb->callback_read && !wcb->callback_write)
watch_delete_callback_from_fd(fd);
else
{
watch_fd(fd, GB_WATCH_READ, wcb->callback_read != NULL);
watch_fd(fd, GB_WATCH_WRITE, wcb->callback_write != NULL);
}
}
}
void WATCH_transfer_watch(void)
{
WATCH_CALLBACK *wcb;
int i;
for (i = 0; i < ARRAY_count(watch_callback); i++)
{
wcb = &watch_callback[i];
if (wcb->callback_read) HOOK(watch)(wcb->fd, GB_WATCH_READ, wcb->callback_read, wcb->param_read);
if (wcb->callback_write) HOOK(watch)(wcb->fd, GB_WATCH_WRITE, wcb->callback_write, wcb->param_write);
}
ARRAY_delete(&watch_callback);
ARRAY_create(&watch_callback);
}
void WATCH_transfer_timer(void)
{
WATCH_TIMER *wt;
int i;
for (i = 0; i < ARRAY_count(_timers); i++)
{
wt = &_timers[i];
HOOK(timer)((GB_TIMER *)wt->timer, TRUE);
}
ARRAY_delete(&_timers);
ARRAY_create(&_timers);
}
static void raise_callback(fd_set *rfd, fd_set *wfd)
{
int i;
WATCH_CALLBACK wcb;
_do_not_really_delete_callback++;
#if DEBUG_WATCH
fprintf(stderr, "raise_callback: max_fd = %d\n", max_fd);
#endif
for (i = 0; i < ARRAY_count(watch_callback); i++)
{
// We copy the callback structure, because the watch_callback array can change during the
// execution of the callbacks.
if (watch_callback[i].fd < 0)
continue;
wcb = watch_callback[i];
#if DEBUG_WATCH
fprintf(stderr, "raise_callback: [%d] fd = %d read = %p (%p) write = %p (%p)\n", i, wcb.fd, wcb.callback_read, (void *)wcb.param_read, wcb.callback_write, (void *)wcb.param_write);
#endif
if (FD_ISSET(wcb.fd, rfd))
{
FD_CLR(wcb.fd, rfd);
if (wcb.callback_read)
{
#ifdef DEBUG_WATCH
fprintf(stderr, "call read callback on fd %d\n", wcb.fd);
#endif
(*(wcb.callback_read))(wcb.fd, GB_WATCH_READ, wcb.param_read);
}
}
if (watch_callback[i].fd < 0)
continue;
if (FD_ISSET(wcb.fd, wfd))
{
FD_CLR(wcb.fd, wfd);
if (wcb.callback_write)
{
#ifdef DEBUG_WATCH
fprintf(stderr, "call write callback on fd %d\n", wcb.fd);
#endif
(*(wcb.callback_write))(wcb.fd, GB_WATCH_WRITE, wcb.param_write);
}
}
}
_do_not_really_delete_callback--;
if (!_do_not_really_delete_callback && _must_delete_callback)
{
#if DEBUG_WATCH
fprintf(stderr, "do must delete callback\n");
#endif
i = 0;
while (i < ARRAY_count(watch_callback))
{
if (watch_callback[i].fd < 0)
{
ARRAY_remove(&watch_callback, i);
}
else
i++;
}
_must_delete_callback = FALSE;
}
}
static int do_select(fd_set *rfd, fd_set *wfd, struct timeval *timeout)
{
int fd;
for (fd = max_fd; fd >= 0; fd--)
{
if (FD_ISSET(fd, &read_fd) || FD_ISSET(fd, &write_fd))
break;
}
if (fd < 0 && !timeout)
return 0;
max_fd = fd;
*rfd = read_fd;
*wfd = write_fd;
return select(max_fd + 1, rfd, wfd, NULL, timeout);
}
static bool do_loop(struct timeval *wait)
{
int ret;
struct timeval tv;
fd_set rfd, wfd;
bool something_done = FALSE;
if (EVENT_check_post())
something_done = TRUE;
#ifdef DEBUG_TIMER
fprintf(stderr, "\ndo_loop: now = %.7g: select (%d)\n", time_to_double(time_now()), something_done);
#endif
if (get_timeout(wait, &tv))
ret = do_select(&rfd, &wfd, NULL);
else
{
ret = do_select(&rfd, &wfd, &tv);
something_done = ARRAY_count(_timers) > 0;
}
if (ret < 0 && errno != EINTR)
THROW_SYSTEM(errno, NULL);
#ifdef DEBUG_TIMER
fprintf(stderr, "do_loop: now = %.7g: timers (%d)\n", time_to_double(time_now()), something_done);
#endif
raise_timers();
#ifdef DEBUG_TIMER
fprintf(stderr, "do_loop: now = %.7g: callbacks\n", time_to_double(time_now()));
#endif
if (ret > 0)
{
raise_callback(&rfd, &wfd);
something_done = TRUE;
}
else if (ret < 0)
{
something_done = TRUE;
}
//if (EVENT_check_post())
// something_done = TRUE;
return something_done;
}
bool WATCH_one_loop(int wait)
{
struct timeval timeout;
if (wait == 0)
return do_loop(NULL);
else
{
time_from_ms(&timeout, wait);
return do_loop(&timeout);
}
}
void WATCH_loop(void)
{
while (do_loop(NULL));
}
void WATCH_wait(int wait)
{
struct timeval *now;
struct timeval timeout;
if (wait == -2)
{
do_loop(NULL);
}
else if (wait <= 0)
{
timeout.tv_sec = 0;
timeout.tv_usec = 0;
do_loop(&timeout);
}
else
{
now = time_now();
time_from_ms(&timeout, wait);
time_add(&timeout, now);
for(;;)
{
now = time_now();
if (time_lower_than(&timeout, now))
break;
do_loop(&timeout);
}
}
}
int WATCH_process(int fd_end, int fd_output, int fd_error, int timeout)
{
fd_set rfd;
int ret, fd_max;
struct timeval tv;
fd_max = fd_end > fd_output ? fd_end : fd_output;
fd_max = fd_error > fd_max ? fd_error : fd_max;
for(;;)
{
FD_ZERO(&rfd);
FD_SET(fd_end, &rfd);
if (fd_output >= 0)
FD_SET(fd_output, &rfd);
if (fd_error >= 0)
FD_SET(fd_error, &rfd);
if (timeout > 0)
{
tv.tv_sec = timeout / 1000;
tv.tv_usec = (timeout % 1000) * 1000;
ret = select(fd_max + 1, &rfd, NULL, NULL, &tv);
if (ret == 0)
break;
}
else
{
ret = select(fd_max + 1, &rfd, NULL, NULL, NULL);
}
//fprintf(stderr, "WATCH_process: select -> %d [%d] / fd_end -> %d\n", ret, errno, FD_ISSET(fd_end, &rfd));
if (ret > 0)
break;
if (errno != EINTR)
break;
}
ret = timeout > 0 ? WP_TIMEOUT : WP_NOTHING;
if (FD_ISSET(fd_end, &rfd)) ret += WP_END;
if (fd_output >= 0 && FD_ISSET(fd_output, &rfd)) ret += WP_OUTPUT;
if (fd_error >= 0 && FD_ISSET(fd_error, &rfd)) ret += WP_ERROR;
return ret;
}
void WATCH_timer(void *t, int on)
{
GB_TIMER *timer = (GB_TIMER *)t;
struct timeval timeout;
if (on)
{
time_from_ms(&timeout, timer->delay);
time_add(&timeout, time_now());
add_timer(timer, &timeout);
timer->id = (intptr_t)timer;
}
else
{
remove_timer(timer);
timer->id = 0;
}
}
void WATCH_little_sleep(void)
{
struct timeval tv;
struct timeval *now;
now = time_now();
tv = *now;
time_sub(&tv, &_last_time);
if (tv.tv_sec == 0 && tv.tv_usec < 10000)
usleep(10000 - tv.tv_usec);
_last_time = *now;
}