在汇编程序中打印浮点数

时间:2014-11-10 08:18:14

标签: linux assembly printf nasm glibc

我尝试从调用printf函数的assemler打印浮点值。它适用于字符串和整数值,但无法打印浮点数。以下是工作代码的示例:

global  main
extern  printf

section .data
  message:    db      "String is: %d %x %s", 10, 0
  end_message:    db    ".. end of string", 0 

section .text
  main:
    mov eax, 0xff
    mov     edi, message
    movsxd rsi, eax
    mov rdx, 0xff
    mov rcx, end_message
    xor rax, rax
    call printf
    ret
  

String is:255 ff .. string of end

因此,参数通过寄存器传递: edi 包含格式化字符串的地址, rsi rdx 包含相同的数字以便打印在十进制和十六进制样式中, rcx 包含字符串的结尾, rax 包含0,因为我们没有要打印的浮点数。 这段代码工作正常但是在尝试打印float时会发生一些变化:

global main

extern printf

section .data
    val: dq 123.456
    msg: db "Result is: %fl",10, 0

section .text
    main:
    mov rdi,msg
    movsd xmm0,[val]
    mov eax,1
    call printf

    mov rax, 0
    ret

此代码剪切可以编译但返回正在执行的分段错误。似乎问题是 xmm0 的错误值,但是试图将 movsd xmm0,[val] 改为 movsd xmm0,val 给出< / p>

  

错误:操作码和操作数的组合无效

消息。 编译器是在openSuSe 12.3上运行的NASM

更新。我试图制作一个c程序并生成一个.S程序集。它给出了一个非常奇怪的解决方案:

main:
.LFB2:
    .cfi_startproc
    pushq   %rbp
    .cfi_def_cfa_offset 16
    .cfi_offset 6, -16
    movq    %rsp, %rbp
    .cfi_def_cfa_register 6
    subq    $32, %rsp
    movl    %edi, -4(%rbp)
    movq    %rsi, -16(%rbp)
    movq    val(%rip), %rax
    movq    %rax, -24(%rbp)
    movsd   -24(%rbp), %xmm0
    movl    $.LC0, %edi
    movl    $1, %eax
    call    printf
    movl    $0, %eax
    leave
    .cfi_def_cfa 7, 8
    ret

是否可以编写一个简单的printf示例?

1 个答案:

答案 0 :(得分:3)

您的汇编程序问题: 你需要在主程序启动之前对齐堆栈。

插入

sub rsp, 8
在主要之后

然后在ret之前再次添加:

add rsp, 8