xref: /openbmc/qemu/util/module.c (revision fb6051e7)
1 /*
2  * QEMU Module Infrastructure
3  *
4  * Copyright IBM, Corp. 2009
5  *
6  * Authors:
7  *  Anthony Liguori   <aliguori@us.ibm.com>
8  *
9  * This work is licensed under the terms of the GNU GPL, version 2.  See
10  * the COPYING file in the top-level directory.
11  *
12  * Contributions after 2012-01-13 are licensed under the terms of the
13  * GNU GPL, version 2 or (at your option) any later version.
14  */
15 
16 #include "qemu/osdep.h"
17 #ifdef CONFIG_MODULES
18 #include <gmodule.h>
19 #endif
20 #include "qemu/queue.h"
21 #include "qemu/module.h"
22 #include "qemu/cutils.h"
23 #include "qemu/config-file.h"
24 #include "qapi/error.h"
25 #ifdef CONFIG_MODULE_UPGRADES
26 #include "qemu-version.h"
27 #endif
28 #include "trace.h"
29 
30 typedef struct ModuleEntry
31 {
32     void (*init)(void);
33     QTAILQ_ENTRY(ModuleEntry) node;
34     module_init_type type;
35 } ModuleEntry;
36 
37 typedef QTAILQ_HEAD(, ModuleEntry) ModuleTypeList;
38 
39 static ModuleTypeList init_type_list[MODULE_INIT_MAX];
40 static bool modules_init_done[MODULE_INIT_MAX];
41 
42 static ModuleTypeList dso_init_list;
43 
44 static void init_lists(void)
45 {
46     static int inited;
47     int i;
48 
49     if (inited) {
50         return;
51     }
52 
53     for (i = 0; i < MODULE_INIT_MAX; i++) {
54         QTAILQ_INIT(&init_type_list[i]);
55     }
56 
57     QTAILQ_INIT(&dso_init_list);
58 
59     inited = 1;
60 }
61 
62 
63 static ModuleTypeList *find_type(module_init_type type)
64 {
65     init_lists();
66 
67     return &init_type_list[type];
68 }
69 
70 void register_module_init(void (*fn)(void), module_init_type type)
71 {
72     ModuleEntry *e;
73     ModuleTypeList *l;
74 
75     e = g_malloc0(sizeof(*e));
76     e->init = fn;
77     e->type = type;
78 
79     l = find_type(type);
80 
81     QTAILQ_INSERT_TAIL(l, e, node);
82 }
83 
84 void register_dso_module_init(void (*fn)(void), module_init_type type)
85 {
86     ModuleEntry *e;
87 
88     init_lists();
89 
90     e = g_malloc0(sizeof(*e));
91     e->init = fn;
92     e->type = type;
93 
94     QTAILQ_INSERT_TAIL(&dso_init_list, e, node);
95 }
96 
97 void module_call_init(module_init_type type)
98 {
99     ModuleTypeList *l;
100     ModuleEntry *e;
101 
102     if (modules_init_done[type]) {
103         return;
104     }
105 
106     l = find_type(type);
107 
108     QTAILQ_FOREACH(e, l, node) {
109         e->init();
110     }
111 
112     modules_init_done[type] = true;
113 }
114 
115 #ifdef CONFIG_MODULES
116 
117 static const QemuModinfo module_info_stub[] = { {
118     /* end of list */
119 } };
120 static const QemuModinfo *module_info = module_info_stub;
121 static const char *module_arch;
122 
123 void module_init_info(const QemuModinfo *info)
124 {
125     module_info = info;
126 }
127 
128 void module_allow_arch(const char *arch)
129 {
130     module_arch = arch;
131 }
132 
133 static bool module_check_arch(const QemuModinfo *modinfo)
134 {
135     if (modinfo->arch) {
136         if (!module_arch) {
137             /* no arch set -> ignore all */
138             return false;
139         }
140         if (strcmp(module_arch, modinfo->arch) != 0) {
141             /* mismatch */
142             return false;
143         }
144     }
145     return true;
146 }
147 
148 /*
149  * module_load_dso: attempt to load an existing dso file
150  *
151  * fname:          full pathname of the file to load
152  * export_symbols: if true, add the symbols to the global name space
153  * errp:           error to set.
154  *
155  * Return value:   true on success, false on error, and errp will be set.
156  */
157 static bool module_load_dso(const char *fname, bool export_symbols,
158                             Error **errp)
159 {
160     GModule *g_module;
161     void (*sym)(void);
162     ModuleEntry *e, *next;
163     int flags;
164 
165     assert(QTAILQ_EMPTY(&dso_init_list));
166 
167     flags = 0;
168     if (!export_symbols) {
169         flags |= G_MODULE_BIND_LOCAL;
170     }
171     g_module = g_module_open(fname, flags);
172     if (!g_module) {
173         error_setg(errp, "failed to open module: %s", g_module_error());
174         return false;
175     }
176     if (!g_module_symbol(g_module, DSO_STAMP_FUN_STR, (gpointer *)&sym)) {
177         error_setg(errp, "failed to initialize module: %s", fname);
178         /*
179          * Print some info if this is a QEMU module (but from different build),
180          * this will make debugging user problems easier.
181          */
182         if (g_module_symbol(g_module, "qemu_module_dummy", (gpointer *)&sym)) {
183             error_append_hint(errp,
184                 "Only modules from the same build can be loaded.\n");
185         }
186         g_module_close(g_module);
187         return false;
188     }
189 
190     QTAILQ_FOREACH(e, &dso_init_list, node) {
191         e->init();
192         register_module_init(e->init, e->type);
193     }
194     trace_module_load_module(fname);
195     QTAILQ_FOREACH_SAFE(e, &dso_init_list, node, next) {
196         QTAILQ_REMOVE(&dso_init_list, e, node);
197         g_free(e);
198     }
199     return true;
200 }
201 
202 int module_load(const char *prefix, const char *name, Error **errp)
203 {
204     int rv = -1;
205 #ifdef CONFIG_MODULE_UPGRADES
206     char *version_dir;
207 #endif
208     const char *search_dir;
209     char *dirs[5];
210     char *module_name;
211     int i = 0, n_dirs = 0;
212     bool export_symbols = false;
213     static GHashTable *loaded_modules;
214     const QemuModinfo *modinfo;
215     const char **sl;
216 
217     if (!g_module_supported()) {
218         error_setg(errp, "%s", "this platform does not support GLib modules");
219         return -1;
220     }
221 
222     if (!loaded_modules) {
223         loaded_modules = g_hash_table_new(g_str_hash, g_str_equal);
224     }
225 
226     /* allocate all resources managed by the out: label here */
227     module_name = g_strdup_printf("%s%s", prefix, name);
228 
229     if (g_hash_table_contains(loaded_modules, module_name)) {
230         g_free(module_name);
231         return 2; /* module already loaded */
232     }
233     g_hash_table_add(loaded_modules, module_name);
234 
235     search_dir = getenv("QEMU_MODULE_DIR");
236     if (search_dir != NULL) {
237         dirs[n_dirs++] = g_strdup_printf("%s", search_dir);
238     }
239     dirs[n_dirs++] = get_relocated_path(CONFIG_QEMU_MODDIR);
240 
241 #ifdef CONFIG_MODULE_UPGRADES
242     version_dir = g_strcanon(g_strdup(QEMU_PKGVERSION),
243                              G_CSET_A_2_Z G_CSET_a_2_z G_CSET_DIGITS "+-.~",
244                              '_');
245     dirs[n_dirs++] = g_strdup_printf("/var/run/qemu/%s", version_dir);
246 #endif
247     assert(n_dirs <= ARRAY_SIZE(dirs));
248 
249     /* end of resources managed by the out: label */
250 
251     for (modinfo = module_info; modinfo->name != NULL; modinfo++) {
252         if (modinfo->arch) {
253             if (strcmp(modinfo->name, module_name) == 0) {
254                 if (!module_check_arch(modinfo)) {
255                     error_setg(errp, "module arch does not match: "
256                         "expected '%s', got '%s'", module_arch, modinfo->arch);
257                     goto out;
258                 }
259             }
260         }
261         if (modinfo->deps) {
262             if (strcmp(modinfo->name, module_name) == 0) {
263                 /* we depend on other module(s) */
264                 for (sl = modinfo->deps; *sl != NULL; sl++) {
265                     int subrv = module_load("", *sl, errp);
266                     if (subrv <= 0) {
267                         rv = subrv;
268                         goto out;
269                     }
270                 }
271             } else {
272                 for (sl = modinfo->deps; *sl != NULL; sl++) {
273                     if (strcmp(module_name, *sl) == 0) {
274                         /* another module depends on us */
275                         export_symbols = true;
276                     }
277                 }
278             }
279         }
280     }
281 
282     for (i = 0; i < n_dirs; i++) {
283         char *fname = g_strdup_printf("%s/%s%s",
284                                       dirs[i], module_name, CONFIG_HOST_DSOSUF);
285         int ret = access(fname, F_OK);
286         if (ret != 0 && (errno == ENOENT || errno == ENOTDIR)) {
287             /*
288              * if we don't find the module in this dir, try the next one.
289              * If we don't find it in any dir, that can be fine too: user
290              * did not install the module. We will return 0 in this case
291              * with no error set.
292              */
293             g_free(fname);
294             continue;
295         } else if (ret != 0) {
296             /* most common is EACCES here */
297             error_setg_errno(errp, errno, "error trying to access %s", fname);
298         } else if (module_load_dso(fname, export_symbols, errp)) {
299             rv = 1; /* module successfully loaded */
300         }
301         g_free(fname);
302         goto out;
303     }
304     rv = 0; /* module not found */
305 
306 out:
307     if (rv <= 0) {
308         g_hash_table_remove(loaded_modules, module_name);
309         g_free(module_name);
310     }
311     for (i = 0; i < n_dirs; i++) {
312         g_free(dirs[i]);
313     }
314     return rv;
315 }
316 
317 static bool module_loaded_qom_all;
318 
319 int module_load_qom(const char *type, Error **errp)
320 {
321     const QemuModinfo *modinfo;
322     const char **sl;
323     int rv = 0;
324 
325     if (!type) {
326         error_setg(errp, "%s", "type is NULL");
327         return -1;
328     }
329 
330     trace_module_lookup_object_type(type);
331     for (modinfo = module_info; modinfo->name != NULL; modinfo++) {
332         if (!modinfo->objs) {
333             continue;
334         }
335         if (!module_check_arch(modinfo)) {
336             continue;
337         }
338         for (sl = modinfo->objs; *sl != NULL; sl++) {
339             if (strcmp(type, *sl) == 0) {
340                 if (rv > 0) {
341                     error_setg(errp, "multiple modules providing '%s'", type);
342                     return -1;
343                 }
344                 rv = module_load("", modinfo->name, errp);
345                 if (rv < 0) {
346                     return rv;
347                 }
348             }
349         }
350     }
351     return rv;
352 }
353 
354 void module_load_qom_all(void)
355 {
356     const QemuModinfo *modinfo;
357 
358     if (module_loaded_qom_all) {
359         return;
360     }
361 
362     for (modinfo = module_info; modinfo->name != NULL; modinfo++) {
363         Error *local_err = NULL;
364         if (!modinfo->objs) {
365             continue;
366         }
367         if (!module_check_arch(modinfo)) {
368             continue;
369         }
370         if (module_load("", modinfo->name, &local_err) < 0) {
371             error_report_err(local_err);
372         }
373     }
374     module_loaded_qom_all = true;
375 }
376 
377 void qemu_load_module_for_opts(const char *group)
378 {
379     const QemuModinfo *modinfo;
380     const char **sl;
381 
382     for (modinfo = module_info; modinfo->name != NULL; modinfo++) {
383         if (!modinfo->opts) {
384             continue;
385         }
386         for (sl = modinfo->opts; *sl != NULL; sl++) {
387             if (strcmp(group, *sl) == 0) {
388                 Error *local_err = NULL;
389                 if (module_load("", modinfo->name, &local_err) < 0) {
390                     error_report_err(local_err);
391                 }
392             }
393         }
394     }
395 }
396 
397 #else
398 
399 void module_allow_arch(const char *arch) {}
400 void qemu_load_module_for_opts(const char *group) {}
401 int module_load(const char *prefix, const char *name, Error **errp) { return 2; }
402 int module_load_qom(const char *type, Error **errp) { return 2; }
403 void module_load_qom_all(void) {}
404 
405 #endif
406