linux nasm汇编中调用printf不报错,但调用scanf报错。抛出了分段错误(核心转储)

当我写了如下汇编时

bash 复制代码
; nasm -f elf64 -g -F dwarf charsin.asm
; gcc charsin.o -no-pie -o charsin
; ld -o eatclib eatclib.o
; gdb eatclib


[SECTION .data]
  SPrompt db 'Enter string data, followed by Enter: ',0
  IPrompt db 'Enter an integer value, followed by Enter: ',10,0
  IFormat db "%d",0
  SShow db 'The string you entered was: %s',0
  IShow db 'The Integer value you entered was: %5d',10,0


[SECTION .bss]
 IntVal resd 1
 InString resb 128 


[SECTION .text]

 extern stdin
 extern fgets
 extern printf
 extern scanf
 global main

 main:
   
   push rbp
   mov rbp,rsp
   push rbx
   push rsi
   push rdi
   
   
   ;push SPrompt
   mov rdi,SPrompt ; 传递参数
   mov rax,0
   call printf
   ;add rsp,8
 
   ; rdi,rsi,rdx,rcx分别保存第1,2,3,4个参数
   ;push qword [stdin]
   ;push 72
   ;push InString
   mov rdi,InString
   mov rsi,72
   mov rdx,[stdin]
   mov rax,0
   call fgets
   ;add rsp,24
   
   ;push InString
   ;push SShow
   mov rdi,SShow
   mov rsi,InString
   ;在汇编代码中,若是调用变长参数的函数,需要使用al这个寄存器记录使用到的向量寄存器数量。
   ;这段代码中eax中存放的不是系统调用号,存放的是向量寄存器使用到的数量0,因为printf使用的是变长参数。
   mov rax,0
   call printf
   ;add rsp,16
   
   mov rax,0
   push IPrompt
   mov rdi,IPrompt ; 传递参数
   call printf
   add rsp,8
   
   ;push IntVal
   ;push IFormat
   mov rdi,IFormat 
   mov rsi,IntVal ; 保留数字的内存地址
   xor rax,rax
   call scanf
   ;add rsp,16
   
   ;mov rax, 0  ;sys_read 系统调用号
   ;mov rdi, 0  ; 文件描述符 标准输入
   ;mov rsi,IntVal ; 保留数字的内存地址
   ;mov rdx, 4
   ;syscall
   
   xor rsi,rsi
   mov rax,0
   mov rdi,IShow ; 传递格式参数
   mov esi,[IntVal]
   call printf
   ;add rsp,16
  
  pop rdi
  pop rsi
  pop rbx
  mov rsp,rbp
  pop rbp
  
  
  ret

编译后,系统会如下段错误(Segmentation fault (core dumped)).

在函数的开始/结束处使用 sub rsp, 8/add rsp, 8 将堆栈重新对齐到 16函数执行调用之前的字节。

bash 复制代码
; nasm -f elf64 -g -F dwarf charsin.asm
; gcc charsin.o -no-pie -o charsin
; ld -o eatclib eatclib.o
; gdb eatclib


[SECTION .data]
  SPrompt db 'Enter string data, followed by Enter: ',0
  IPrompt db 'Enter an integer value, followed by Enter: ',10,0
  IFormat db "%d",0
  SShow db 'The string you entered was: %s',0
  IShow db 'The Integer value you entered was: %5d',10,0


[SECTION .bss]
 IntVal resd 1
 InString resb 128 


[SECTION .text]

 extern stdin
 extern fgets
 extern printf
 extern scanf
 global main

 main:
   
   push rbp
   mov rbp,rsp
   push rbx
   push rsi
   push rdi
   
   ;在函数的开始/结束处使用 sub rsp, 8/add rsp, 8 将堆栈重新对齐到 16函数执行调用之前的字节。
   sub rsp, 8
   
   ;push SPrompt
   mov rdi,SPrompt ; 传递参数
   mov rax,0
   call printf
   ;add rsp,8
 
   ; rdi,rsi,rdx,rcx分别保存第1,2,3,4个参数
   ;push qword [stdin]
   ;push 72
   ;push InString
   mov rdi,InString
   mov rsi,72
   mov rdx,[stdin]
   mov rax,0
   call fgets
   ;add rsp,24
   
   ;push InString
   ;push SShow
   mov rdi,SShow
   mov rsi,InString
   ;在汇编代码中,若是调用变长参数的函数,需要使用al这个寄存器记录使用到的向量寄存器数量。
   ;这段代码中eax中存放的不是系统调用号,存放的是向量寄存器使用到的数量0,因为printf使用的是变长参数。
   mov rax,0
   call printf
   ;add rsp,16
   
   mov rax,0
   push IPrompt
   mov rdi,IPrompt ; 传递参数
   call printf
   add rsp,8
   
   ;push IntVal
   ;push IFormat
   mov rdi,IFormat 
   mov rsi,IntVal ; 保留数字的内存地址
   xor rax,rax
   call scanf
   ;add rsp,16
   
   ;mov rax, 0  ;sys_read 系统调用号
   ;mov rdi, 0  ; 文件描述符 标准输入
   ;mov rsi,IntVal ; 保留数字的内存地址
   ;mov rdx, 4
   ;syscall
   
   xor rsi,rsi
   mov rax,0
   mov rdi,IShow ; 传递格式参数
   mov esi,[IntVal]
   call printf
   ;add rsp,16
  
  ;在函数的开始/结束处使用 sub rsp, 8/add rsp, 8 将堆栈重新对齐到 16函数执行调用之前的字节。
  add rsp, 8
  
  pop rdi
  pop rsi
  pop rbx
  mov rsp,rbp
  pop rbp
  
  
  ret

编译后,程序运行正常。

参考段错误
StackOverflow

相关推荐
Sheffield5 小时前
Alpine是什么,为什么是Docker首选?
linux·docker·容器
舒一笑1 天前
程序员效率神器:一文掌握 tmux(服务器开发必备工具)
运维·后端·程序员
Johny_Zhao1 天前
centos7安装部署openclaw
linux·人工智能·信息安全·云计算·yum源·系统运维·openclaw
haibindev1 天前
在 Windows+WSL2 上部署 OpenClaw AI员工的实践与踩坑
linux·wsl2·openclaw
NineData1 天前
数据库管理工具NineData,一年进化成为数万+开发者的首选数据库工具?
运维·数据结构·数据库
梦想很大很大2 天前
拒绝“盲猜式”调优:在 Go Gin 项目中落地 OpenTelemetry 链路追踪
运维·后端·go
Sinclair2 天前
内网服务器离线安装 Nginx+PHP+MySQL 的方法
运维
叶落阁主2 天前
Tailscale 完全指南:从入门到私有 DERP 部署
运维·安全·远程工作
0xDevNull2 天前
Linux切换JDK版本详细教程
linux