NetBSD/usr.sbin/mrouted/callout.c

228 lines
4.1 KiB
C
Raw Normal View History

/* $NetBSD: callout.c,v 1.5 2002/07/14 16:30:41 wiz Exp $ */
1995-10-09 06:51:32 +03:00
1995-06-01 06:25:45 +04:00
/*
* The mrouted program is covered by the license in the accompanying file
* named "LICENSE". Use of the mrouted program represents acceptance of
* the terms and conditions listed in that file.
*
* The mrouted program is COPYRIGHT 1989 by The Board of Trustees of
* Leland Stanford Junior University.
*/
#include "defs.h"
/* the code below implements a callout queue */
static int id = 0;
static struct timeout_q *Q = 0; /* pointer to the beginning of timeout queue */
1995-12-10 13:06:55 +03:00
static int in_callout = 0;
1995-06-01 06:25:45 +04:00
struct timeout_q {
struct timeout_q *next; /* next event */
int id;
1995-12-10 13:06:55 +03:00
cfunc_t func; /* function to call */
1995-06-01 06:25:45 +04:00
char *data; /* func's data */
int time; /* time offset to next event*/
};
1995-12-10 13:06:55 +03:00
#ifdef IGMP_DEBUG
static void print_Q(void);
1995-12-10 13:06:55 +03:00
#else
#define print_Q()
#endif
1995-06-01 06:25:45 +04:00
int secs_remaining(int);
1997-10-17 14:28:39 +04:00
1995-12-10 13:06:55 +03:00
void
callout_init(void)
1995-06-01 06:25:45 +04:00
{
Q = (struct timeout_q *) 0;
}
/*
* signal handler for SIGALARM that is called once every second
*/
1995-12-10 13:06:55 +03:00
void
age_callout_queue(void)
1995-06-01 06:25:45 +04:00
{
struct timeout_q *ptr;
if (in_callout)
return;
in_callout = 1;
ptr = Q;
1995-12-10 13:06:55 +03:00
while (ptr) {
if (!ptr->time) {
1995-06-01 06:25:45 +04:00
/* timeout has happened */
Q = Q->next;
1995-12-10 13:06:55 +03:00
in_callout = 0;
if (ptr->func)
ptr->func(ptr->data);
in_callout = 1;
1995-06-01 06:25:45 +04:00
free(ptr);
ptr = Q;
}
else {
ptr->time --;
#ifdef IGMP_DEBUG
log(LOG_DEBUG,0,"[callout, age_callout_queue] -- time (%d)", ptr->time);
1995-12-10 13:06:55 +03:00
#endif /* IGMP_DEBUG */
1995-06-01 06:25:45 +04:00
in_callout = 0; return;
}
}
in_callout = 0;
return;
}
/*
* sets the timer
*
* delay: number of units for timeout
* action: function to be called on timeout
* data: what to call the timeout function with
1995-06-01 06:25:45 +04:00
*/
1995-12-10 13:06:55 +03:00
int
timer_setTimer(int delay, cfunc_t action, char *data)
1995-06-01 06:25:45 +04:00
{
struct timeout_q *ptr, *node, *prev;
if (in_callout)
return -1;
in_callout = 1;
/* create a node */
1995-12-10 13:06:55 +03:00
node = (struct timeout_q *)malloc(sizeof(struct timeout_q));
1995-06-01 06:25:45 +04:00
if (node == 0) {
log(LOG_WARNING, 0, "Malloc Failed in timer_settimer\n");
in_callout = 0;
return -1;
}
node->func = action;
node->data = data;
node->time = delay;
node->next = 0;
node->id = ++id;
prev = ptr = Q;
/* insert node in the queue */
/* if the queue is empty, insert the node and return */
if (!Q)
Q = node;
else {
/* chase the pointer looking for the right place */
1995-12-10 13:06:55 +03:00
while (ptr) {
1995-06-01 06:25:45 +04:00
1995-12-10 13:06:55 +03:00
if (delay < ptr->time) {
1995-06-01 06:25:45 +04:00
/* right place */
node->next = ptr;
if (ptr == Q)
Q = node;
else
prev->next = node;
ptr->time -= node->time;
print_Q();
in_callout = 0;
return node->id;
1995-12-10 13:06:55 +03:00
} else {
1995-06-01 06:25:45 +04:00
/* keep moving */
delay -= ptr->time; node->time = delay;
prev = ptr;
ptr = ptr->next;
}
}
prev->next = node;
}
print_Q();
in_callout = 0;
return node->id;
}
/* clears the associated timer */
1995-12-10 13:06:55 +03:00
void
timer_clearTimer(int timer_id)
1995-06-01 06:25:45 +04:00
{
struct timeout_q *ptr, *prev;
1995-12-10 13:06:55 +03:00
if (in_callout)
return;
if (!timer_id)
return;
1995-06-01 06:25:45 +04:00
in_callout = 1;
prev = ptr = Q;
/*
* find the right node, delete it. the subsequent node's time
* gets bumped up
*/
print_Q();
1995-12-10 13:06:55 +03:00
while (ptr) {
if (ptr->id == timer_id) {
1995-06-01 06:25:45 +04:00
/* got the right node */
/* unlink it from the queue */
1995-12-10 13:06:55 +03:00
if (ptr == Q)
1995-06-01 06:25:45 +04:00
Q = Q->next;
else
prev->next = ptr->next;
/* increment next node if any */
if (ptr->next != 0)
(ptr->next)->time += ptr->time;
free(ptr->data);
free(ptr);
print_Q();
in_callout = 0;
return;
}
prev = ptr;
ptr = ptr->next;
}
print_Q();
in_callout = 0;
}
1995-12-10 13:06:55 +03:00
#ifdef IGMP_DEBUG
1995-06-01 06:25:45 +04:00
/*
* debugging utility
*/
1995-12-10 13:06:55 +03:00
static void
print_Q(void)
1995-06-01 06:25:45 +04:00
{
struct timeout_q *ptr;
for(ptr = Q; ptr; ptr = ptr->next)
log(LOG_DEBUG,0,"(%d,%d) ", ptr->id, ptr->time);
}
1995-12-10 13:06:55 +03:00
#endif /* IGMP_DEBUG */
1997-10-17 14:28:39 +04:00
1995-12-10 13:06:55 +03:00
int
secs_remaining(int timer_id)
1995-12-10 13:06:55 +03:00
{
struct timeout_q *ptr;
int left=0;
for (ptr = Q; ptr && ptr->id != timer_id; ptr = ptr->next)
left += ptr->time;
1995-06-01 06:25:45 +04:00
1995-12-10 13:06:55 +03:00
if (!ptr) /* not found */
return 0;
return left + ptr->time;
}