Почему dladdr возвращает путь к исполняемому файлу в позиционно-независимом коде на arm-linux?

Я столкнулся с проблемой с dladdr в некоторых моих кодах, которые я кросс-компилировал для arm-linux, используя кросс-компилятор Emdebian g++-4.7 для armhf. Кажется, проблема в том, что если указатель функции, который вы передаете в dladdr, является функцией, которая также вызывается в исполняемом файле, dli_fname вернет путь к исполняемому файлу вместо разделяемой библиотеки, из которой поступил символ.

Пример скомпилирован с помощью:

/usr/bin/arm-linux-gnueabihf-g++-4.7 main.cpp -o main -ldl -fPIC

#include <dlfcn.h>
#include <iostream>
#include <stdio.h>
#include <string>

std::string findLibName(void* funcPtr) {
  Dl_info info;
  if (dladdr(funcPtr,&info) != 0) {
    return std::string(info.dli_fname);
  } else {
    return ""; 
  }   
}

int main() {    
  printf("printf comes from %s\n",findLibName((void*)printf).c_str());
  return 0;
}

Результат: printf исходит от./main

Если я изменю тело main() на

int main() {    
  std::cout<<"printf comes from "<<findLibName((void*)printf)<<"\n";
  return 0;
}

Результат: printf взят из /lib/arm-linux-gnueabihf/libc.so.6

Я заметил, что если функция, указатель на функцию которой вы передаете в dladdr, вызывается внутри исполняемого файла, запись будет выглядеть следующим образом:

Relocation section '.rel.plt' at offset 0x63c contains 14 entries:
Offset     Info    Type            Sym.Value  Sym. Name
00010be8  00000d16 R_ARM_JUMP_SLOT   00008730   printf

создается в разделе.rel.plt исполняемого файла, и кажется, что dladdr находит этот адрес printf из PLT, а не правильный адрес из libc.so.6, который он найдет через GOT. Справочная страница dladdr предполагает, что компиляция кода, чтобы быть независимым от позиции, должна исправить это, но это не похоже на этот компилятор. Для справки, тот же код, показанный выше, работает, как и ожидалось, в x86_64 Linux с g++-4.7.

0 ответов

Другие вопросы по тегам