- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我花了整整一天的时间试图编译一些简单的程序,但到目前为止运气不佳。我要做的是编译和运行以nasm汇编语言编写的程序。
我已升级到最新的nasm(v2.10.09)。因此,让我跳入代码,因为我对这些事情还不了解。这是在linux上使用elf
格式和链接的gcc gcc运行的汇编代码的一部分(评论是我对正在发生的事情的理解):
bits 32
extern printf
global main
section .data
message db "Hello world!", 10, 0
section .text
main:
pushad ;push all registers on stack
push dword message ;push the string on stack
call printf ;call printf
add esp, 4 ;clear stack
popad ;pop all registers back
ret ;return to whoever called me
printf
和
scanf
?
scanf
和
printf
值返回的另一种徒劳尝试(实际上是编译和链接-甚至运行!):
[bits 32] ; why the []?
section .data
input_string db "Enter limit: %d", 0
output_string db "Value %d", 10, 0
counter dd 10
limit dd 0
;nasm -f macho -o test.o test.asm
;ld -lc -o test -arch i386 test.o -macosx_version_min 10.7
section .text
global start
extern _printf
extern _scanf
extern _exit
start:
push ebp ;push stack base
mov ebp, esp ;base is now current top
and esp, 0xFFFFFFF0 ;align the stack - WHY? I just googled this?
sub esp, 16 ;16 bytes for variables - 16 to keep the stack "aligned". WHY?
mov dword[esp], input_string ;ordinary push just isint good nuff for mac... WHY?
mov dword[esp + 4], limit
call _scanf ;does scan something but doesnt print the message, it just scans and then prints the message
mov eax, [limit] ;since i cant push this lets put it in eax first
mov dword[esp + 8], output_string ;push the output string. WHY again MOV?
mov dword[esp + 12], eax ;and the previusly "scanned" variable
call _printf ;print it out
mov dword[esp], 0 ;return value
call _exit ;return
nasm -f macho -o test.o test.asm
进行编译,并将其与
d -lc -o test -arch i386 test.o -macosx_version_min 10.7
链接。无法正常工作。在linux上,此scanf和printf东西 super 容易。怎么了能做得更简单吗?
nasm -f macho -o out.o test.asm
进行编译,但是不使用
gcc -o test out.o
或通过使用
ld -lc -o test -arch i386 out.o -macosx_version_min 10.7
进行链接,并附加平面
-arch i386
也不能解决该问题。如果可以编写“类似于Linux的程序集”,我将非常高兴,因为我不必担心堆栈对齐和类似的问题。
ld: warning: ignoring file out.o, file was built for i386 which is not the architecture being linked (x86_64): out.o
Undefined symbols for architecture x86_64:
"_main", referenced from:
start in crt1.10.6.o
ld: symbol(s) not found for architecture x86_64
Undefined symbols for architecture i386:
"printf", referenced from:
main in out.o
"start", referenced from:
-u command line option
ld: symbol(s) not found for architecture i386
最佳答案
您在询问有关代码的很多问题,而您实际上根本不了解其中的汇编代码。
首先,由于您编写代码的方式,main
例程将成为C风格程序的入口。由于mac os x链接的工作方式;您必须将其命名为_main
,以匹配链接程序在生成可执行文件时引入/usr/lib/crt1.o
时作为默认程序入口点的链接程序所寻找符号的名称(如果对文件进行nm处理,则将看到类似U _main
的条目。类似地,所有库例程都以前导下划线开头,因此如果要使用它们,则必须使用该前缀。
其次,MAC OS调用约定要求所有call
的堆栈的16字节对齐,这意味着您必须确保堆栈指针在每个点都进行了相关对齐。在主例程的入口点,您已经知道由于将返回地址存储在堆栈中以便从main返回而错位了。这意味着,即使您要进行单个调用,也必须将堆栈向下移动至少12个字节才能进行调用。
有了这些信息,我们将省略ebp
,只为代码目的而专门使用esp。
这是假设的序言:
bits 32
extern _printf
global _main
section .data
message db "Hello world!", 10, 0
section .text
_main:
_main
时,重新调整堆栈:
sub esp, 12
mov dword[esp], message
call _printf
add esp, 12
main
的返回码,并返回:
mov eax, 0
ret
eax
作为例程的返回码,只要它适合寄存器即可。编译并链接代码后:
nasm -f macho -o test.o test.asm
ld -o test -arch i386 test.o -macosx_version_min 10.7 -lc /usr/lib/crt1.o
0
退出。
scan_string db "%d", 0
limit dd 0
scan_string
的地址存储在
esp
中,然后将limit的地址存储在
esp + 4
中,然后调用scanf:
mov dword[esp], scan_string
mov dword[esp + 4], limit
call _scanf
output_string db "Value %d", 10, 0
mov dword[esp], output_string
esp + 4
-即printf的第二个参数:
mov eax, [limit]
mov dword[esp + 4], eax
call _printf
_exit
函数-这与简单的print变体不同,因为我们实际上是在调用exit,而不是简单地返回。
mov dword[esp], 0
call _exit
Because that's how Mac OS X does it
It is, but we aligned the stack at the start of the routine and an aligned stack is a functioning stack, by pushing and popping you're messing with the alignment. This is one of the purposes behind using the
ebp
register rather than theesp
register.
ebp
寄存器,则函数prolog如下所示:
push ebp
mov ebp, esp
sub esp, 8 ; in this case to obtain alignment
add esp, 8
pop ebp
关于c - Mac OS X使用main和scanf/printf的32位nasm汇编程序?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/20082414/
在几个 SO 的问题中,有这些行可以访问代码的父目录,例如os.path.join(os.path.dirname(__file__)) returns nothing和 os.path.join(o
我想用 Python 更改文件模式。 os 模块具有三个功能上看似相同的功能: os.chmod os.fchmod os.lchmod 这三个版本有什么区别? 最佳答案 chmod 用于更改路径指定
考虑: pipe_read, pipe_write = os.pipe() 现在,我想知道两件事: (1) 我有两个线程。如果我保证只有一个正在读取 os.read(pipe_read,n) 而另一个
这个问题不太可能帮助任何 future 的访问者;它只与一个小的地理区域、一个特定的时间点或一个非常狭窄的情况有关,这些情况并不普遍适用于互联网的全局受众。为了帮助使这个问题更广泛地适用,visit
让我们以硬盘驱动器/网络接口(interface)为例。它由多个进程共享。现在多个进程可能会向硬盘驱动器发出并发命令来读取数据。当数据可用时,内核如何知道哪个进程的数据已准备好?操作系统和硬盘驱动器之
嗨,我正在尝试编写像这样的原子写入函数...... with tempfile.NamedTemporaryFile(mode= "w", dir= target_directory) as f:
net.Conn接口(interface)提供了 SetTimeout 方法,我应该用 os.Timeout 检查返回的错误.但是我看不到在返回的 os.Error 上调用 os.Timeout 的方
我正在使用 os 模块在我的 Django 项目 settings.py 文件中具有相对路径。变量 SITE_ROOT 设置为 settings.py 文件的当前工作目录,然后用于引用同样位于同一目录
正如我们所知,Windows 接受 "\" 和 "/" 作为分隔符。但是在python中,使用的是"\"。例如,调用 os.path.join("foo","bar"),将返回 'foo\\bar'。
我有以下工作目录:/Users/jordan/Coding/Employer/code_base ,我想要获取绝对路径的文件位于 /Users/jordan/Coding/Employer/code_
在 Python 中,如果路径中包含“~”,我能否确定扩展的用户调用将是绝对路径? 例如,这个表达式是否总是为真? path = '~/.my_app' os.path.expanduser(path
我是 Django 项目的初学者。Django 项目的 settings.py 文件包含这两行: BASE_DIR = os.path.dirname(os.path.dirname(os.path.
我有一个旧 MAC OS 文件存储中的文件集合。我知道集合存在文件名/路径名问题。问题源于我认为在原始操作系统中呈现为破折号的路径中包含一个代码点,但 Windows 与代码点斗争,并且其中一个包含
Ubuntu怎么安装mac os x主题呢?下文小编将为大家分享ubuntu14.04安装mac os x主题教程,安装MAC OS X&
我有一个 Firefox OS 应用程序,我希望在该应用程序之外打开一个链接(该链接指向不同的站点,在应用程序中打开它会使应用程序在没有强制的情况下无法使用)。我怎么做? Related bug re
我想为 Firefox OS 编写我的应用程序.使用什么样的语言(如 Android 的 Java 和 iOS 的 Objective C++)和工具(如 Eclipse、Xcode)? 最佳答案 适
我正在尝试创建一个 Palm OS 应用程序,以每 X 分钟或几小时检查一次网站,并在有数据可用时提供通知。我知道这种事情可以在新的 Palm 上完成——例如,当应用程序不在顶部时,我的 Centro
我需要在 Firefox OS 中显示全屏图像。我有一个具有 qHD 分辨率(960x540 像素)的“峰值”开发预览手机。 如何确保我的应用程序在其他具有不同屏幕分辨率的 firefox-os 设备
我正在尝试在 Firefox OS 中安装一个新的语言环境,但我不确定我是否正确地按照这些步骤操作。 首先,我尝试使用 Mercurial 下载所需的语言环境:它对我不起作用,Mercurial 说访
我有这个shell脚本Test.sh: #! /bin/bash FILE_TO_CHECK="/Users/test/start.txt" EXIT=0 while [ $EXIT -eq 0 ];
我是一名优秀的程序员,十分优秀!