В моей программе у меня есть код вроде следующего
/* libname may be a relative path */
void loadLib(char const *libname) {
void *handle = dlopen(libname);
/* ... */
dlclose(handle);
}
Внутри /* .. */
мне нужно прочитать файл карты памяти /proc/self/maps
, чтобы найти адрес виртуальной памяти, с которым сопоставляется libname
, и мне также нужно открыть библиотеку, чтобы найти в ней определенные разделы. Для этого мне нужно абсолютное имя, которое dlopen
найдено путем поиска в разных местах (например, в файле кеша ldconfig
). Как я могу получить это имя?
Вот что я, наконец, закончил (да, это код на С++, тем не менее, тег C имеет смысл для этого вопроса, потому что dlopen
используется как с С++, так и с C, и мой вопрос подходит для обоих, и POSIX указывает его для C.).
boost::shared_ptr<void> dl;
if(void *handle = dlopen(libfile, RTLD_LAZY)) {
dl.reset(handle, &dlclose);
} else {
printdlerr();
return -1;
}
/* update sofile to be an absolute file name */
{
struct link_map *map;
dlinfo(dl.get(), RTLD_DI_LINKMAP, &map);
if(!map) {
return -1;
}
char *real = realpath(map->l_name, NULL);
if(!real)
return -1;
sofile.reset(real, &free);
}
libfile
- это относительное/обычное имя файла. На карте будет отображаться не-равное имя файла (т.е. foo.so
, но может быть ./foo.so
). Впоследствии я использовал realpath
, чтобы получить окончательное абсолютное имя пути. Он работает красиво!