2009-05-14 18:29:53 +00:00
|
|
|
/*
|
|
|
|
* QEMU Module Infrastructure
|
|
|
|
*
|
|
|
|
* Copyright IBM, Corp. 2009
|
|
|
|
*
|
|
|
|
* Authors:
|
|
|
|
* Anthony Liguori <aliguori@us.ibm.com>
|
|
|
|
*
|
|
|
|
* This work is licensed under the terms of the GNU GPL, version 2. See
|
|
|
|
* the COPYING file in the top-level directory.
|
|
|
|
*
|
2012-01-13 16:44:23 +00:00
|
|
|
* Contributions after 2012-01-13 are licensed under the terms of the
|
|
|
|
* GNU GPL, version 2 or (at your option) any later version.
|
2009-05-14 18:29:53 +00:00
|
|
|
*/
|
|
|
|
|
2016-01-29 17:49:55 +00:00
|
|
|
#include "qemu/osdep.h"
|
2014-02-25 16:36:55 +00:00
|
|
|
#ifdef CONFIG_MODULES
|
2014-02-10 06:48:57 +00:00
|
|
|
#include <gmodule.h>
|
2014-02-25 16:36:55 +00:00
|
|
|
#endif
|
2012-12-17 17:20:00 +00:00
|
|
|
#include "qemu/queue.h"
|
|
|
|
#include "qemu/module.h"
|
2020-08-18 10:00:18 +00:00
|
|
|
#include "qemu/cutils.h"
|
2021-06-24 10:38:18 +00:00
|
|
|
#include "qemu/config-file.h"
|
2020-03-10 14:58:06 +00:00
|
|
|
#ifdef CONFIG_MODULE_UPGRADES
|
|
|
|
#include "qemu-version.h"
|
|
|
|
#endif
|
2009-05-14 18:29:53 +00:00
|
|
|
|
|
|
|
typedef struct ModuleEntry
|
|
|
|
{
|
|
|
|
void (*init)(void);
|
2009-09-12 07:36:22 +00:00
|
|
|
QTAILQ_ENTRY(ModuleEntry) node;
|
2014-02-10 06:48:57 +00:00
|
|
|
module_init_type type;
|
2009-05-14 18:29:53 +00:00
|
|
|
} ModuleEntry;
|
|
|
|
|
2009-09-12 07:36:22 +00:00
|
|
|
typedef QTAILQ_HEAD(, ModuleEntry) ModuleTypeList;
|
2009-05-14 18:29:53 +00:00
|
|
|
|
2009-05-14 22:57:31 +00:00
|
|
|
static ModuleTypeList init_type_list[MODULE_INIT_MAX];
|
2020-02-20 04:10:59 +00:00
|
|
|
static bool modules_init_done[MODULE_INIT_MAX];
|
2009-05-14 18:29:53 +00:00
|
|
|
|
2014-02-10 06:48:57 +00:00
|
|
|
static ModuleTypeList dso_init_list;
|
|
|
|
|
|
|
|
static void init_lists(void)
|
2009-05-14 18:29:53 +00:00
|
|
|
{
|
2009-05-14 22:57:31 +00:00
|
|
|
static int inited;
|
|
|
|
int i;
|
2009-05-14 18:29:53 +00:00
|
|
|
|
2009-05-14 22:57:31 +00:00
|
|
|
if (inited) {
|
|
|
|
return;
|
2009-05-14 18:29:53 +00:00
|
|
|
}
|
|
|
|
|
2009-05-14 22:57:31 +00:00
|
|
|
for (i = 0; i < MODULE_INIT_MAX; i++) {
|
2009-09-12 07:36:22 +00:00
|
|
|
QTAILQ_INIT(&init_type_list[i]);
|
2009-05-14 22:57:31 +00:00
|
|
|
}
|
2009-05-14 18:29:53 +00:00
|
|
|
|
2014-02-10 06:48:57 +00:00
|
|
|
QTAILQ_INIT(&dso_init_list);
|
|
|
|
|
2009-05-14 22:57:31 +00:00
|
|
|
inited = 1;
|
|
|
|
}
|
2009-05-14 18:29:53 +00:00
|
|
|
|
|
|
|
|
2009-05-14 22:57:31 +00:00
|
|
|
static ModuleTypeList *find_type(module_init_type type)
|
|
|
|
{
|
2014-02-10 06:48:57 +00:00
|
|
|
init_lists();
|
2009-05-14 22:57:31 +00:00
|
|
|
|
2016-06-13 21:57:58 +00:00
|
|
|
return &init_type_list[type];
|
2009-05-14 18:29:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void register_module_init(void (*fn)(void), module_init_type type)
|
|
|
|
{
|
|
|
|
ModuleEntry *e;
|
|
|
|
ModuleTypeList *l;
|
|
|
|
|
2011-08-21 03:09:37 +00:00
|
|
|
e = g_malloc0(sizeof(*e));
|
2009-05-14 18:29:53 +00:00
|
|
|
e->init = fn;
|
2014-02-10 06:48:57 +00:00
|
|
|
e->type = type;
|
2009-05-14 18:29:53 +00:00
|
|
|
|
2009-05-14 22:57:31 +00:00
|
|
|
l = find_type(type);
|
2009-05-14 18:29:53 +00:00
|
|
|
|
2009-09-12 07:36:22 +00:00
|
|
|
QTAILQ_INSERT_TAIL(l, e, node);
|
2009-05-14 18:29:53 +00:00
|
|
|
}
|
|
|
|
|
2014-02-10 06:48:57 +00:00
|
|
|
void register_dso_module_init(void (*fn)(void), module_init_type type)
|
|
|
|
{
|
|
|
|
ModuleEntry *e;
|
|
|
|
|
|
|
|
init_lists();
|
|
|
|
|
|
|
|
e = g_malloc0(sizeof(*e));
|
|
|
|
e->init = fn;
|
|
|
|
e->type = type;
|
|
|
|
|
|
|
|
QTAILQ_INSERT_TAIL(&dso_init_list, e, node);
|
|
|
|
}
|
|
|
|
|
2009-05-14 18:29:53 +00:00
|
|
|
void module_call_init(module_init_type type)
|
|
|
|
{
|
|
|
|
ModuleTypeList *l;
|
|
|
|
ModuleEntry *e;
|
|
|
|
|
2020-02-20 04:10:59 +00:00
|
|
|
if (modules_init_done[type]) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-05-14 22:57:31 +00:00
|
|
|
l = find_type(type);
|
2009-05-14 18:29:53 +00:00
|
|
|
|
2009-09-12 07:36:22 +00:00
|
|
|
QTAILQ_FOREACH(e, l, node) {
|
2009-05-14 18:29:53 +00:00
|
|
|
e->init();
|
|
|
|
}
|
2020-02-20 04:10:59 +00:00
|
|
|
|
|
|
|
modules_init_done[type] = true;
|
2009-05-14 18:29:53 +00:00
|
|
|
}
|
2014-02-10 06:48:57 +00:00
|
|
|
|
|
|
|
#ifdef CONFIG_MODULES
|
2021-06-24 10:38:05 +00:00
|
|
|
|
|
|
|
static const QemuModinfo module_info_stub[] = { {
|
|
|
|
/* end of list */
|
|
|
|
} };
|
|
|
|
static const QemuModinfo *module_info = module_info_stub;
|
|
|
|
|
|
|
|
void module_init_info(const QemuModinfo *info)
|
|
|
|
{
|
|
|
|
module_info = info;
|
|
|
|
}
|
|
|
|
|
2020-10-19 07:52:20 +00:00
|
|
|
static int module_load_file(const char *fname, bool mayfail, bool export_symbols)
|
2014-02-10 06:48:57 +00:00
|
|
|
{
|
|
|
|
GModule *g_module;
|
|
|
|
void (*sym)(void);
|
2020-08-04 16:14:26 +00:00
|
|
|
const char *dsosuf = CONFIG_HOST_DSOSUF;
|
2014-02-10 06:48:57 +00:00
|
|
|
int len = strlen(fname);
|
|
|
|
int suf_len = strlen(dsosuf);
|
|
|
|
ModuleEntry *e, *next;
|
2020-10-19 07:52:20 +00:00
|
|
|
int ret, flags;
|
2014-02-10 06:48:57 +00:00
|
|
|
|
|
|
|
if (len <= suf_len || strcmp(&fname[len - suf_len], dsosuf)) {
|
|
|
|
/* wrong suffix */
|
|
|
|
ret = -EINVAL;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
if (access(fname, F_OK)) {
|
|
|
|
ret = -ENOENT;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
assert(QTAILQ_EMPTY(&dso_init_list));
|
|
|
|
|
2020-10-28 05:49:44 +00:00
|
|
|
flags = 0;
|
2020-10-19 07:52:20 +00:00
|
|
|
if (!export_symbols) {
|
|
|
|
flags |= G_MODULE_BIND_LOCAL;
|
|
|
|
}
|
|
|
|
g_module = g_module_open(fname, flags);
|
2014-02-10 06:48:57 +00:00
|
|
|
if (!g_module) {
|
2020-09-23 09:12:17 +00:00
|
|
|
if (!mayfail) {
|
|
|
|
fprintf(stderr, "Failed to open module: %s\n",
|
|
|
|
g_module_error());
|
|
|
|
}
|
2014-02-10 06:48:57 +00:00
|
|
|
ret = -EINVAL;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
if (!g_module_symbol(g_module, DSO_STAMP_FUN_STR, (gpointer *)&sym)) {
|
|
|
|
fprintf(stderr, "Failed to initialize module: %s\n",
|
|
|
|
fname);
|
|
|
|
/* Print some info if this is a QEMU module (but from different build),
|
|
|
|
* this will make debugging user problems easier. */
|
|
|
|
if (g_module_symbol(g_module, "qemu_module_dummy", (gpointer *)&sym)) {
|
|
|
|
fprintf(stderr,
|
|
|
|
"Note: only modules from the same build can be loaded.\n");
|
|
|
|
}
|
|
|
|
g_module_close(g_module);
|
|
|
|
ret = -EINVAL;
|
|
|
|
} else {
|
|
|
|
QTAILQ_FOREACH(e, &dso_init_list, node) {
|
blockdev: Add dynamic module loading for block drivers
Extend the current module interface to allow for block drivers to be
loaded dynamically on request. The only block drivers that can be
converted into modules are the drivers that don't perform any init
operation except for registering themselves.
In addition, only the protocol drivers are being modularized, as they
are the only ones which see significant performance benefits. The format
drivers do not generally link to external libraries, so modularizing
them is of no benefit from a performance perspective.
All the necessary module information is located in a new structure found
in module_block.h
This spoils the purpose of 5505e8b76f (block/dmg: make it modular).
Before this patch, if module build is enabled, block-dmg.so is linked to
libbz2, whereas the main binary is not. In downstream, theoretically, it
means only the qemu-block-extra package depends on libbz2, while the
main QEMU package needn't to. With this patch, we (temporarily) change
the case so that the main QEMU depends on libbz2 again.
Signed-off-by: Marc Marí <markmb@redhat.com>
Signed-off-by: Colin Lord <clord@redhat.com>
Reviewed-by: Stefan Hajnoczi <stefanha@redhat.com>
Message-id: 1471008424-16465-4-git-send-email-clord@redhat.com
Reviewed-by: Max Reitz <mreitz@redhat.com>
[mreitz: Do a signed comparison against the length of
block_driver_modules[], so it will not cause a compile error when
empty]
Signed-off-by: Max Reitz <mreitz@redhat.com>
2016-08-12 13:27:03 +00:00
|
|
|
e->init();
|
2014-02-10 06:48:57 +00:00
|
|
|
register_module_init(e->init, e->type);
|
|
|
|
}
|
|
|
|
ret = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
QTAILQ_FOREACH_SAFE(e, &dso_init_list, node, next) {
|
|
|
|
QTAILQ_REMOVE(&dso_init_list, e, node);
|
|
|
|
g_free(e);
|
|
|
|
}
|
|
|
|
out:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2020-09-23 09:12:17 +00:00
|
|
|
bool module_load_one(const char *prefix, const char *lib_name, bool mayfail)
|
2014-02-10 06:48:57 +00:00
|
|
|
{
|
2019-07-22 13:13:23 +00:00
|
|
|
bool success = false;
|
|
|
|
|
2014-02-10 06:48:57 +00:00
|
|
|
#ifdef CONFIG_MODULES
|
|
|
|
char *fname = NULL;
|
2020-03-10 14:58:06 +00:00
|
|
|
#ifdef CONFIG_MODULE_UPGRADES
|
|
|
|
char *version_dir;
|
|
|
|
#endif
|
2018-07-04 18:10:10 +00:00
|
|
|
const char *search_dir;
|
2020-04-11 01:07:46 +00:00
|
|
|
char *dirs[5];
|
2016-09-05 02:50:44 +00:00
|
|
|
char *module_name;
|
2018-07-04 18:10:10 +00:00
|
|
|
int i = 0, n_dirs = 0;
|
2021-06-24 10:38:16 +00:00
|
|
|
int ret;
|
2020-10-19 07:52:20 +00:00
|
|
|
bool export_symbols = false;
|
2016-09-05 02:50:44 +00:00
|
|
|
static GHashTable *loaded_modules;
|
2021-06-24 10:38:16 +00:00
|
|
|
const QemuModinfo *modinfo;
|
|
|
|
const char **sl;
|
2014-02-10 06:48:57 +00:00
|
|
|
|
|
|
|
if (!g_module_supported()) {
|
|
|
|
fprintf(stderr, "Module is not supported by system.\n");
|
2019-07-22 13:13:23 +00:00
|
|
|
return false;
|
2014-02-10 06:48:57 +00:00
|
|
|
}
|
|
|
|
|
2016-09-05 02:50:44 +00:00
|
|
|
if (!loaded_modules) {
|
|
|
|
loaded_modules = g_hash_table_new(g_str_hash, g_str_equal);
|
|
|
|
}
|
|
|
|
|
|
|
|
module_name = g_strdup_printf("%s%s", prefix, lib_name);
|
|
|
|
|
2021-03-16 13:44:56 +00:00
|
|
|
if (g_hash_table_contains(loaded_modules, module_name)) {
|
2016-09-05 02:50:44 +00:00
|
|
|
g_free(module_name);
|
2019-07-22 13:13:23 +00:00
|
|
|
return true;
|
2016-09-05 02:50:44 +00:00
|
|
|
}
|
2021-03-16 13:44:56 +00:00
|
|
|
g_hash_table_add(loaded_modules, module_name);
|
2016-09-05 02:50:44 +00:00
|
|
|
|
2021-06-24 10:38:16 +00:00
|
|
|
for (modinfo = module_info; modinfo->name != NULL; modinfo++) {
|
|
|
|
if (modinfo->deps) {
|
|
|
|
if (strcmp(modinfo->name, module_name) == 0) {
|
|
|
|
/* we depend on other module(s) */
|
|
|
|
for (sl = modinfo->deps; *sl != NULL; sl++) {
|
|
|
|
module_load_one("", *sl, false);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
for (sl = modinfo->deps; *sl != NULL; sl++) {
|
|
|
|
if (strcmp(module_name, *sl) == 0) {
|
|
|
|
/* another module depends on us */
|
|
|
|
export_symbols = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-04 18:10:10 +00:00
|
|
|
search_dir = getenv("QEMU_MODULE_DIR");
|
|
|
|
if (search_dir != NULL) {
|
|
|
|
dirs[n_dirs++] = g_strdup_printf("%s", search_dir);
|
|
|
|
}
|
2020-08-18 10:00:18 +00:00
|
|
|
dirs[n_dirs++] = get_relocated_path(CONFIG_QEMU_MODDIR);
|
2020-08-18 10:11:02 +00:00
|
|
|
dirs[n_dirs++] = g_strdup(qemu_get_exec_dir());
|
2020-03-10 14:58:06 +00:00
|
|
|
|
|
|
|
#ifdef CONFIG_MODULE_UPGRADES
|
|
|
|
version_dir = g_strcanon(g_strdup(QEMU_PKGVERSION),
|
|
|
|
G_CSET_A_2_Z G_CSET_a_2_z G_CSET_DIGITS "+-.~",
|
|
|
|
'_');
|
|
|
|
dirs[n_dirs++] = g_strdup_printf("/var/run/qemu/%s", version_dir);
|
|
|
|
#endif
|
|
|
|
|
2018-07-04 18:10:10 +00:00
|
|
|
assert(n_dirs <= ARRAY_SIZE(dirs));
|
|
|
|
|
|
|
|
for (i = 0; i < n_dirs; i++) {
|
2016-09-05 02:50:44 +00:00
|
|
|
fname = g_strdup_printf("%s/%s%s",
|
2020-08-04 16:14:26 +00:00
|
|
|
dirs[i], module_name, CONFIG_HOST_DSOSUF);
|
2020-10-19 07:52:20 +00:00
|
|
|
ret = module_load_file(fname, mayfail, export_symbols);
|
blockdev: Add dynamic module loading for block drivers
Extend the current module interface to allow for block drivers to be
loaded dynamically on request. The only block drivers that can be
converted into modules are the drivers that don't perform any init
operation except for registering themselves.
In addition, only the protocol drivers are being modularized, as they
are the only ones which see significant performance benefits. The format
drivers do not generally link to external libraries, so modularizing
them is of no benefit from a performance perspective.
All the necessary module information is located in a new structure found
in module_block.h
This spoils the purpose of 5505e8b76f (block/dmg: make it modular).
Before this patch, if module build is enabled, block-dmg.so is linked to
libbz2, whereas the main binary is not. In downstream, theoretically, it
means only the qemu-block-extra package depends on libbz2, while the
main QEMU package needn't to. With this patch, we (temporarily) change
the case so that the main QEMU depends on libbz2 again.
Signed-off-by: Marc Marí <markmb@redhat.com>
Signed-off-by: Colin Lord <clord@redhat.com>
Reviewed-by: Stefan Hajnoczi <stefanha@redhat.com>
Message-id: 1471008424-16465-4-git-send-email-clord@redhat.com
Reviewed-by: Max Reitz <mreitz@redhat.com>
[mreitz: Do a signed comparison against the length of
block_driver_modules[], so it will not cause a compile error when
empty]
Signed-off-by: Max Reitz <mreitz@redhat.com>
2016-08-12 13:27:03 +00:00
|
|
|
g_free(fname);
|
|
|
|
fname = NULL;
|
|
|
|
/* Try loading until loaded a module file */
|
|
|
|
if (!ret) {
|
2019-07-22 13:13:23 +00:00
|
|
|
success = true;
|
blockdev: Add dynamic module loading for block drivers
Extend the current module interface to allow for block drivers to be
loaded dynamically on request. The only block drivers that can be
converted into modules are the drivers that don't perform any init
operation except for registering themselves.
In addition, only the protocol drivers are being modularized, as they
are the only ones which see significant performance benefits. The format
drivers do not generally link to external libraries, so modularizing
them is of no benefit from a performance perspective.
All the necessary module information is located in a new structure found
in module_block.h
This spoils the purpose of 5505e8b76f (block/dmg: make it modular).
Before this patch, if module build is enabled, block-dmg.so is linked to
libbz2, whereas the main binary is not. In downstream, theoretically, it
means only the qemu-block-extra package depends on libbz2, while the
main QEMU package needn't to. With this patch, we (temporarily) change
the case so that the main QEMU depends on libbz2 again.
Signed-off-by: Marc Marí <markmb@redhat.com>
Signed-off-by: Colin Lord <clord@redhat.com>
Reviewed-by: Stefan Hajnoczi <stefanha@redhat.com>
Message-id: 1471008424-16465-4-git-send-email-clord@redhat.com
Reviewed-by: Max Reitz <mreitz@redhat.com>
[mreitz: Do a signed comparison against the length of
block_driver_modules[], so it will not cause a compile error when
empty]
Signed-off-by: Max Reitz <mreitz@redhat.com>
2016-08-12 13:27:03 +00:00
|
|
|
break;
|
2014-02-10 06:48:57 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-22 13:13:23 +00:00
|
|
|
if (!success) {
|
|
|
|
g_hash_table_remove(loaded_modules, module_name);
|
2019-12-20 01:34:10 +00:00
|
|
|
g_free(module_name);
|
2019-07-22 13:13:23 +00:00
|
|
|
}
|
|
|
|
|
2018-07-04 18:10:10 +00:00
|
|
|
for (i = 0; i < n_dirs; i++) {
|
2014-02-10 06:48:57 +00:00
|
|
|
g_free(dirs[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
2019-07-22 13:13:23 +00:00
|
|
|
return success;
|
2014-02-10 06:48:57 +00:00
|
|
|
}
|
2020-06-24 13:10:36 +00:00
|
|
|
|
2021-06-24 10:38:17 +00:00
|
|
|
#ifdef CONFIG_MODULES
|
2020-06-24 13:10:36 +00:00
|
|
|
|
|
|
|
static bool module_loaded_qom_all;
|
|
|
|
|
|
|
|
void module_load_qom_one(const char *type)
|
|
|
|
{
|
2021-06-24 10:38:17 +00:00
|
|
|
const QemuModinfo *modinfo;
|
|
|
|
const char **sl;
|
2020-06-24 13:10:36 +00:00
|
|
|
|
2020-07-20 10:03:51 +00:00
|
|
|
if (!type) {
|
|
|
|
return;
|
|
|
|
}
|
2021-06-24 10:38:17 +00:00
|
|
|
|
|
|
|
for (modinfo = module_info; modinfo->name != NULL; modinfo++) {
|
|
|
|
if (!modinfo->objs) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
for (sl = modinfo->objs; *sl != NULL; sl++) {
|
|
|
|
if (strcmp(type, *sl) == 0) {
|
|
|
|
module_load_one("", modinfo->name, false);
|
|
|
|
}
|
2020-06-24 13:10:36 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void module_load_qom_all(void)
|
|
|
|
{
|
2021-06-24 10:38:17 +00:00
|
|
|
const QemuModinfo *modinfo;
|
2020-06-24 13:10:36 +00:00
|
|
|
|
|
|
|
if (module_loaded_qom_all) {
|
|
|
|
return;
|
|
|
|
}
|
2021-06-24 10:38:17 +00:00
|
|
|
|
|
|
|
for (modinfo = module_info; modinfo->name != NULL; modinfo++) {
|
|
|
|
if (!modinfo->objs) {
|
2020-06-24 13:10:36 +00:00
|
|
|
continue;
|
|
|
|
}
|
2021-06-24 10:38:17 +00:00
|
|
|
module_load_one("", modinfo->name, false);
|
2020-06-24 13:10:36 +00:00
|
|
|
}
|
|
|
|
module_loaded_qom_all = true;
|
|
|
|
}
|
2021-06-24 10:38:17 +00:00
|
|
|
|
2021-06-24 10:38:18 +00:00
|
|
|
void qemu_load_module_for_opts(const char *group)
|
|
|
|
{
|
|
|
|
const QemuModinfo *modinfo;
|
|
|
|
const char **sl;
|
|
|
|
|
|
|
|
for (modinfo = module_info; modinfo->name != NULL; modinfo++) {
|
|
|
|
if (!modinfo->opts) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
for (sl = modinfo->opts; *sl != NULL; sl++) {
|
|
|
|
if (strcmp(group, *sl) == 0) {
|
|
|
|
module_load_one("", modinfo->name, false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-06-24 10:38:17 +00:00
|
|
|
#else
|
|
|
|
|
2021-06-24 10:38:18 +00:00
|
|
|
void qemu_load_module_for_opts(const char *group) {}
|
2021-06-24 10:38:17 +00:00
|
|
|
void module_load_qom_one(const char *type) {}
|
|
|
|
void module_load_qom_all(void) {}
|
|
|
|
|
|
|
|
#endif
|