mirror of
https://gitlab.isc.org/isc-projects/bind9
synced 2025-08-30 05:57:52 +00:00
Merge branch 'ondrej-repair-isc_task_purgeevent' into 'main'
Repair isc_task_purgeevent(), clean isc_task_unsend{,range}() Closes #3252 See merge request isc-projects/bind9!6053
This commit is contained in:
commit
abb5e9a575
7
CHANGES
7
CHANGES
@ -1,3 +1,10 @@
|
||||
5845. [bug] Refactor the timer to keep track of posted events
|
||||
as to use isc_task_purgeevent() instead of using
|
||||
isc_task_purgerange(). The isc_task_purgeevent()
|
||||
has been refactored to purge a single event instead
|
||||
of walking through the list of posted events.
|
||||
[GL #3252]
|
||||
|
||||
5844. [bug] dig +nssearch was hanging until manually interrupted.
|
||||
[GL #3145]
|
||||
|
||||
|
@ -963,7 +963,7 @@ cleaner_shutdown_action(isc_task_t *task, isc_event_t *event) {
|
||||
}
|
||||
|
||||
/* Make sure we don't reschedule anymore. */
|
||||
(void)isc_task_purge(task, NULL, DNS_EVENT_CACHECLEAN, NULL);
|
||||
(void)isc_task_purgeevent(task, cache->cleaner.resched_event);
|
||||
|
||||
isc_refcount_decrementz(&cache->live_tasks);
|
||||
|
||||
|
@ -82,6 +82,3 @@
|
||||
#define DNS_EVENT_TRYSTALE (ISC_EVENTCLASS_DNS + 59)
|
||||
#define DNS_EVENT_ZONEFLUSH (ISC_EVENTCLASS_DNS + 60)
|
||||
#define DNS_EVENT_CHECKDSSENDTOADDR (ISC_EVENTCLASS_DNS + 61)
|
||||
|
||||
#define DNS_EVENT_FIRSTEVENT (ISC_EVENTCLASS_DNS + 0)
|
||||
#define DNS_EVENT_LASTEVENT (ISC_EVENTCLASS_DNS + 65535)
|
||||
|
@ -849,6 +849,11 @@ unsigned int dns_zone_mkey_month = MONTH;
|
||||
|
||||
#define SEND_BUFFER_SIZE 2048
|
||||
|
||||
static void
|
||||
zone_timer_start(dns_zone_t *zone, isc_time_t *next, isc_time_t *now);
|
||||
static void
|
||||
zone_timer_stop(dns_zone_t *zone);
|
||||
|
||||
static void
|
||||
zone_settimer(dns_zone_t *, isc_time_t *);
|
||||
static void
|
||||
@ -2426,6 +2431,9 @@ zone_asyncload(isc_task_t *task, isc_event_t *event) {
|
||||
(asl->loaded)(asl->loaded_arg, zone, task);
|
||||
}
|
||||
|
||||
/* Reduce the quantum */
|
||||
isc_task_setquantum(zone->loadtask, 1);
|
||||
|
||||
isc_mem_put(zone->mctx, asl, sizeof(*asl));
|
||||
dns_zone_idetach(&zone);
|
||||
}
|
||||
@ -15074,6 +15082,7 @@ zone_shutdown(isc_task_t *task, isc_event_t *event) {
|
||||
forward_cancel(zone);
|
||||
|
||||
if (zone->timer != NULL) {
|
||||
zone_timer_stop(zone);
|
||||
isc_timer_detach(&zone->timer);
|
||||
isc_refcount_decrement(&zone->irefs);
|
||||
}
|
||||
@ -15126,11 +15135,41 @@ zone_timer(isc_task_t *task, isc_event_t *event) {
|
||||
isc_event_free(&event);
|
||||
}
|
||||
|
||||
static void
|
||||
zone_timer_start(dns_zone_t *zone, isc_time_t *next, isc_time_t *now) {
|
||||
isc_interval_t interval;
|
||||
isc_result_t result;
|
||||
|
||||
if (isc_time_compare(next, now) <= 0) {
|
||||
isc_interval_set(&interval, 0, 1);
|
||||
} else {
|
||||
isc_time_subtract(next, now, &interval);
|
||||
}
|
||||
|
||||
result = isc_timer_reset(zone->timer, isc_timertype_once, &interval,
|
||||
true);
|
||||
if (result != ISC_R_SUCCESS) {
|
||||
dns_zone_log(zone, ISC_LOG_ERROR,
|
||||
"could not reset zone timer: %s",
|
||||
isc_result_totext(result));
|
||||
}
|
||||
}
|
||||
|
||||
static void
|
||||
zone_timer_stop(dns_zone_t *zone) {
|
||||
isc_result_t result = isc_timer_reset(
|
||||
zone->timer, isc_timertype_inactive, NULL, true);
|
||||
if (result != ISC_R_SUCCESS) {
|
||||
dns_zone_log(zone, ISC_LOG_ERROR,
|
||||
"could not deactivate zone timer: %s",
|
||||
isc_result_totext(result));
|
||||
}
|
||||
}
|
||||
|
||||
static void
|
||||
zone_settimer(dns_zone_t *zone, isc_time_t *now) {
|
||||
const char me[] = "zone_settimer";
|
||||
isc_time_t next;
|
||||
isc_result_t result;
|
||||
|
||||
REQUIRE(DNS_ZONE_VALID(zone));
|
||||
REQUIRE(LOCKED_ZONE(zone));
|
||||
@ -15269,28 +15308,9 @@ zone_settimer(dns_zone_t *zone, isc_time_t *now) {
|
||||
|
||||
if (isc_time_isepoch(&next)) {
|
||||
zone_debuglog(zone, me, 10, "settimer inactive");
|
||||
result = isc_timer_reset(zone->timer, isc_timertype_inactive,
|
||||
NULL, true);
|
||||
if (result != ISC_R_SUCCESS) {
|
||||
dns_zone_log(zone, ISC_LOG_ERROR,
|
||||
"could not deactivate zone timer: %s",
|
||||
isc_result_totext(result));
|
||||
}
|
||||
zone_timer_stop(zone);
|
||||
} else {
|
||||
isc_interval_t interval;
|
||||
if (isc_time_compare(&next, now) <= 0) {
|
||||
isc_interval_set(&interval, 0, 1);
|
||||
} else {
|
||||
isc_time_subtract(&next, now, &interval);
|
||||
}
|
||||
|
||||
result = isc_timer_reset(zone->timer, isc_timertype_once,
|
||||
&interval, true);
|
||||
if (result != ISC_R_SUCCESS) {
|
||||
dns_zone_log(zone, ISC_LOG_ERROR,
|
||||
"could not reset zone timer: %s",
|
||||
isc_result_totext(result));
|
||||
}
|
||||
zone_timer_start(zone, &next, now);
|
||||
}
|
||||
}
|
||||
|
||||
@ -19167,7 +19187,7 @@ dns_zonemgr_setsize(dns_zonemgr_t *zmgr, int num_zones) {
|
||||
pool = NULL;
|
||||
if (zmgr->loadtasks == NULL) {
|
||||
result = isc_taskpool_create(zmgr->taskmgr, zmgr->mctx, ntasks,
|
||||
2, true, &pool);
|
||||
UINT_MAX, true, &pool);
|
||||
} else {
|
||||
result = isc_taskpool_expand(&zmgr->loadtasks, ntasks, true,
|
||||
&pool);
|
||||
|
@ -91,9 +91,7 @@
|
||||
|
||||
typedef isc_event_t isc_appevent_t;
|
||||
|
||||
#define ISC_APPEVENT_FIRSTEVENT (ISC_EVENTCLASS_APP + 0)
|
||||
#define ISC_APPEVENT_SHUTDOWN (ISC_EVENTCLASS_APP + 1)
|
||||
#define ISC_APPEVENT_LASTEVENT (ISC_EVENTCLASS_APP + 65535)
|
||||
#define ISC_APPEVENT_SHUTDOWN (ISC_EVENTCLASS_APP + 0)
|
||||
|
||||
ISC_LANG_BEGINDECLS
|
||||
|
||||
|
@ -42,7 +42,6 @@ typedef void (*isc_eventdestructor_t)(isc_event_t *);
|
||||
* definition. Attributes of 0xffffff00 may be used by the application
|
||||
* or non-ISC libraries.
|
||||
*/
|
||||
#define ISC_EVENTATTR_NOPURGE 0x00000001
|
||||
|
||||
/*%
|
||||
* The ISC_EVENTATTR_CANCELED attribute is intended to indicate
|
||||
@ -76,9 +75,6 @@ struct isc_event {
|
||||
ISC_EVENT_COMMON(struct isc_event);
|
||||
};
|
||||
|
||||
#define ISC_EVENTTYPE_FIRSTEVENT 0x00000000
|
||||
#define ISC_EVENTTYPE_LASTEVENT 0xffffffff
|
||||
|
||||
#define ISC_EVENT_PTR(p) ((isc_event_t **)(void *)(p))
|
||||
|
||||
ISC_LANG_BEGINDECLS
|
||||
|
@ -54,18 +54,6 @@
|
||||
*
|
||||
* Purging calls isc_event_free() on the matching events.
|
||||
*
|
||||
* Unsending returns a list of events that matched the pattern.
|
||||
* The caller is then responsible for them.
|
||||
*
|
||||
* Consumers of events should purge, not unsend.
|
||||
*
|
||||
* Producers of events often want to remove events when the caller indicates
|
||||
* it is no longer interested in the object, e.g. by canceling a timer.
|
||||
* Sometimes this can be done by purging, but for some event types, the
|
||||
* calls to isc_event_free() cause deadlock because the event free routine
|
||||
* wants to acquire a lock the caller is already holding. Unsending instead
|
||||
* of purging solves this problem. As a general rule, producers should only
|
||||
* unsend events which they have sent.
|
||||
*/
|
||||
|
||||
/***
|
||||
@ -80,10 +68,8 @@
|
||||
#include <isc/stdtime.h>
|
||||
#include <isc/types.h>
|
||||
|
||||
#define ISC_TASKEVENT_FIRSTEVENT (ISC_EVENTCLASS_TASK + 0)
|
||||
#define ISC_TASKEVENT_SHUTDOWN (ISC_EVENTCLASS_TASK + 1)
|
||||
#define ISC_TASKEVENT_TEST (ISC_EVENTCLASS_TASK + 1)
|
||||
#define ISC_TASKEVENT_LASTEVENT (ISC_EVENTCLASS_TASK + 65535)
|
||||
#define ISC_TASKEVENT_SHUTDOWN (ISC_EVENTCLASS_TASK + 0)
|
||||
#define ISC_TASKEVENT_TEST (ISC_EVENTCLASS_TASK + 1)
|
||||
|
||||
/*****
|
||||
***** Tasks.
|
||||
@ -248,63 +234,6 @@ isc_task_sendanddetach(isc_task_t **taskp, isc_event_t **eventp);
|
||||
* all resources used by the task will be freed.
|
||||
*/
|
||||
|
||||
unsigned int
|
||||
isc_task_purgerange(isc_task_t *task, void *sender, isc_eventtype_t first,
|
||||
isc_eventtype_t last, void *tag);
|
||||
/*%<
|
||||
* Purge events from a task's event queue.
|
||||
*
|
||||
* Requires:
|
||||
*
|
||||
*\li 'task' is a valid task.
|
||||
*
|
||||
*\li last >= first
|
||||
*
|
||||
* Ensures:
|
||||
*
|
||||
*\li Events in the event queue of 'task' whose sender is 'sender', whose
|
||||
* type is >= first and <= last, and whose tag is 'tag' will be purged,
|
||||
* unless they are marked as unpurgable.
|
||||
*
|
||||
*\li A sender of NULL will match any sender. A NULL tag matches any
|
||||
* tag.
|
||||
*
|
||||
* Returns:
|
||||
*
|
||||
*\li The number of events purged.
|
||||
*/
|
||||
|
||||
unsigned int
|
||||
isc_task_purge(isc_task_t *task, void *sender, isc_eventtype_t type, void *tag);
|
||||
/*%<
|
||||
* Purge events from a task's event queue.
|
||||
*
|
||||
* Notes:
|
||||
*
|
||||
*\li This function is equivalent to
|
||||
*
|
||||
*\code
|
||||
* isc_task_purgerange(task, sender, type, type, tag);
|
||||
*\endcode
|
||||
*
|
||||
* Requires:
|
||||
*
|
||||
*\li 'task' is a valid task.
|
||||
*
|
||||
* Ensures:
|
||||
*
|
||||
*\li Events in the event queue of 'task' whose sender is 'sender', whose
|
||||
* type is 'type', and whose tag is 'tag' will be purged, unless they
|
||||
* are marked as unpurgable.
|
||||
*
|
||||
*\li A sender of NULL will match any sender. A NULL tag matches any
|
||||
* tag.
|
||||
*
|
||||
* Returns:
|
||||
*
|
||||
*\li The number of events purged.
|
||||
*/
|
||||
|
||||
bool
|
||||
isc_task_purgeevent(isc_task_t *task, isc_event_t *event);
|
||||
/*%<
|
||||
@ -337,65 +266,6 @@ isc_task_purgeevent(isc_task_t *task, isc_event_t *event);
|
||||
* or was marked unpurgeable.
|
||||
*/
|
||||
|
||||
unsigned int
|
||||
isc_task_unsendrange(isc_task_t *task, void *sender, isc_eventtype_t first,
|
||||
isc_eventtype_t last, void *tag, isc_eventlist_t *events);
|
||||
/*%<
|
||||
* Remove events from a task's event queue.
|
||||
*
|
||||
* Requires:
|
||||
*
|
||||
*\li 'task' is a valid task.
|
||||
*
|
||||
*\li last >= first.
|
||||
*
|
||||
*\li *events is a valid list.
|
||||
*
|
||||
* Ensures:
|
||||
*
|
||||
*\li Events in the event queue of 'task' whose sender is 'sender', whose
|
||||
* type is >= first and <= last, and whose tag is 'tag' will be dequeued
|
||||
* and appended to *events.
|
||||
*
|
||||
*\li A sender of NULL will match any sender. A NULL tag matches any
|
||||
* tag.
|
||||
*
|
||||
* Returns:
|
||||
*
|
||||
*\li The number of events unsent.
|
||||
*/
|
||||
|
||||
unsigned int
|
||||
isc_task_unsend(isc_task_t *task, void *sender, isc_eventtype_t type, void *tag,
|
||||
isc_eventlist_t *events);
|
||||
/*%<
|
||||
* Remove events from a task's event queue.
|
||||
*
|
||||
* Notes:
|
||||
*
|
||||
*\li This function is equivalent to
|
||||
*
|
||||
*\code
|
||||
* isc_task_unsendrange(task, sender, type, type, tag, events);
|
||||
*\endcode
|
||||
*
|
||||
* Requires:
|
||||
*
|
||||
*\li 'task' is a valid task.
|
||||
*
|
||||
*\li *events is a valid list.
|
||||
*
|
||||
* Ensures:
|
||||
*
|
||||
*\li Events in the event queue of 'task' whose sender is 'sender', whose
|
||||
* type is 'type', and whose tag is 'tag' will be dequeued and appended
|
||||
* to *events.
|
||||
*
|
||||
* Returns:
|
||||
*
|
||||
*\li The number of events unsent.
|
||||
*/
|
||||
|
||||
isc_result_t
|
||||
isc_task_onshutdown(isc_task_t *task, isc_taskaction_t action, void *arg);
|
||||
/*%<
|
||||
@ -528,6 +398,15 @@ isc_task_gettag(isc_task_t *task);
|
||||
*\li 'task' is a valid task.
|
||||
*/
|
||||
|
||||
void
|
||||
isc_task_setquantum(isc_task_t *task, unsigned int quantum);
|
||||
/*%<
|
||||
* Set future 'task' quantum to 'quantum'. The current 'task' quantum will be
|
||||
* kept for the current isc_task_run() loop, and will be changed for the next
|
||||
* run. Therefore, the function is save to use from the event callback as it
|
||||
* will not affect the current event loop processing.
|
||||
*/
|
||||
|
||||
isc_result_t
|
||||
isc_task_beginexclusive(isc_task_t *task);
|
||||
/*%<
|
||||
|
@ -81,15 +81,16 @@ typedef enum {
|
||||
isc_timertype_inactive = 3 /*%< Inactive */
|
||||
} isc_timertype_t;
|
||||
|
||||
typedef struct isc_timerevent {
|
||||
typedef struct isc_timerevent isc_timerevent_t;
|
||||
|
||||
struct isc_timerevent {
|
||||
struct isc_event common;
|
||||
isc_time_t due;
|
||||
} isc_timerevent_t;
|
||||
ISC_LINK(isc_timerevent_t) ev_timerlink;
|
||||
};
|
||||
|
||||
#define ISC_TIMEREVENT_FIRSTEVENT (ISC_EVENTCLASS_TIMER + 0)
|
||||
#define ISC_TIMEREVENT_TICK (ISC_EVENTCLASS_TIMER + 1)
|
||||
#define ISC_TIMEREVENT_ONCE (ISC_EVENTCLASS_TIMER + 2)
|
||||
#define ISC_TIMEREVENT_LASTEVENT (ISC_EVENTCLASS_TIMER + 65535)
|
||||
#define ISC_TIMEREVENT_TICK (ISC_EVENTCLASS_TIMER + 0)
|
||||
#define ISC_TIMEREVENT_ONCE (ISC_EVENTCLASS_TIMER + 1)
|
||||
|
||||
/***
|
||||
*** Timer and Timer Manager Functions
|
||||
|
149
lib/isc/task.c
149
lib/isc/task.c
@ -523,101 +523,12 @@ isc_task_sendtoanddetach(isc_task_t **taskp, isc_event_t **eventp, int c) {
|
||||
*taskp = NULL;
|
||||
}
|
||||
|
||||
#define PURGE_OK(event) (((event)->ev_attributes & ISC_EVENTATTR_NOPURGE) == 0)
|
||||
|
||||
static unsigned int
|
||||
dequeue_events(isc_task_t *task, void *sender, isc_eventtype_t first,
|
||||
isc_eventtype_t last, void *tag, isc_eventlist_t *events,
|
||||
bool purging) {
|
||||
isc_event_t *event, *next_event;
|
||||
unsigned int count = 0;
|
||||
|
||||
REQUIRE(VALID_TASK(task));
|
||||
REQUIRE(last >= first);
|
||||
|
||||
XTRACE("dequeue_events");
|
||||
|
||||
/*
|
||||
* Events matching 'sender', whose type is >= first and <= last, and
|
||||
* whose tag is 'tag' will be dequeued. If 'purging', matching events
|
||||
* which are marked as unpurgable will not be dequeued.
|
||||
*
|
||||
* sender == NULL means "any sender", and tag == NULL means "any tag".
|
||||
*/
|
||||
|
||||
LOCK(&task->lock);
|
||||
|
||||
for (event = HEAD(task->events); event != NULL; event = next_event) {
|
||||
next_event = NEXT(event, ev_link);
|
||||
if (event->ev_type >= first && event->ev_type <= last &&
|
||||
(sender == NULL || event->ev_sender == sender) &&
|
||||
(tag == NULL || event->ev_tag == tag) &&
|
||||
(!purging || PURGE_OK(event)))
|
||||
{
|
||||
DEQUEUE(task->events, event, ev_link);
|
||||
task->nevents--;
|
||||
ENQUEUE(*events, event, ev_link);
|
||||
count++;
|
||||
}
|
||||
}
|
||||
|
||||
UNLOCK(&task->lock);
|
||||
|
||||
return (count);
|
||||
}
|
||||
|
||||
unsigned int
|
||||
isc_task_purgerange(isc_task_t *task, void *sender, isc_eventtype_t first,
|
||||
isc_eventtype_t last, void *tag) {
|
||||
unsigned int count;
|
||||
isc_eventlist_t events;
|
||||
isc_event_t *event, *next_event;
|
||||
REQUIRE(VALID_TASK(task));
|
||||
|
||||
/*
|
||||
* Purge events from a task's event queue.
|
||||
*/
|
||||
|
||||
XTRACE("isc_task_purgerange");
|
||||
|
||||
ISC_LIST_INIT(events);
|
||||
|
||||
count = dequeue_events(task, sender, first, last, tag, &events, true);
|
||||
|
||||
for (event = HEAD(events); event != NULL; event = next_event) {
|
||||
next_event = NEXT(event, ev_link);
|
||||
ISC_LIST_UNLINK(events, event, ev_link);
|
||||
isc_event_free(&event);
|
||||
}
|
||||
|
||||
/*
|
||||
* Note that purging never changes the state of the task.
|
||||
*/
|
||||
|
||||
return (count);
|
||||
}
|
||||
|
||||
unsigned int
|
||||
isc_task_purge(isc_task_t *task, void *sender, isc_eventtype_t type,
|
||||
void *tag) {
|
||||
/*
|
||||
* Purge events from a task's event queue.
|
||||
*/
|
||||
REQUIRE(VALID_TASK(task));
|
||||
|
||||
XTRACE("isc_task_purge");
|
||||
|
||||
return (isc_task_purgerange(task, sender, type, type, tag));
|
||||
}
|
||||
|
||||
bool
|
||||
isc_task_purgeevent(isc_task_t *task, isc_event_t *event) {
|
||||
isc_event_t *curr_event, *next_event;
|
||||
bool found = false;
|
||||
|
||||
/*
|
||||
* Purge 'event' from a task's event queue.
|
||||
*
|
||||
* XXXRTH: WARNING: This method may be removed before beta.
|
||||
*/
|
||||
|
||||
REQUIRE(VALID_TASK(task));
|
||||
@ -633,52 +544,22 @@ isc_task_purgeevent(isc_task_t *task, isc_event_t *event) {
|
||||
*/
|
||||
|
||||
LOCK(&task->lock);
|
||||
for (curr_event = HEAD(task->events); curr_event != NULL;
|
||||
curr_event = next_event)
|
||||
{
|
||||
next_event = NEXT(curr_event, ev_link);
|
||||
if (curr_event == event && PURGE_OK(event)) {
|
||||
DEQUEUE(task->events, curr_event, ev_link);
|
||||
task->nevents--;
|
||||
break;
|
||||
}
|
||||
if (ISC_LINK_LINKED(event, ev_link)) {
|
||||
DEQUEUE(task->events, event, ev_link);
|
||||
task->nevents--;
|
||||
found = true;
|
||||
}
|
||||
UNLOCK(&task->lock);
|
||||
|
||||
if (curr_event == NULL) {
|
||||
if (!found) {
|
||||
return (false);
|
||||
}
|
||||
|
||||
isc_event_free(&curr_event);
|
||||
isc_event_free(&event);
|
||||
|
||||
return (true);
|
||||
}
|
||||
|
||||
unsigned int
|
||||
isc_task_unsendrange(isc_task_t *task, void *sender, isc_eventtype_t first,
|
||||
isc_eventtype_t last, void *tag, isc_eventlist_t *events) {
|
||||
/*
|
||||
* Remove events from a task's event queue.
|
||||
*/
|
||||
REQUIRE(VALID_TASK(task));
|
||||
|
||||
XTRACE("isc_task_unsendrange");
|
||||
|
||||
return (dequeue_events(task, sender, first, last, tag, events, false));
|
||||
}
|
||||
|
||||
unsigned int
|
||||
isc_task_unsend(isc_task_t *task, void *sender, isc_eventtype_t type, void *tag,
|
||||
isc_eventlist_t *events) {
|
||||
/*
|
||||
* Remove events from a task's event queue.
|
||||
*/
|
||||
|
||||
XTRACE("isc_task_unsend");
|
||||
|
||||
return (dequeue_events(task, sender, type, type, tag, events, false));
|
||||
}
|
||||
|
||||
isc_result_t
|
||||
isc_task_onshutdown(isc_task_t *task, isc_taskaction_t action, void *arg) {
|
||||
bool disallowed = false;
|
||||
@ -779,6 +660,16 @@ isc_task_getnetmgr(isc_task_t *task) {
|
||||
return (task->manager->netmgr);
|
||||
}
|
||||
|
||||
void
|
||||
isc_task_setquantum(isc_task_t *task, unsigned int quantum) {
|
||||
REQUIRE(VALID_TASK(task));
|
||||
|
||||
LOCK(&task->lock);
|
||||
task->quantum = (quantum > 0) ? quantum
|
||||
: task->manager->default_quantum;
|
||||
UNLOCK(&task->lock);
|
||||
}
|
||||
|
||||
/***
|
||||
*** Task Manager.
|
||||
***/
|
||||
@ -789,11 +680,13 @@ task_run(isc_task_t *task) {
|
||||
bool finished = false;
|
||||
isc_event_t *event = NULL;
|
||||
isc_result_t result = ISC_R_SUCCESS;
|
||||
uint32_t quantum;
|
||||
|
||||
REQUIRE(VALID_TASK(task));
|
||||
|
||||
LOCK(&task->lock);
|
||||
/* FIXME */
|
||||
quantum = task->quantum;
|
||||
|
||||
if (task->state != task_state_ready) {
|
||||
goto done;
|
||||
}
|
||||
@ -866,7 +759,7 @@ task_run(isc_task_t *task) {
|
||||
task->state = task_state_idle;
|
||||
}
|
||||
break;
|
||||
} else if (dispatch_count >= task->quantum) {
|
||||
} else if (dispatch_count >= quantum) {
|
||||
/*
|
||||
* Our quantum has expired, but there is more work to be
|
||||
* done. We'll requeue it to the ready queue later.
|
||||
|
@ -534,8 +534,6 @@ basic(void **state) {
|
||||
isc_task_send(task1, &event);
|
||||
}
|
||||
|
||||
(void)isc_task_purge(task3, NULL, 0, 0);
|
||||
|
||||
isc_task_detach(&task1);
|
||||
isc_task_detach(&task2);
|
||||
isc_task_detach(&task3);
|
||||
@ -959,349 +957,10 @@ post_shutdown(void **state) {
|
||||
#define TAGCNT 5
|
||||
#define NEVENTS (SENDERCNT * TYPECNT * TAGCNT)
|
||||
|
||||
static bool testrange;
|
||||
static void *purge_sender;
|
||||
static isc_eventtype_t purge_type_first;
|
||||
static isc_eventtype_t purge_type_last;
|
||||
static void *purge_tag;
|
||||
static int eventcnt;
|
||||
|
||||
atomic_bool started;
|
||||
|
||||
static void
|
||||
pg_event1(isc_task_t *task, isc_event_t *event) {
|
||||
UNUSED(task);
|
||||
|
||||
LOCK(&lock);
|
||||
while (!atomic_load(&started)) {
|
||||
WAIT(&cv, &lock);
|
||||
}
|
||||
UNLOCK(&lock);
|
||||
|
||||
isc_event_free(&event);
|
||||
}
|
||||
|
||||
static void
|
||||
pg_event2(isc_task_t *task, isc_event_t *event) {
|
||||
bool sender_match = false;
|
||||
bool type_match = false;
|
||||
bool tag_match = false;
|
||||
|
||||
UNUSED(task);
|
||||
|
||||
if ((purge_sender == NULL) || (purge_sender == event->ev_sender)) {
|
||||
sender_match = true;
|
||||
}
|
||||
|
||||
if (testrange) {
|
||||
if ((purge_type_first <= event->ev_type) &&
|
||||
(event->ev_type <= purge_type_last)) {
|
||||
type_match = true;
|
||||
}
|
||||
} else {
|
||||
if (purge_type_first == event->ev_type) {
|
||||
type_match = true;
|
||||
}
|
||||
}
|
||||
|
||||
if ((purge_tag == NULL) || (purge_tag == event->ev_tag)) {
|
||||
tag_match = true;
|
||||
}
|
||||
|
||||
if (sender_match && type_match && tag_match) {
|
||||
if ((event->ev_attributes & ISC_EVENTATTR_NOPURGE) != 0) {
|
||||
if (verbose) {
|
||||
print_message("# event %p,%d,%p "
|
||||
"matched but was not "
|
||||
"purgeable\n",
|
||||
event->ev_sender,
|
||||
(int)event->ev_type,
|
||||
event->ev_tag);
|
||||
}
|
||||
++eventcnt;
|
||||
} else if (verbose) {
|
||||
print_message("# event %p,%d,%p not purged\n",
|
||||
event->ev_sender, (int)event->ev_type,
|
||||
event->ev_tag);
|
||||
}
|
||||
} else {
|
||||
++eventcnt;
|
||||
}
|
||||
|
||||
isc_event_free(&event);
|
||||
}
|
||||
|
||||
static void
|
||||
pg_sde(isc_task_t *task, isc_event_t *event) {
|
||||
UNUSED(task);
|
||||
|
||||
LOCK(&lock);
|
||||
atomic_store(&done, true);
|
||||
SIGNAL(&cv);
|
||||
UNLOCK(&lock);
|
||||
|
||||
isc_event_free(&event);
|
||||
}
|
||||
|
||||
static void
|
||||
test_purge(int sender, int type, int tag, int exp_purged) {
|
||||
isc_result_t result;
|
||||
isc_task_t *task = NULL;
|
||||
isc_event_t *eventtab[NEVENTS];
|
||||
isc_event_t *event = NULL;
|
||||
isc_interval_t interval;
|
||||
isc_time_t now;
|
||||
int sender_cnt, type_cnt, tag_cnt, event_cnt, i;
|
||||
int purged = 0;
|
||||
|
||||
atomic_init(&started, false);
|
||||
atomic_init(&done, false);
|
||||
eventcnt = 0;
|
||||
|
||||
isc_condition_init(&cv);
|
||||
|
||||
result = isc_task_create(taskmgr, 0, &task);
|
||||
assert_int_equal(result, ISC_R_SUCCESS);
|
||||
|
||||
result = isc_task_onshutdown(task, pg_sde, NULL);
|
||||
assert_int_equal(result, ISC_R_SUCCESS);
|
||||
|
||||
/*
|
||||
* Block the task on cv.
|
||||
*/
|
||||
event = isc_event_allocate(test_mctx, (void *)1, 9999, pg_event1, NULL,
|
||||
sizeof(*event));
|
||||
|
||||
assert_non_null(event);
|
||||
isc_task_send(task, &event);
|
||||
|
||||
/*
|
||||
* Fill the task's queue with some messages with varying
|
||||
* sender, type, tag, and purgeable attribute values.
|
||||
*/
|
||||
event_cnt = 0;
|
||||
for (sender_cnt = 0; sender_cnt < SENDERCNT; ++sender_cnt) {
|
||||
for (type_cnt = 0; type_cnt < TYPECNT; ++type_cnt) {
|
||||
for (tag_cnt = 0; tag_cnt < TAGCNT; ++tag_cnt) {
|
||||
eventtab[event_cnt] = isc_event_allocate(
|
||||
test_mctx,
|
||||
&senders[sender + sender_cnt],
|
||||
(isc_eventtype_t)(type + type_cnt),
|
||||
pg_event2, NULL, sizeof(*event));
|
||||
|
||||
assert_non_null(eventtab[event_cnt]);
|
||||
|
||||
eventtab[event_cnt]->ev_tag =
|
||||
(void *)((uintptr_t)tag + tag_cnt);
|
||||
|
||||
/*
|
||||
* Mark events as non-purgeable if
|
||||
* sender, type and tag are all
|
||||
* odd-numbered. (There should be 4
|
||||
* of these out of 60 events total.)
|
||||
*/
|
||||
if (((sender_cnt % 2) != 0) &&
|
||||
((type_cnt % 2) != 0) &&
|
||||
((tag_cnt % 2) != 0)) {
|
||||
eventtab[event_cnt]->ev_attributes |=
|
||||
ISC_EVENTATTR_NOPURGE;
|
||||
}
|
||||
++event_cnt;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
for (i = 0; i < event_cnt; ++i) {
|
||||
isc_task_send(task, &eventtab[i]);
|
||||
}
|
||||
|
||||
if (testrange) {
|
||||
/*
|
||||
* We're testing isc_task_purgerange.
|
||||
*/
|
||||
purged = isc_task_purgerange(
|
||||
task, purge_sender, (isc_eventtype_t)purge_type_first,
|
||||
(isc_eventtype_t)purge_type_last, purge_tag);
|
||||
assert_int_equal(purged, exp_purged);
|
||||
} else {
|
||||
/*
|
||||
* We're testing isc_task_purge.
|
||||
*/
|
||||
if (verbose) {
|
||||
print_message("# purge events %p,%u,%p\n", purge_sender,
|
||||
purge_type_first, purge_tag);
|
||||
}
|
||||
purged = isc_task_purge(task, purge_sender,
|
||||
(isc_eventtype_t)purge_type_first,
|
||||
purge_tag);
|
||||
if (verbose) {
|
||||
print_message("# purged %d expected %d\n", purged,
|
||||
exp_purged);
|
||||
}
|
||||
|
||||
assert_int_equal(purged, exp_purged);
|
||||
}
|
||||
|
||||
/*
|
||||
* Unblock the task, allowing event processing.
|
||||
*/
|
||||
LOCK(&lock);
|
||||
atomic_store(&started, true);
|
||||
SIGNAL(&cv);
|
||||
|
||||
isc_task_shutdown(task);
|
||||
|
||||
isc_interval_set(&interval, 5, 0);
|
||||
|
||||
/*
|
||||
* Wait for shutdown processing to complete.
|
||||
*/
|
||||
while (!atomic_load(&done)) {
|
||||
result = isc_time_nowplusinterval(&now, &interval);
|
||||
assert_int_equal(result, ISC_R_SUCCESS);
|
||||
|
||||
WAITUNTIL(&cv, &lock, &now);
|
||||
}
|
||||
|
||||
UNLOCK(&lock);
|
||||
|
||||
isc_task_detach(&task);
|
||||
|
||||
assert_int_equal(eventcnt, event_cnt - exp_purged);
|
||||
}
|
||||
|
||||
/*
|
||||
* Purge test:
|
||||
* A call to isc_task_purge(task, sender, type, tag) purges all events of
|
||||
* type 'type' and with tag 'tag' not marked as unpurgeable from sender
|
||||
* from the task's " queue and returns the number of events purged.
|
||||
*/
|
||||
static void
|
||||
purge(void **state) {
|
||||
UNUSED(state);
|
||||
|
||||
/* Try purging on a specific sender. */
|
||||
if (verbose) {
|
||||
print_message("# testing purge on 2,4,8 expecting 1\n");
|
||||
}
|
||||
purge_sender = &senders[2];
|
||||
purge_type_first = 4;
|
||||
purge_type_last = 4;
|
||||
purge_tag = (void *)8;
|
||||
testrange = false;
|
||||
test_purge(1, 4, 7, 1);
|
||||
|
||||
/* Try purging on all senders. */
|
||||
if (verbose) {
|
||||
print_message("# testing purge on 0,4,8 expecting 3\n");
|
||||
}
|
||||
purge_sender = NULL;
|
||||
purge_type_first = 4;
|
||||
purge_type_last = 4;
|
||||
purge_tag = (void *)8;
|
||||
testrange = false;
|
||||
test_purge(1, 4, 7, 3);
|
||||
|
||||
/* Try purging on all senders, specified type, all tags. */
|
||||
if (verbose) {
|
||||
print_message("# testing purge on 0,4,0 expecting 15\n");
|
||||
}
|
||||
purge_sender = NULL;
|
||||
purge_type_first = 4;
|
||||
purge_type_last = 4;
|
||||
purge_tag = NULL;
|
||||
testrange = false;
|
||||
test_purge(1, 4, 7, 15);
|
||||
|
||||
/* Try purging on a specified tag, no such type. */
|
||||
if (verbose) {
|
||||
print_message("# testing purge on 0,99,8 expecting 0\n");
|
||||
}
|
||||
purge_sender = NULL;
|
||||
purge_type_first = 99;
|
||||
purge_type_last = 99;
|
||||
purge_tag = (void *)8;
|
||||
testrange = false;
|
||||
test_purge(1, 4, 7, 0);
|
||||
|
||||
/* Try purging on specified sender, type, all tags. */
|
||||
if (verbose) {
|
||||
print_message("# testing purge on 3,5,0 expecting 5\n");
|
||||
}
|
||||
purge_sender = &senders[3];
|
||||
purge_type_first = 5;
|
||||
purge_type_last = 5;
|
||||
purge_tag = NULL;
|
||||
testrange = false;
|
||||
test_purge(1, 4, 7, 5);
|
||||
}
|
||||
|
||||
/*
|
||||
* Purge range test:
|
||||
* A call to isc_event_purgerange(task, sender, first, last, tag) purges
|
||||
* all events not marked unpurgeable from sender 'sender' and of type within
|
||||
* the range 'first' to 'last' inclusive from the task's event queue and
|
||||
* returns the number of tasks purged.
|
||||
*/
|
||||
|
||||
static void
|
||||
purgerange(void **state) {
|
||||
UNUSED(state);
|
||||
|
||||
/* Now let's try some ranges. */
|
||||
/* testing purgerange on 2,4-5,8 expecting 1 */
|
||||
purge_sender = &senders[2];
|
||||
purge_type_first = 4;
|
||||
purge_type_last = 5;
|
||||
purge_tag = (void *)8;
|
||||
testrange = true;
|
||||
test_purge(1, 4, 7, 1);
|
||||
|
||||
/* Try purging on all senders. */
|
||||
if (verbose) {
|
||||
print_message("# testing purge on 0,4-5,8 expecting 5\n");
|
||||
}
|
||||
purge_sender = NULL;
|
||||
purge_type_first = 4;
|
||||
purge_type_last = 5;
|
||||
purge_tag = (void *)8;
|
||||
testrange = true;
|
||||
test_purge(1, 4, 7, 5);
|
||||
|
||||
/* Try purging on all senders, specified type, all tags. */
|
||||
if (verbose) {
|
||||
print_message("# testing purge on 0,5-6,0 expecting 28\n");
|
||||
}
|
||||
purge_sender = NULL;
|
||||
purge_type_first = 5;
|
||||
purge_type_last = 6;
|
||||
purge_tag = NULL;
|
||||
testrange = true;
|
||||
test_purge(1, 4, 7, 28);
|
||||
|
||||
/* Try purging on a specified tag, no such type. */
|
||||
if (verbose) {
|
||||
print_message("# testing purge on 0,99-101,8 expecting 0\n");
|
||||
}
|
||||
purge_sender = NULL;
|
||||
purge_type_first = 99;
|
||||
purge_type_last = 101;
|
||||
purge_tag = (void *)8;
|
||||
testrange = true;
|
||||
test_purge(1, 4, 7, 0);
|
||||
|
||||
/* Try purging on specified sender, type, all tags. */
|
||||
if (verbose) {
|
||||
print_message("# testing purge on 3,5-6,0 expecting 10\n");
|
||||
}
|
||||
purge_sender = &senders[3];
|
||||
purge_type_first = 5;
|
||||
purge_type_last = 6;
|
||||
purge_tag = NULL;
|
||||
testrange = true;
|
||||
test_purge(1, 4, 7, 10);
|
||||
}
|
||||
|
||||
/*
|
||||
* Helpers for purge event tests
|
||||
*/
|
||||
@ -1339,7 +998,7 @@ pge_sde(isc_task_t *task, isc_event_t *event) {
|
||||
}
|
||||
|
||||
static void
|
||||
try_purgeevent(bool purgeable) {
|
||||
try_purgeevent(void) {
|
||||
isc_result_t result;
|
||||
isc_task_t *task = NULL;
|
||||
bool purged;
|
||||
@ -1375,16 +1034,11 @@ try_purgeevent(bool purgeable) {
|
||||
|
||||
event2_clone = event2;
|
||||
|
||||
if (purgeable) {
|
||||
event2->ev_attributes &= ~ISC_EVENTATTR_NOPURGE;
|
||||
} else {
|
||||
event2->ev_attributes |= ISC_EVENTATTR_NOPURGE;
|
||||
}
|
||||
|
||||
isc_task_send(task, &event2);
|
||||
|
||||
purged = isc_task_purgeevent(task, event2_clone);
|
||||
assert_int_equal(purgeable, purged);
|
||||
|
||||
assert_true(purged);
|
||||
|
||||
/*
|
||||
* Unblock the task, allowing event processing.
|
||||
@ -1410,8 +1064,6 @@ try_purgeevent(bool purgeable) {
|
||||
UNLOCK(&lock);
|
||||
|
||||
isc_task_detach(&task);
|
||||
|
||||
assert_int_equal(eventcnt, (purgeable ? 0 : 1));
|
||||
}
|
||||
|
||||
/*
|
||||
@ -1425,21 +1077,7 @@ static void
|
||||
purgeevent(void **state) {
|
||||
UNUSED(state);
|
||||
|
||||
try_purgeevent(true);
|
||||
}
|
||||
|
||||
/*
|
||||
* Purge event not purgeable test:
|
||||
* When the event is not marked as purgable, a call to
|
||||
* isc_task_purgeevent(task, event) does not purge the event
|
||||
* 'event' from the task's queue and returns false.
|
||||
*/
|
||||
|
||||
static void
|
||||
purgeevent_notpurge(void **state) {
|
||||
UNUSED(state);
|
||||
|
||||
try_purgeevent(false);
|
||||
try_purgeevent();
|
||||
}
|
||||
|
||||
int
|
||||
@ -1455,11 +1093,7 @@ main(int argc, char **argv) {
|
||||
_teardown),
|
||||
cmocka_unit_test_setup_teardown(privileged_events, _setup,
|
||||
_teardown),
|
||||
cmocka_unit_test_setup_teardown(purge, _setup2, _teardown),
|
||||
cmocka_unit_test_setup_teardown(purgeevent, _setup2, _teardown),
|
||||
cmocka_unit_test_setup_teardown(purgeevent_notpurge, _setup,
|
||||
_teardown),
|
||||
cmocka_unit_test_setup_teardown(purgerange, _setup, _teardown),
|
||||
cmocka_unit_test_setup_teardown(task_shutdown, _setup4,
|
||||
_teardown),
|
||||
cmocka_unit_test_setup_teardown(task_exclusive, _setup4,
|
||||
|
@ -70,6 +70,7 @@ struct isc_timer {
|
||||
void *arg;
|
||||
unsigned int index;
|
||||
isc_time_t due;
|
||||
ISC_LIST(isc_timerevent_t) active;
|
||||
LINK(isc_timer_t) link;
|
||||
};
|
||||
|
||||
@ -189,18 +190,50 @@ deschedule(isc_timer_t *timer) {
|
||||
}
|
||||
}
|
||||
|
||||
static void
|
||||
timerevent_unlink(isc_timer_t *timer, isc_timerevent_t *event) {
|
||||
REQUIRE(ISC_LINK_LINKED(event, ev_timerlink));
|
||||
ISC_LIST_UNLINK(timer->active, event, ev_timerlink);
|
||||
}
|
||||
|
||||
static void
|
||||
timerevent_destroy(isc_event_t *event0) {
|
||||
isc_timer_t *timer = event0->ev_destroy_arg;
|
||||
isc_timerevent_t *event = (isc_timerevent_t *)event0;
|
||||
|
||||
if (ISC_LINK_LINKED(event, ev_timerlink)) {
|
||||
/* The event was unlinked via timer_purge() */
|
||||
timerevent_unlink(timer, event);
|
||||
}
|
||||
|
||||
isc_mem_put(timer->manager->mctx, event, event0->ev_size);
|
||||
isc_timer_detach(&timer);
|
||||
}
|
||||
|
||||
static void
|
||||
timer_purge(isc_timer_t *timer) {
|
||||
isc_timerevent_t *event = NULL;
|
||||
|
||||
while ((event = ISC_LIST_HEAD(timer->active)) != NULL) {
|
||||
bool purged = isc_task_purgeevent(timer->task,
|
||||
(isc_event_t *)event);
|
||||
if (!purged) {
|
||||
/*
|
||||
* The event has already been executed, but not
|
||||
* yet destroyed.
|
||||
*/
|
||||
timerevent_unlink(timer, event);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static void
|
||||
destroy(isc_timer_t *timer) {
|
||||
isc_timermgr_t *manager = timer->manager;
|
||||
|
||||
/*
|
||||
* The caller must ensure it is safe to destroy the timer.
|
||||
*/
|
||||
|
||||
LOCK(&manager->lock);
|
||||
|
||||
(void)isc_task_purgerange(timer->task, timer, ISC_TIMEREVENT_FIRSTEVENT,
|
||||
ISC_TIMEREVENT_LASTEVENT, NULL);
|
||||
timer_purge(timer);
|
||||
deschedule(timer);
|
||||
|
||||
UNLINK(manager->timers, timer, link);
|
||||
@ -248,6 +281,8 @@ isc_timer_create(isc_timermgr_t *manager, isc_task_t *task,
|
||||
isc_mutex_init(&timer->lock);
|
||||
ISC_LINK_INIT(timer, link);
|
||||
|
||||
ISC_LIST_INIT(timer->active);
|
||||
|
||||
timer->magic = TIMER_MAGIC;
|
||||
|
||||
/*
|
||||
@ -303,9 +338,7 @@ isc_timer_reset(isc_timer_t *timer, isc_timertype_t type,
|
||||
LOCK(&timer->lock);
|
||||
|
||||
if (purge) {
|
||||
(void)isc_task_purgerange(timer->task, timer,
|
||||
ISC_TIMEREVENT_FIRSTEVENT,
|
||||
ISC_TIMEREVENT_LASTEVENT, NULL);
|
||||
timer_purge(timer);
|
||||
}
|
||||
timer->type = type;
|
||||
timer->interval = *interval;
|
||||
@ -356,7 +389,6 @@ isc_timer_attach(isc_timer_t *timer, isc_timer_t **timerp) {
|
||||
void
|
||||
isc_timer_detach(isc_timer_t **timerp) {
|
||||
isc_timer_t *timer;
|
||||
|
||||
/*
|
||||
* Detach *timerp from its timer.
|
||||
*/
|
||||
@ -376,20 +408,22 @@ static void
|
||||
post_event(isc_timermgr_t *manager, isc_timer_t *timer, isc_eventtype_t type) {
|
||||
isc_timerevent_t *event;
|
||||
XTRACEID("posting", timer);
|
||||
/*
|
||||
* XXX We could preallocate this event.
|
||||
*/
|
||||
|
||||
event = (isc_timerevent_t *)isc_event_allocate(
|
||||
manager->mctx, timer, type, timer->action, timer->arg,
|
||||
sizeof(*event));
|
||||
|
||||
if (event != NULL) {
|
||||
event->due = timer->due;
|
||||
isc_task_send(timer->task, ISC_EVENT_PTR(&event));
|
||||
} else {
|
||||
UNEXPECTED_ERROR(__FILE__, __LINE__, "%s",
|
||||
"couldn't allocate event");
|
||||
}
|
||||
ISC_LINK_INIT(event, ev_timerlink);
|
||||
((isc_event_t *)event)->ev_destroy = timerevent_destroy;
|
||||
|
||||
isc_timer_attach(timer, &(isc_timer_t *){ NULL });
|
||||
((isc_event_t *)event)->ev_destroy_arg = timer;
|
||||
|
||||
event->due = timer->due;
|
||||
|
||||
ISC_LIST_APPEND(timer->active, event, ev_timerlink);
|
||||
|
||||
isc_task_send(timer->task, ISC_EVENT_PTR(&event));
|
||||
}
|
||||
|
||||
static void
|
||||
|
@ -38,6 +38,3 @@
|
||||
*/
|
||||
|
||||
#define ISCCC_EVENT_CCMSG (ISC_EVENTCLASS_ISCCC + 0)
|
||||
|
||||
#define ISCCC_EVENT_FIRSTEVENT (ISC_EVENTCLASS_ISCCC + 0)
|
||||
#define ISCCC_EVENT_LASTEVENT (ISC_EVENTCLASS_ISCCC + 65535)
|
||||
|
Loading…
x
Reference in New Issue
Block a user