如何将文件指针从c传递给asm中的调用

内容来源于 Stack Overflow,并遵循CC BY-SA 3.0许可协议进行翻译与使用

  • 回答 (2)
  • 关注 (0)
  • 查看 (37)

我正在使用c打开一个文件,然后我想使用为打开文件返回的指针来处理文本。但是,当我用rdi中的指针调用fgetc时,我得到“没有这样的文件或目录”的错误。

我究竟做错了什么?

int64_t asmFunc(FILE* a, char* b);

int main()
{
   int num;
   FILE *fptr;
   size_t line_buf_size = 0;
   char *ret = malloc(100);
   fptr = fopen("./test.txt","r");

   if(fptr == NULL)
   {
      printf("Error!");   
      exit(1);             
   }

   printf("%ld", asmFunc(fptr, ret));

   return 0;
}
global asmFunc

section .text

extern fgetc

asmFunc:
  call fgetc  ; segfault occurs here.
(...)
  ret

提问于
用户回答回答于

您需要学习并遵循Linux x86-64 ABI规范中记录的调用约定,特别是其§3.2.3参数传递部分。所以指针值是,并且指针值在,你可能应该推送一个调用框架fptr%rdiret%rsiasmFunc

另请阅读x86调用约定 wikipage。

如果你能够asmFunc在某个example.c文件中编写C语言中的等效语句(甚至是简化的),我建议使用它来编译gcc -O -fverbose-asm -Wall -S example.c并查看发出的example.s汇编程序文件以获得灵感。大多数情况下,这样一个函数的第一个机器指令不是call(但是有些东西,称为函数序言,更改堆栈指针%esp并在调用堆栈上分配一些调用帧)

例如,在我的Linux / Debian / x86-64上使用gcc-8

void asmfunc(FILE* fil, char*s) {
   fputc ('\t', fil);
   fputs (s, fil);
   fputc ('\n', fil);
   fflush (fil);
}

编译成:

    .text
    .globl  asmfunc
    .type   asmfunc, @function
asmfunc:
.LFB11:
    .cfi_startproc
    pushq   %rbp    #
    .cfi_def_cfa_offset 16
    .cfi_offset 6, -16
    pushq   %rbx    #
    .cfi_def_cfa_offset 24
    .cfi_offset 3, -24
    subq    $8, %rsp    #,
    .cfi_def_cfa_offset 32
    movq    %rdi, %rbx  # fil, fil
    movq    %rsi, %rbp  # s, s
# /tmp/example.c:4:    fputc ('\t', fil);
    movq    %rdi, %rsi  # fil,
    movl    $9, %edi    #,
    call    fputc@PLT   #
# /tmp/example.c:5:    fputs (s, fil);
    movq    %rbx, %rsi  # fil,
    movq    %rbp, %rdi  # s,
    call    fputs@PLT   #
# /tmp/example.c:6:    fputc ('\n', fil);
    movq    %rbx, %rsi  # fil,
    movl    $10, %edi   #,
    call    fputc@PLT   #
# /tmp/example.c:7:    fflush (fil);
    movq    %rbx, %rdi  # fil,
    call    fflush@PLT  #
# /tmp/example.c:8: }
    addq    $8, %rsp    #,
    .cfi_def_cfa_offset 24
    popq    %rbx    #
    .cfi_def_cfa_offset 16
    popq    %rbp    #
    .cfi_def_cfa_offset 8
    ret 
    .cfi_endproc
.LFE11:
    .size   asmfunc, .-asmfunc
    .ident  "GCC: (Debian 8.3.0-6) 8.3.0"

但请注意,在某些情况下,GCC能够(例如-O2)进行尾调用优化,并可能专门调用一些叶子函数

用户回答回答于

asmFunc的第一条指令也不是调用,但我删除了一些设置内容以供以后的操作使其更易于阅读。

好吧,这只是击败了MCVE的全部目的。您需要简化重新运行测试以确保它仍然显示与完整版本相同的问题。但是对于这个答案,我假设您的设置没有破坏fptrRDI中的arg或修改RSP。

asmFunc:
  call fgetc  ; segfault occurs here.

fptr仍然在RDI中,你的调用者通过它,所以这是正确的int fgetc(FILE *fp)

所以可能fgetc是segfaulting,因为你用一个未对齐的堆栈调用它。 (它在call跳转到之前是16字节对齐asmFunc,但你没有奇数次推或任何sub rsp, 8*n)。glibc的现代版本实际上依赖于scanf的16字节对齐(glibc scanf从不对齐RSP的函数调用时出现分段错误)所以很容易想象fgetc包含的代码也包含了movaps一些内容。堆。

一旦你修复了这个 bug,你就会遇到call fgetc破坏你的char *retarg 的问题,因为你的调用者在RSI中传递了它。 Arg传递寄存器被调用破坏通过linux x86-64函数调用保留了哪些寄存器

asmFunc:            ; (FILE *fptr,  char *ret)
  push   rsi        ; save ret
  call   fgetc
  pop    rsi
  mov    [rsi], al
  ret

AC编译器通常会保存/恢复RBX并用于mov保存ret

asmFunc:            ; (FILE *fptr,  char *ret)
  push   rbx
  mov    rbx, rsi   ; save ret
  call   fgetc
  mov    [rbx], al

  pop    rbx        ; restore rbx
  ret

但是,当我用rdi中的指针调用fgetc时,我得到一个“没有这样的文件或目录”,然后是段错误。

不知道你是怎么得到“没有这样的文件或目录”。你的调试器是否正在寻找glibc函数的源代码?如果它是你的程序本身打印的一部分,那就几乎没有意义,因为你在什么exit(1)时候做得正确 fptr == NULL。并且您不使用perror()或查找errno代码的任何其他内容来生成标准错误字符串。

所属标签

可能回答问题的人

  • 学生

    3 粉丝476 提问7 回答
  • uncle_light

    5 粉丝518 提问6 回答
  • 最爱开车啦

    8 粉丝503 提问5 回答
  • 骑牛看晨曦

    4 粉丝522 提问5 回答

扫码关注云+社区

领取腾讯云代金券