From 0bfe3ca51ebddbf2cc099fa34f359bd1ac4f65e6 Mon Sep 17 00:00:00 2001 From: Anthony Liguori Date: Thu, 14 May 2009 19:29:53 +0100 Subject: Constructor support Allow devices/drivers to register themselves via constructors. Destructors are not needed (can be registered from a constructor) and "priority" has been renamed and changed to an enum for clarity. Signed-off-by: Paul Brook Signed-off-by: Anthony Liguori --- module.c | 91 ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 91 insertions(+) create mode 100644 module.c (limited to 'module.c') diff --git a/module.c b/module.c new file mode 100644 index 000000000..113eeefc6 --- /dev/null +++ b/module.c @@ -0,0 +1,91 @@ +/* + * QEMU Module Infrastructure + * + * Copyright IBM, Corp. 2009 + * + * Authors: + * Anthony Liguori + * + * This work is licensed under the terms of the GNU GPL, version 2. See + * the COPYING file in the top-level directory. + * + */ + +#include "qemu-common.h" +#include "sys-queue.h" +#include "module.h" + +typedef struct ModuleEntry +{ + module_init_type type; + void (*init)(void); + TAILQ_ENTRY(ModuleEntry) node; +} ModuleEntry; + +typedef struct ModuleTypeList +{ + module_init_type type; + TAILQ_HEAD(, ModuleEntry) entry_list; + TAILQ_ENTRY(ModuleTypeList) node; +} ModuleTypeList; + +static TAILQ_HEAD(, ModuleTypeList) init_type_list; + +static ModuleTypeList *find_type_or_alloc(module_init_type type, int alloc) +{ + ModuleTypeList *n; + + TAILQ_FOREACH(n, &init_type_list, node) { + if (type >= n->type) + break; + } + + if (!n || n->type != type) { + ModuleTypeList *o; + + if (!alloc) + return NULL; + + o = qemu_mallocz(sizeof(*o)); + o->type = type; + TAILQ_INIT(&o->entry_list); + + if (n) { + TAILQ_INSERT_AFTER(&init_type_list, n, o, node); + } else { + TAILQ_INSERT_HEAD(&init_type_list, o, node); + } + + n = o; + } + + return n; +} + +void register_module_init(void (*fn)(void), module_init_type type) +{ + ModuleEntry *e; + ModuleTypeList *l; + + e = qemu_mallocz(sizeof(*e)); + e->init = fn; + + l = find_type_or_alloc(type, 1); + + TAILQ_INSERT_TAIL(&l->entry_list, e, node); +} + +void module_call_init(module_init_type type) +{ + ModuleTypeList *l; + ModuleEntry *e; + + l = find_type_or_alloc(type, 0); + if (!l) { + return; + } + + TAILQ_FOREACH(e, &l->entry_list, node) { + e->init(); + } +} -- cgit v1.2.3