Rework the eventhandler locking for hopefully the last time. The scheme
used popped into my head during my morning commute a few weeks ago, but it is also very similar (though a bit simpler) to a patch that mini@ developed a while ago. Basically, each eventhandler list has a mutex and a run count. During an eventhandler invocation, the mutex is held while we traverse the list but is dropped while we execute actual handlers. Also, a runcount counter is incremented at the start of an invocation and decremented at the end of an invocation. Adding to the list is not a big deal since the reference of a thread currently executing the handlers remains valid across an add operation. Whether or not new handlers are executed by threads currently executing the handlers for a given list is indeterminate however. The harder case is when a handler is removed from the list. If the runcount is zero, the handler is simply removed from the list directly. If the runcount is not zero, then another thread is currently executing the handlers of this list, so the priority of this handler is set to a magic value (currently -1) to mark it as dead. Dead handlers are not executed during an invocation. If the runcount is zero after it is decremented at the end of an invocation, then a new eventhandler_prune_list() function is called to remove dead handlers from the list. Additional minor notes: - All the common parts of EVENTHANDLER_INVOKE() and EVENTHANDLER_FAST_INVOKE() have been merged into a common _EVENTHANDLER_INVOKE() macro to reduce duplication and ease maintenance. - KTR logging for eventhandlers is now available via the KTR_EVH mask. - The global eventhander_mutex is no longer recursive. Tested by: scottl (SMP i386)
This commit is contained in:
parent
50bb124650
commit
ed498389de
@ -48,6 +48,8 @@ struct eventhandler_entry_generic
|
|||||||
void (* func)(void);
|
void (* func)(void);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static struct eventhandler_list *_eventhandler_find_list(char *name);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Initialize the eventhandler mutex and list.
|
* Initialize the eventhandler mutex and list.
|
||||||
*/
|
*/
|
||||||
@ -55,8 +57,8 @@ static void
|
|||||||
eventhandler_init(void *dummy __unused)
|
eventhandler_init(void *dummy __unused)
|
||||||
{
|
{
|
||||||
TAILQ_INIT(&eventhandler_lists);
|
TAILQ_INIT(&eventhandler_lists);
|
||||||
mtx_init(&eventhandler_mutex, "eventhandler", NULL, MTX_DEF | MTX_RECURSE);
|
mtx_init(&eventhandler_mutex, "eventhandler", NULL, MTX_DEF);
|
||||||
eventhandler_lists_initted = 1;
|
atomic_store_rel_int(&eventhandler_lists_initted, 1);
|
||||||
}
|
}
|
||||||
SYSINIT(eventhandlers, SI_SUB_EVENTHANDLER, SI_ORDER_FIRST, eventhandler_init,
|
SYSINIT(eventhandlers, SI_SUB_EVENTHANDLER, SI_ORDER_FIRST, eventhandler_init,
|
||||||
NULL)
|
NULL)
|
||||||
@ -69,6 +71,7 @@ eventhandler_tag
|
|||||||
eventhandler_register(struct eventhandler_list *list, char *name,
|
eventhandler_register(struct eventhandler_list *list, char *name,
|
||||||
void *func, void *arg, int priority)
|
void *func, void *arg, int priority)
|
||||||
{
|
{
|
||||||
|
struct eventhandler_list *new_list;
|
||||||
struct eventhandler_entry_generic *eg;
|
struct eventhandler_entry_generic *eg;
|
||||||
struct eventhandler_entry *ep;
|
struct eventhandler_entry *ep;
|
||||||
|
|
||||||
@ -80,51 +83,62 @@ eventhandler_register(struct eventhandler_list *list, char *name,
|
|||||||
/* Do we need to find/create the (slow) list? */
|
/* Do we need to find/create the (slow) list? */
|
||||||
if (list == NULL) {
|
if (list == NULL) {
|
||||||
/* look for a matching, existing list */
|
/* look for a matching, existing list */
|
||||||
list = eventhandler_find_list(name);
|
list = _eventhandler_find_list(name);
|
||||||
|
|
||||||
/* Do we need to create the list? */
|
/* Do we need to create the list? */
|
||||||
if (list == NULL) {
|
if (list == NULL) {
|
||||||
if ((list = malloc(sizeof(struct eventhandler_list) + strlen(name)
|
mtx_unlock(&eventhandler_mutex);
|
||||||
+ 1, M_EVENTHANDLER, M_NOWAIT)) == NULL) {
|
|
||||||
mtx_unlock(&eventhandler_mutex);
|
new_list = malloc(sizeof(struct eventhandler_list) +
|
||||||
return(NULL);
|
strlen(name) + 1, M_EVENTHANDLER, M_WAITOK);
|
||||||
|
|
||||||
|
/* If someone else created it already, then use that one. */
|
||||||
|
mtx_lock(&eventhandler_mutex);
|
||||||
|
list = _eventhandler_find_list(name);
|
||||||
|
if (list != NULL) {
|
||||||
|
free(new_list, M_EVENTHANDLER);
|
||||||
|
} else {
|
||||||
|
CTR2(KTR_EVH, "%s: creating list \"%s\"", __func__, name);
|
||||||
|
list = new_list;
|
||||||
|
list->el_flags = 0;
|
||||||
|
list->el_runcount = 0;
|
||||||
|
bzero(&list->el_lock, sizeof(list->el_lock));
|
||||||
|
list->el_name = (char *)list + sizeof(struct eventhandler_list);
|
||||||
|
strcpy(list->el_name, name);
|
||||||
|
TAILQ_INSERT_HEAD(&eventhandler_lists, list, el_link);
|
||||||
}
|
}
|
||||||
list->el_flags = 0;
|
|
||||||
bzero(&list->el_lock, sizeof(list->el_lock));
|
|
||||||
list->el_name = (char *)list + sizeof(struct eventhandler_list);
|
|
||||||
strcpy(list->el_name, name);
|
|
||||||
TAILQ_INSERT_HEAD(&eventhandler_lists, list, el_link);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (!(list->el_flags & EHE_INITTED)) {
|
if (!(list->el_flags & EHL_INITTED)) {
|
||||||
TAILQ_INIT(&list->el_entries);
|
TAILQ_INIT(&list->el_entries);
|
||||||
sx_init(&list->el_lock, name);
|
mtx_init(&list->el_lock, name, "eventhandler list", MTX_DEF);
|
||||||
list->el_flags = EHE_INITTED;
|
atomic_store_rel_int(&list->el_flags, EHL_INITTED);
|
||||||
}
|
}
|
||||||
mtx_unlock(&eventhandler_mutex);
|
mtx_unlock(&eventhandler_mutex);
|
||||||
|
|
||||||
/* allocate an entry for this handler, populate it */
|
/* allocate an entry for this handler, populate it */
|
||||||
if ((eg = malloc(sizeof(struct eventhandler_entry_generic),
|
eg = malloc(sizeof(struct eventhandler_entry_generic), M_EVENTHANDLER,
|
||||||
M_EVENTHANDLER, M_NOWAIT)) == NULL) {
|
M_WAITOK | M_ZERO);
|
||||||
return(NULL);
|
|
||||||
}
|
|
||||||
eg->func = func;
|
eg->func = func;
|
||||||
eg->ee.ee_arg = arg;
|
eg->ee.ee_arg = arg;
|
||||||
eg->ee.ee_priority = priority;
|
eg->ee.ee_priority = priority;
|
||||||
|
KASSERT(priority != EHE_DEAD_PRIORITY,
|
||||||
|
("%s: handler for %s registered with dead priority", __func__, name));
|
||||||
|
|
||||||
/* sort it into the list */
|
/* sort it into the list */
|
||||||
EHE_LOCK(list);
|
CTR4(KTR_EVH, "%s: adding item %p (function %p) to \"%s\"", __func__, eg,
|
||||||
for (ep = TAILQ_FIRST(&list->el_entries);
|
func, name);
|
||||||
ep != NULL;
|
EHL_LOCK(list);
|
||||||
ep = TAILQ_NEXT(ep, ee_link)) {
|
TAILQ_FOREACH(ep, &list->el_entries, ee_link) {
|
||||||
if (eg->ee.ee_priority < ep->ee_priority) {
|
if (ep->ee_priority != EHE_DEAD_PRIORITY &&
|
||||||
|
eg->ee.ee_priority < ep->ee_priority) {
|
||||||
TAILQ_INSERT_BEFORE(ep, &eg->ee, ee_link);
|
TAILQ_INSERT_BEFORE(ep, &eg->ee, ee_link);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (ep == NULL)
|
if (ep == NULL)
|
||||||
TAILQ_INSERT_TAIL(&list->el_entries, &eg->ee, ee_link);
|
TAILQ_INSERT_TAIL(&list->el_entries, &eg->ee, ee_link);
|
||||||
EHE_UNLOCK(list);
|
EHL_UNLOCK(list);
|
||||||
return(&eg->ee);
|
return(&eg->ee);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -133,23 +147,59 @@ eventhandler_deregister(struct eventhandler_list *list, eventhandler_tag tag)
|
|||||||
{
|
{
|
||||||
struct eventhandler_entry *ep = tag;
|
struct eventhandler_entry *ep = tag;
|
||||||
|
|
||||||
/* XXX insert diagnostic check here? */
|
EHL_LOCK_ASSERT(list, MA_OWNED);
|
||||||
EHE_LOCK(list);
|
|
||||||
if (ep != NULL) {
|
if (ep != NULL) {
|
||||||
/* remove just this entry */
|
/* remove just this entry */
|
||||||
TAILQ_REMOVE(&list->el_entries, ep, ee_link);
|
if (list->el_runcount == 0) {
|
||||||
free(ep, M_EVENTHANDLER);
|
CTR3(KTR_EVH, "%s: removing item %p from \"%s\"", __func__, ep,
|
||||||
} else {
|
list->el_name);
|
||||||
/* remove entire list */
|
|
||||||
while (!TAILQ_EMPTY(&list->el_entries)) {
|
|
||||||
ep = TAILQ_FIRST(&list->el_entries);
|
|
||||||
TAILQ_REMOVE(&list->el_entries, ep, ee_link);
|
TAILQ_REMOVE(&list->el_entries, ep, ee_link);
|
||||||
free(ep, M_EVENTHANDLER);
|
free(ep, M_EVENTHANDLER);
|
||||||
|
} else {
|
||||||
|
CTR3(KTR_EVH, "%s: marking item %p from \"%s\" as dead", __func__,
|
||||||
|
ep, list->el_name);
|
||||||
|
ep->ee_priority = EHE_DEAD_PRIORITY;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
/* remove entire list */
|
||||||
|
if (list->el_runcount == 0) {
|
||||||
|
CTR2(KTR_EVH, "%s: removing all items from \"%s\"", __func__,
|
||||||
|
list->el_name);
|
||||||
|
TAILQ_REMOVE(&list->el_entries, ep, ee_link);
|
||||||
|
while (!TAILQ_EMPTY(&list->el_entries)) {
|
||||||
|
ep = TAILQ_FIRST(&list->el_entries);
|
||||||
|
TAILQ_REMOVE(&list->el_entries, ep, ee_link);
|
||||||
|
free(ep, M_EVENTHANDLER);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
CTR2(KTR_EVH, "%s: marking all items from \"%s\" as dead",
|
||||||
|
__func__, list->el_name);
|
||||||
|
TAILQ_FOREACH(ep, &list->el_entries, ee_link)
|
||||||
|
ep->ee_priority = EHE_DEAD_PRIORITY;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
EHE_UNLOCK(list);
|
EHL_UNLOCK(list);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Internal version for use when eventhandler list is already locked.
|
||||||
|
*/
|
||||||
|
static struct eventhandler_list *
|
||||||
|
_eventhandler_find_list(char *name)
|
||||||
|
{
|
||||||
|
struct eventhandler_list *list;
|
||||||
|
|
||||||
|
mtx_assert(&eventhandler_mutex, MA_OWNED);
|
||||||
|
TAILQ_FOREACH(list, &eventhandler_lists, el_link) {
|
||||||
|
if (!strcmp(name, list->el_name))
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return (list);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Lookup a "slow" list by name. Returns with the list locked.
|
||||||
|
*/
|
||||||
struct eventhandler_list *
|
struct eventhandler_list *
|
||||||
eventhandler_find_list(char *name)
|
eventhandler_find_list(char *name)
|
||||||
{
|
{
|
||||||
@ -160,14 +210,31 @@ eventhandler_find_list(char *name)
|
|||||||
|
|
||||||
/* scan looking for the requested list */
|
/* scan looking for the requested list */
|
||||||
mtx_lock(&eventhandler_mutex);
|
mtx_lock(&eventhandler_mutex);
|
||||||
for (list = TAILQ_FIRST(&eventhandler_lists);
|
list = _eventhandler_find_list(name);
|
||||||
list != NULL;
|
if (list != NULL)
|
||||||
list = TAILQ_NEXT(list, el_link)) {
|
EHL_LOCK(list);
|
||||||
if (!strcmp(name, list->el_name))
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
mtx_unlock(&eventhandler_mutex);
|
mtx_unlock(&eventhandler_mutex);
|
||||||
|
|
||||||
return(list);
|
return(list);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Prune "dead" entries from an eventhandler list.
|
||||||
|
*/
|
||||||
|
void
|
||||||
|
eventhandler_prune_list(struct eventhandler_list *list)
|
||||||
|
{
|
||||||
|
struct eventhandler_entry *ep, *en;
|
||||||
|
|
||||||
|
CTR2(KTR_EVH, "%s: pruning list \"%s\"", __func__, list->el_name);
|
||||||
|
EHL_LOCK_ASSERT(list, MA_OWNED);
|
||||||
|
ep = TAILQ_FIRST(&list->el_entries);
|
||||||
|
while (ep != NULL) {
|
||||||
|
en = TAILQ_NEXT(ep, ee_link);
|
||||||
|
if (ep->ee_priority == EHE_DEAD_PRIORITY) {
|
||||||
|
TAILQ_REMOVE(&list->el_entries, ep, ee_link);
|
||||||
|
free(ep, M_EVENTHANDLER);
|
||||||
|
}
|
||||||
|
ep = en;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -30,28 +30,65 @@
|
|||||||
#define SYS_EVENTHANDLER_H
|
#define SYS_EVENTHANDLER_H
|
||||||
|
|
||||||
#include <sys/lock.h>
|
#include <sys/lock.h>
|
||||||
#include <sys/sx.h>
|
#include <sys/ktr.h>
|
||||||
|
#include <sys/mutex.h>
|
||||||
#include <sys/queue.h>
|
#include <sys/queue.h>
|
||||||
|
|
||||||
struct eventhandler_entry {
|
struct eventhandler_entry {
|
||||||
TAILQ_ENTRY(eventhandler_entry) ee_link;
|
TAILQ_ENTRY(eventhandler_entry) ee_link;
|
||||||
int ee_priority;
|
int ee_priority;
|
||||||
|
#define EHE_DEAD_PRIORITY (-1)
|
||||||
void *ee_arg;
|
void *ee_arg;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct eventhandler_list {
|
struct eventhandler_list {
|
||||||
char *el_name;
|
char *el_name;
|
||||||
int el_flags;
|
int el_flags;
|
||||||
#define EHE_INITTED (1<<0)
|
#define EHL_INITTED (1<<0)
|
||||||
struct sx el_lock;
|
u_int el_runcount;
|
||||||
|
struct mtx el_lock;
|
||||||
TAILQ_ENTRY(eventhandler_list) el_link;
|
TAILQ_ENTRY(eventhandler_list) el_link;
|
||||||
TAILQ_HEAD(,eventhandler_entry) el_entries;
|
TAILQ_HEAD(,eventhandler_entry) el_entries;
|
||||||
};
|
};
|
||||||
|
|
||||||
typedef struct eventhandler_entry *eventhandler_tag;
|
typedef struct eventhandler_entry *eventhandler_tag;
|
||||||
|
|
||||||
#define EHE_LOCK(p) sx_xlock(&(p)->el_lock)
|
#define EHL_LOCK(p) mtx_lock(&(p)->el_lock)
|
||||||
#define EHE_UNLOCK(p) sx_xunlock(&(p)->el_lock)
|
#define EHL_UNLOCK(p) mtx_unlock(&(p)->el_lock)
|
||||||
|
#define EHL_LOCK_ASSERT(p, x) mtx_assert(&(p)->el_lock, x)
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Macro to invoke the handlers for a given event.
|
||||||
|
*/
|
||||||
|
#define _EVENTHANDLER_INVOKE(name, list, ...) do { \
|
||||||
|
struct eventhandler_entry *_ep; \
|
||||||
|
struct eventhandler_entry_ ## name *_t; \
|
||||||
|
\
|
||||||
|
KASSERT((list)->el_flags & EHL_INITTED, \
|
||||||
|
("eventhandler_invoke: running non-inited list")); \
|
||||||
|
EHL_LOCK_ASSERT((list), MA_OWNED); \
|
||||||
|
(list)->el_runcount++; \
|
||||||
|
KASSERT((list)->el_runcount > 0, \
|
||||||
|
("eventhandler_invoke: runcount overflow")); \
|
||||||
|
CTR0(KTR_EVH, "eventhandler_invoke(\"" __STRING(name) "\")"); \
|
||||||
|
TAILQ_FOREACH(_ep, &((list)->el_entries), ee_link) { \
|
||||||
|
if (_ep->ee_priority != EHE_DEAD_PRIORITY) { \
|
||||||
|
EHL_UNLOCK((list)); \
|
||||||
|
_t = (struct eventhandler_entry_ ## name *)_ep; \
|
||||||
|
CTR1(KTR_EVH, "eventhandler_invoke: executing %p", \
|
||||||
|
(void *)_t->eh_func); \
|
||||||
|
_t->eh_func(_ep->ee_arg , ## __VA_ARGS__); \
|
||||||
|
EHL_LOCK((list)); \
|
||||||
|
} \
|
||||||
|
} \
|
||||||
|
KASSERT((list)->el_runcount > 0, \
|
||||||
|
("eventhandler_invoke: runcount underflow")); \
|
||||||
|
(list)->el_runcount--; \
|
||||||
|
if ((list)->el_runcount == 0) \
|
||||||
|
eventhandler_prune_list(list); \
|
||||||
|
EHL_UNLOCK((list)); \
|
||||||
|
} while (0)
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Fast handler lists require the eventhandler list be present
|
* Fast handler lists require the eventhandler list be present
|
||||||
@ -75,22 +112,12 @@ struct __hack
|
|||||||
struct eventhandler_list Xeventhandler_list_ ## name = { #name }; \
|
struct eventhandler_list Xeventhandler_list_ ## name = { #name }; \
|
||||||
struct __hack
|
struct __hack
|
||||||
|
|
||||||
#define EVENTHANDLER_FAST_INVOKE(name, ...) \
|
#define EVENTHANDLER_FAST_INVOKE(name, ...) do { \
|
||||||
do { \
|
|
||||||
struct eventhandler_list *_el = &Xeventhandler_list_ ## name ; \
|
struct eventhandler_list *_el = &Xeventhandler_list_ ## name ; \
|
||||||
struct eventhandler_entry *_ep, *_en; \
|
|
||||||
struct eventhandler_entry_ ## name *_t; \
|
|
||||||
\
|
\
|
||||||
if (_el->el_flags & EHE_INITTED) { \
|
if (_el->el_flags & EHL_INITTED) { \
|
||||||
EHE_LOCK(_el); \
|
EHL_LOCK(_el); \
|
||||||
_ep = TAILQ_FIRST(&(_el->el_entries)); \
|
_EVENTHANDLER_INVOKE(name, _el , ## __VA_ARGS__); \
|
||||||
while (_ep != NULL) { \
|
|
||||||
_en = TAILQ_NEXT(_ep, ee_link); \
|
|
||||||
_t = (struct eventhandler_entry_ ## name *)_ep; \
|
|
||||||
_t->eh_func(_ep->ee_arg , __VA_ARGS__); \
|
|
||||||
_ep = _en; \
|
|
||||||
} \
|
|
||||||
EHE_UNLOCK(_el); \
|
|
||||||
} \
|
} \
|
||||||
} while (0)
|
} while (0)
|
||||||
|
|
||||||
@ -98,8 +125,14 @@ do { \
|
|||||||
eventhandler_register(&Xeventhandler_list_ ## name, \
|
eventhandler_register(&Xeventhandler_list_ ## name, \
|
||||||
#name, func, arg, priority)
|
#name, func, arg, priority)
|
||||||
|
|
||||||
#define EVENTHANDLER_FAST_DEREGISTER(name, tag) \
|
#define EVENTHANDLER_FAST_DEREGISTER(name, tag) do { \
|
||||||
eventhandler_deregister(&Xeventhandler_list_ ## name, tag)
|
struct eventhandler_list *_el = &Xeventhandler_list_ ## name ; \
|
||||||
|
\
|
||||||
|
KASSERT(_el->el_flags & EHL_INITTED, \
|
||||||
|
("eventhandler_fast_deregister on un-inited list %s", ## name)); \
|
||||||
|
EHL_LOCK(_el); \
|
||||||
|
eventhandler_deregister(_el, tag); \
|
||||||
|
} while (0)
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Slow handlers are entirely dynamic; lists are created
|
* Slow handlers are entirely dynamic; lists are created
|
||||||
@ -119,21 +152,9 @@ struct __hack
|
|||||||
#define EVENTHANDLER_INVOKE(name, ...) \
|
#define EVENTHANDLER_INVOKE(name, ...) \
|
||||||
do { \
|
do { \
|
||||||
struct eventhandler_list *_el; \
|
struct eventhandler_list *_el; \
|
||||||
struct eventhandler_entry *_ep, *_en; \
|
|
||||||
struct eventhandler_entry_ ## name *_t; \
|
|
||||||
\
|
\
|
||||||
if (((_el = eventhandler_find_list(#name)) != NULL) && \
|
if ((_el = eventhandler_find_list(#name)) != NULL) \
|
||||||
(_el->el_flags & EHE_INITTED)) { \
|
_EVENTHANDLER_INVOKE(name, _el , ## __VA_ARGS__); \
|
||||||
EHE_LOCK(_el); \
|
|
||||||
_ep = TAILQ_FIRST(&(_el->el_entries)); \
|
|
||||||
while (_ep != NULL) { \
|
|
||||||
_en = TAILQ_NEXT(_ep, ee_link); \
|
|
||||||
_t = (struct eventhandler_entry_ ## name *)_ep; \
|
|
||||||
_t->eh_func(_ep->ee_arg , __VA_ARGS__); \
|
|
||||||
_ep = _en; \
|
|
||||||
} \
|
|
||||||
EHE_UNLOCK(_el); \
|
|
||||||
} \
|
|
||||||
} while (0)
|
} while (0)
|
||||||
|
|
||||||
#define EVENTHANDLER_REGISTER(name, func, arg, priority) \
|
#define EVENTHANDLER_REGISTER(name, func, arg, priority) \
|
||||||
@ -148,14 +169,12 @@ do { \
|
|||||||
} while(0)
|
} while(0)
|
||||||
|
|
||||||
|
|
||||||
extern eventhandler_tag eventhandler_register(struct eventhandler_list *list,
|
eventhandler_tag eventhandler_register(struct eventhandler_list *list,
|
||||||
char *name,
|
char *name, void *func, void *arg, int priority);
|
||||||
void *func,
|
void eventhandler_deregister(struct eventhandler_list *list,
|
||||||
void *arg,
|
eventhandler_tag tag);
|
||||||
int priority);
|
struct eventhandler_list *eventhandler_find_list(char *name);
|
||||||
extern void eventhandler_deregister(struct eventhandler_list *list,
|
void eventhandler_prune_list(struct eventhandler_list *list);
|
||||||
eventhandler_tag tag);
|
|
||||||
extern struct eventhandler_list *eventhandler_find_list(char *name);
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Standard system event queues.
|
* Standard system event queues.
|
||||||
|
Loading…
x
Reference in New Issue
Block a user