xref: /openbmc/qemu/util/path.c (revision c4107e82)
1baacf047SPaolo Bonzini /* Code to mangle pathnames into those matching a given prefix.
2baacf047SPaolo Bonzini    eg. open("/lib/foo.so") => open("/usr/gnemul/i386-linux/lib/foo.so");
3baacf047SPaolo Bonzini 
4baacf047SPaolo Bonzini    The assumption is that this area does not change.
5baacf047SPaolo Bonzini */
6aafd7584SPeter Maydell #include "qemu/osdep.h"
7baacf047SPaolo Bonzini #include <sys/param.h>
8baacf047SPaolo Bonzini #include <dirent.h>
9f348b6d1SVeronia Bahaa #include "qemu/cutils.h"
10f348b6d1SVeronia Bahaa #include "qemu/path.h"
11*f3a8bdc1SRichard Henderson #include "qemu/thread.h"
12baacf047SPaolo Bonzini 
13*f3a8bdc1SRichard Henderson static const char *base;
14*f3a8bdc1SRichard Henderson static GHashTable *hash;
15*f3a8bdc1SRichard Henderson static QemuMutex lock;
16baacf047SPaolo Bonzini 
init_paths(const char * prefix)17baacf047SPaolo Bonzini void init_paths(const char *prefix)
18baacf047SPaolo Bonzini {
19*f3a8bdc1SRichard Henderson     if (prefix[0] == '\0' || !strcmp(prefix, "/")) {
20baacf047SPaolo Bonzini         return;
21baacf047SPaolo Bonzini     }
22*f3a8bdc1SRichard Henderson 
23*f3a8bdc1SRichard Henderson     if (prefix[0] == '/') {
24*f3a8bdc1SRichard Henderson         base = g_strdup(prefix);
25*f3a8bdc1SRichard Henderson     } else {
26*f3a8bdc1SRichard Henderson         char *cwd = g_get_current_dir();
27*f3a8bdc1SRichard Henderson         base = g_build_filename(cwd, prefix, NULL);
28*f3a8bdc1SRichard Henderson         g_free(cwd);
29*f3a8bdc1SRichard Henderson     }
30*f3a8bdc1SRichard Henderson 
31*f3a8bdc1SRichard Henderson     hash = g_hash_table_new(g_str_hash, g_str_equal);
32*f3a8bdc1SRichard Henderson     qemu_mutex_init(&lock);
33baacf047SPaolo Bonzini }
34baacf047SPaolo Bonzini 
35baacf047SPaolo Bonzini /* Look for path in emulation dir, otherwise return name. */
path(const char * name)36baacf047SPaolo Bonzini const char *path(const char *name)
37baacf047SPaolo Bonzini {
38*f3a8bdc1SRichard Henderson     gpointer key, value;
39*f3a8bdc1SRichard Henderson     const char *ret;
40baacf047SPaolo Bonzini 
41*f3a8bdc1SRichard Henderson     /* Only do absolute paths: quick and dirty, but should mostly be OK.  */
42*f3a8bdc1SRichard Henderson     if (!base || !name || name[0] != '/') {
43*f3a8bdc1SRichard Henderson         return name;
44*f3a8bdc1SRichard Henderson     }
45*f3a8bdc1SRichard Henderson 
46*f3a8bdc1SRichard Henderson     qemu_mutex_lock(&lock);
47*f3a8bdc1SRichard Henderson 
48*f3a8bdc1SRichard Henderson     /* Have we looked up this file before?  */
49*f3a8bdc1SRichard Henderson     if (g_hash_table_lookup_extended(hash, name, &key, &value)) {
50*f3a8bdc1SRichard Henderson         ret = value ? value : name;
51*f3a8bdc1SRichard Henderson     } else {
52*f3a8bdc1SRichard Henderson         char *save = g_strdup(name);
53*f3a8bdc1SRichard Henderson         char *full = g_build_filename(base, name, NULL);
54*f3a8bdc1SRichard Henderson 
55*f3a8bdc1SRichard Henderson         /* Look for the path; record the result, pass or fail.  */
56*f3a8bdc1SRichard Henderson         if (access(full, F_OK) == 0) {
57*f3a8bdc1SRichard Henderson             /* Exists.  */
58*f3a8bdc1SRichard Henderson             g_hash_table_insert(hash, save, full);
59*f3a8bdc1SRichard Henderson             ret = full;
60*f3a8bdc1SRichard Henderson         } else {
61*f3a8bdc1SRichard Henderson             /* Does not exist.  */
62*f3a8bdc1SRichard Henderson             g_free(full);
63*f3a8bdc1SRichard Henderson             g_hash_table_insert(hash, save, NULL);
64*f3a8bdc1SRichard Henderson             ret = name;
65*f3a8bdc1SRichard Henderson         }
66*f3a8bdc1SRichard Henderson     }
67*f3a8bdc1SRichard Henderson 
68*f3a8bdc1SRichard Henderson     qemu_mutex_unlock(&lock);
69*f3a8bdc1SRichard Henderson     return ret;
70baacf047SPaolo Bonzini }
71