内联程序集返回:在创建共享对象时,不能使用针对未定义符号的重定位R_X86_64_32S [重复]

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了内联程序集返回:在创建共享对象时,不能使用针对未定义符号的重定位R_X86_64_32S [重复]相关的知识,希望对你有一定的参考价值。

我正在研究Xeno Kovah's example in slide 18 of Intermediate Assembly。他正在使用Visual Studio和Intel Assembly,内联。我已经尝试将其改编为GCC,如下所示。我正在用-masm=intel -fPIC编译

#include <stdio.h>
int main(){
  unsigned int maxBasicCPUID;
  char vendorString[13];
  char * vendorStringPtr = (char *)vendorString; //Move the address into its own register
  //because it makes the asm syntax easier
  //First we will check whether we can even use CPUID
  //Such a check is actually more complicated than it seems (OMITED FROM SLIDES)

  __asm (
    "mov edi, vendorStringPtr;" //Get the base address of the char[] into a register
    "mov eax, 0;" //We're going to do CPUID with input of 0
    "cpuid;" //As stated, the instruction doesn't have any operands
    //Get back the results which are now stored in eax, ebx, ecx, edx
    //and will have values as specified by the manual
    "mov maxBasicCPUID, eax;"
    "mov [edi], ebx;" //We order which register we put into which address
    "mov [edi+4], edx;" //so that they all end up forming a human readable string
    "mov [edi+8], ecx;"
  );
  vendorString[12] = 0;
  printf("maxBasicCPUID = %#x, vendorString = %s
", maxBasicCPUID, vendorString);
  return 0xb45eba11;
}

我不确定我做错了什么,但是我收到了以下错误

/usr/bin/ld: /tmp/ccSapgOG.o: relocation R_X86_64_32S against undefined symbol `vendorStringPtr' can not be used when making a shared object; recompile with -fPIC
/usr/bin/ld: final link failed: Nonrepresentable section on output
collect2: error: ld returned 1 exit status
答案

在gcc中,您不能直接通过汇编代码中的名称引用局部变量。

此外,您需要告诉编译器您使用的所有寄存器(clobber)。

但是,从好的方面来说,你可以让编译器为你做更多的工作,你可以在下面的代码重写中看到:

   uint32_t *str = (uint32_t *)vendorString;
   __asm("cpuid"
       : "=a"(maxBasicCPUID), "=b"(str[0]), "=d"(str[1]), "=c"(str[2])
       : "a"(0));

第一行参数告诉编译器存储结果的位置,第二行告诉编译器在执行内联汇编之前要加载哪些值。

有关所有细节,请参阅https://gcc.gnu.org/onlinedocs/gcc/Extended-Asm.html。 (感谢@MichaelPetch的链接。)

以上是关于内联程序集返回:在创建共享对象时,不能使用针对未定义符号的重定位R_X86_64_32S [重复]的主要内容,如果未能解决你的问题,请参考以下文章

测试指令元素时未定义父级

带有清晰表单的内联表单集 - 仅显示一次标签

LINQ语句中的.AsEnumerable() 和 .AsQueryable()的区别

用户同时编辑时,Django 内联表单集抛出 IndexError

单元测试 django 内联表单集

错误 C2403 使用内联 x86 C++ 程序集将寄存器内容移动到变量