下面我将给出一段汇编代码,请分析每一行指令的作用,并解释main,p1和p2这三个函数的堆栈变化过程
.file "test.c"
.text
.section .rodata
.LC0:
.string "%c\n"
.text
.globl p1
.type p1, @function
p1:
.LFB0:
.cfi_startproc
pushq %rbp
.cfi_def_cfa_offset 16
.cfi_offset 6, -16
movq %rsp, %rbp
.cfi_def_cfa_register 6
subq $16, %rsp
movl %edi, %eax
movb %al, -4(%rbp)
movsbl -4(%rbp), %eax
movl %eax, %esi
leaq .LC0(%rip), %rax
movq %rax, %rdi
movl $0, %eax
call printf@PLT
nop
leave
.cfi_def_cfa 7, 8
ret
.cfi_endproc
.LFE0:
.size p1, .-p1
.globl p2
.type p2, @function
p2:
.LFB1:
.cfi_startproc
pushq %rbp
.cfi_def_cfa_offset 16
.cfi_offset 6, -16
movq %rsp, %rbp
.cfi_def_cfa_register 6
movl %edi, -4(%rbp)
movl %esi, -8(%rbp)
movl -4(%rbp), %edx
movl -8(%rbp), %eax
addl %edx, %eax
popq %rbp
.cfi_def_cfa 7, 8
ret
.cfi_endproc
.LFE1:
.size p2, .-p2
.section .rodata
.LC1:
.string "%d=%d+%d\n"
.text
.globl main
.type main, @function
main:
.LFB2:
.cfi_startproc
pushq %rbp
.cfi_def_cfa_offset 16
.cfi_offset 6, -16
movq %rsp, %rbp
.cfi_def_cfa_register 6
subq $16, %rsp
movb $97, -1(%rbp)
movl $1, -8(%rbp)
movl $2, -12(%rbp)
movsbl -1(%rbp), %eax
movl %eax, %edi
call p1
movl -12(%rbp), %edx
movl -8(%rbp), %eax
movl %edx, %esi
movl %eax, %edi
call p2
movl %eax, -16(%rbp)
movl -12(%rbp), %ecx
movl -8(%rbp), %edx
movl -16(%rbp), %eax
movl %eax, %esi
leaq .LC1(%rip), %rax
movq %rax, %rdi
movl $0, %eax
call printf@PLT
movl $0, %eax
leave
.cfi_def_cfa 7, 8
ret
.cfi_endproc
.LFE2:
.size main, .-main
.ident "GCC: (Debian 13.1.0-6) 13.1.0"
.section .note.GNU-stack,"",@progbits