X-Git-Url: https://git.draconx.ca/gitweb/upkg.git/blobdiff_plain/9d3cb178db367962f68521985ef9e239e1d03fa9..b46bbe4f64a68fdec38675690f9576c2cbdf4cd3:/src/module.c diff --git a/src/module.c b/src/module.c index 5edc5e2..bd22cfc 100644 --- a/src/module.c +++ b/src/module.c @@ -1,24 +1,122 @@ #include +#include +#include +#include #include #include #include "module.h" +#include "avl.h" static unsigned initialized; +static struct avl_table *package_tree; + +static char *str_cpy_lower(char *dst, const char *src) +{ + size_t i; + + for (i = 0; src[i]; i++) + dst[i] = tolower(src[i]); + dst[i] = 0; + + return dst; +} + +G_DEFINE_TYPE(UPkgModule, upkg_module, G_TYPE_TYPE_MODULE); static void dl_print_errors(const char *prefix) { const char *err; while (err = lt_dlerror()) { if (prefix) fprintf(stderr, "%s: ", prefix); - puts(lt_dlerror()); + puts(err); + } +} + +static gboolean module_load(GTypeModule *m) +{ + UPkgModule *mod = UPKG_MODULE(m); + int (*init_func)(GTypeModule *); + + mod->dl = lt_dlopenext(G_TYPE_MODULE(m)->name); + if (!mod->dl) { + dl_print_errors(__func__); + return FALSE; + } + + init_func = lt_dlsym(mod->dl, "init"); + if (!init_func || init_func(m) != 0) { + dl_print_errors(__func__); + lt_dlclose(mod->dl); + return FALSE; + } + + return TRUE; +} + +static void module_unload(GTypeModule *m) +{ + UPkgModule *mod = UPKG_MODULE(m); + + if (lt_dlclose(mod->dl) != 0) { + dl_print_errors(__func__); + } +} + +static void upkg_module_init(UPkgModule *mod) +{ +} + +static void upkg_module_class_init(UPkgModuleClass *class) +{ + GTypeModuleClass *modclass = G_TYPE_MODULE_CLASS(class); + + modclass->load = module_load; + modclass->unload = module_unload; +} + +UPkgModule *upkg_module_new(const char *name) +{ + char *name2; + + if (!name) { + return NULL; + } + + name2 = malloc(strlen(name)+1); + if (!name2) { + return NULL; + } + + UPkgModule *mod = g_object_new(UPKG_MODULE_TYPE, NULL); + if (!mod) { + free(name2); + return NULL; } + + G_TYPE_MODULE(mod)->name = str_cpy_lower(name2, name); + return mod; +} + +static int modcmp(const void *a, const void *b, void *_data) +{ + const GTypeModule *ma = a; + const GTypeModule *mb = b; + + return strcmp(ma->name, mb->name); } int module_init(void) { if (!initialized) { + package_tree = avl_create(modcmp, NULL, NULL); + if (!package_tree) { + fprintf(stderr, "%s: failed to create package tree.\n", __func__); + return -1; + } + if (lt_dlinit() != 0) { + avl_destroy(package_tree, NULL); dl_print_errors(__func__); return -1; } @@ -39,4 +137,36 @@ int module_exit(void) dl_print_errors(__func__); return -1; } + + return 0; +} + +GType module_get_class(const char *package, const char *class) +{ + char buf[strlen(package) + strlen(class) + 1]; + GTypeModule search = { .name = str_cpy_lower(buf, package) }; + + GTypeModule *mod = avl_find(package_tree, &search); + if (!mod) { + void **p; + + mod = G_TYPE_MODULE(upkg_module_new(package)); + if (!mod) { + return 0; + } + + p = avl_probe(package_tree, mod); + if (!p) { + g_object_unref(mod); + return 0; + } + } + + if (!g_type_module_use(mod)) + return 0; + + str_cpy_lower(buf+strlen(package), class); + buf[0] = toupper(buf[0]); + buf[strlen(package)] = toupper(buf[strlen(package)]); + return g_type_from_name(buf); }