gambas-source-code/main/gbx/gbx_c_timer.c
Benoît Minisini 33ec6b88e3 'GB.Every()' API now creates a timer that is ignored by the event loop.
[INTERPRETER]
* NEW: 'GB.Every()' API now creates a timer that is ignored by the event loop.
2022-05-26 20:47:52 +02:00

262 lines
4.9 KiB
C

/***************************************************************************
gbx_c_timer.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_TIMER_C
#include "gbx_info.h"
#ifndef GBX_INFO
#include "gb_common.h"
#include "gbx_watch.h"
#include "gbx_api.h"
#include "gbx_exec.h"
#include "gbx_event.h"
#include "gbx_c_timer.h"
#define EXT(_ob) ((CTIMER_EXT *)((CTIMER *)_ob)->ext)
int CTIMER_active_count = 0;
DECLARE_EVENT(EVENT_Timer);
static void enable_timer(CTIMER *_object, bool on)
{
if (on == (THIS->id != 0))
return;
HOOK_DEFAULT(timer, WATCH_timer)((GB_TIMER *)THIS, on);
if (on && (THIS->id == 0))
GB_Error("Too many active timers");
if (!THIS->ignore)
{
CTIMER_active_count += on ? 1 : -1;
/*if (!on)
fprintf(stderr, "disable_timer: %d -> %d\n", THIS->delay, CTIMER_active_count);*/
}
}
CTIMER *CTIMER_every(int delay, GB_TIMER_CALLBACK callback, intptr_t param)
{
CTIMER *timer;
timer = OBJECT_create(CLASS_Timer, NULL, NULL, 0);
OBJECT_REF(timer);
timer->delay = delay;
timer->task = EXEC_task;
timer->ignore = TRUE;
ALLOC_ZERO(&timer->ext, sizeof(CTIMER_EXT));
EXT(timer)->callback = callback;
EXT(timer)->tag = param;
enable_timer(timer, TRUE);
return timer;
}
void CTIMER_raise(void *_object)
{
if (THIS->task == EXEC_task)
{
if (THIS_EXT && THIS_EXT->callback)
{
if (!(*(THIS_EXT->callback))(THIS_EXT->tag))
return;
}
else
{
void *parent = OBJECT_parent(THIS);
if (parent && OBJECT_is_valid(parent) && !GB_Raise(THIS, EVENT_Timer, 0))
return;
}
}
enable_timer(THIS, FALSE);
}
BEGIN_METHOD(Timer_new, GB_INTEGER delay)
int delay;
THIS->id = 0;
delay = VARGOPT(delay, -1);
if (delay < 0)
delay = 1000;
THIS->task = EXEC_task;
THIS->delay = delay;
if (!MISSING(delay))
enable_timer(THIS, TRUE);
END_METHOD
BEGIN_METHOD_VOID(Timer_free)
enable_timer(THIS, FALSE);
if (THIS->ext)
IFREE(THIS->ext);
END_METHOD
BEGIN_METHOD_VOID(Timer_Start)
enable_timer(THIS, TRUE);
END_METHOD
BEGIN_METHOD_VOID(Timer_Stop)
enable_timer(THIS, FALSE);
THIS->triggered = FALSE;
END_METHOD
BEGIN_METHOD_VOID(Timer_Restart)
enable_timer(THIS, FALSE);
enable_timer(THIS, TRUE);
END_METHOD
BEGIN_PROPERTY(Timer_Enabled)
if (READ_PROPERTY)
GB_ReturnBoolean(THIS->id != 0);
else
enable_timer(THIS, !!VPROP(GB_BOOLEAN));
END_PROPERTY
BEGIN_PROPERTY(Timer_Delay)
if (READ_PROPERTY)
GB_ReturnInteger(THIS->delay);
else
{
int delay = VPROP(GB_INTEGER);
bool enabled = THIS->id != 0;
if (delay < 0 || delay >= (1<<30))
{
GB_Error(GB_ERR_ARG);
return;
}
if (enabled)
HOOK_DEFAULT(timer, WATCH_timer)((GB_TIMER *)THIS, FALSE);
THIS->delay = delay;
if (enabled)
HOOK_DEFAULT(timer, WATCH_timer)((GB_TIMER *)THIS, TRUE);
}
END_PROPERTY
static void trigger_timer(void *_object)
{
if (THIS->triggered)
{
THIS->triggered = FALSE;
GB_Raise(THIS, EVENT_Timer, 0);
}
OBJECT_UNREF(_object);
}
BEGIN_METHOD_VOID(Timer_Trigger)
if (THIS->triggered)
return;
THIS->triggered = TRUE;
OBJECT_REF(THIS);
EVENT_post(trigger_timer, (intptr_t)THIS);
END_METHOD
BEGIN_PROPERTY(Timer_Ignore)
if (READ_PROPERTY)
GB_ReturnBoolean(THIS->ignore);
else
{
bool v = VPROP(GB_BOOLEAN);
if (THIS->ignore == v)
return;
THIS->ignore = v;
if (THIS->id)
CTIMER_active_count += THIS->ignore ? -1 : 1;
}
END_PROPERTY
#endif
GB_DESC NATIVE_Timer[] =
{
GB_DECLARE("Timer", sizeof(CTIMER)),
GB_METHOD("_new", NULL, Timer_new, "[(Delay)i]"),
GB_METHOD("_free", NULL, Timer_free, NULL),
GB_PROPERTY("Enabled", "b", Timer_Enabled),
GB_PROPERTY("Delay", "i", Timer_Delay),
GB_PROPERTY("Ignore", "b", Timer_Ignore),
//GB_PROPERTY_READ("Timeout", "f", Timer_Timeout),
GB_METHOD("Start", NULL, Timer_Start, NULL),
GB_METHOD("Stop", NULL, Timer_Stop, NULL),
GB_METHOD("Restart", NULL, Timer_Restart, NULL),
GB_METHOD("Trigger", NULL, Timer_Trigger, NULL),
GB_CONSTANT("_IsControl", "b", TRUE),
GB_CONSTANT("_IsVirtual", "b", TRUE),
GB_CONSTANT("_Group", "s", "Special"),
GB_CONSTANT("_Properties", "s", "Enabled,Delay{Range:0;86400000;10;ms}=1000,Ignore"),
GB_CONSTANT("_DefaultEvent", "s", "Timer"),
GB_EVENT("Timer", NULL, NULL, &EVENT_Timer),
GB_END_DECLARE
};