我正在尝试使用 GAS 语法创建一个汇编程序,该程序可以.data
在 x86-64 架构上以与位置无关的方式从部分访问它的变量,并强制执行32 位架构和 IS(%eip
而不是%rip
)。
无论我尝试了什么寄存器,我得到的最好结果都是一个Segmentation fault: 11
,甚至是访问我根本不应该做的 EIP,因此是 SF。最好的结果,因为这至少告诉了我“嗯,它不会做”以外的东西。
我gcc
在 macOS 10.13.6 mid 2010 Intel Core 2 Duo 上编译文件(这clang
可能就是为什么):
$ gcc --version
Configured with: --prefix=/Applications/Xcode.app/Contents/Developer/usr --with-gxx-include-dir=/usr/include/c++/4.2.1
Apple LLVM version 9.1.0 (clang-902.0.39.2)
Target: x86_64-apple-darwin17.7.0
Thread model: posix
InstalledDir: /Applications/Xcode.app/Contents/Developer/Toolchains/XcodeDefault.xctoolchain/usr/bin
并将一些选项传递给链接器:
gcc -m32 -Wl,-fatal_warnings,-arch_errors_fatal,-warn_commons,-pie test.s
ld:警告:PIE 已禁用。在代码签名的 PIE 中不允许使用绝对寻址(可能是 -mdynamic-no-pic),但在来自 /whatever.../test-a07cf9.o 的 _main 中使用。要修复此警告,请不要使用 -mdynamic-no-pic 进行编译或使用 -Wl,-no_pie 进行链接-v 查看调用)1
测试.s
.text
.global _main
_main:
xor %eax, %eax
xor %ebx, %ebx
# lea var1(%esi/edi/ebp/esp), %ebx # can't compile, not PIE
# lea var1(%eip), %ebx # segfault, obvs
# lea (%esp), %ebx # EBX = 17
# lea (%non-esp), %ebx # segfault
# lea 0(%esi), %ebx # segfault
# lea 0(%edi), %ebx # segfault
# lea 0(%ebp), %ebx # EBX = 0
# lea 0(%esp), %ebx # EBX = 17
# lea 0(%eip), %ebx # segfault, obvs
movl (%ebx), %eax
ret
.data
var1: .long 6
.end
我正在运行它./a.out; echo $?
以检查最后的 EAX 值ret
。
我查看了各种来源,但主要是英特尔语法或这些问题之一 - 1 , 2 , 3。我试图反汇编我能想到的最简单的 C 示例,即全局变量 +return
来自main()
- gcc -S test.c -fPIE -pie -fpie -m32
:
int var1 = 6;
int main() { return var1; }
这基本上导致:
.section __TEXT,__text,regular,pure_instructions
.macosx_version_min 10, 13
.globl _main ## -- Begin function main
.p2align 4, 0x90
_main: ## @main
.cfi_startproc
## BB#0:
pushl %ebp
Lcfi0:
.cfi_def_cfa_offset 8
Lcfi1:
.cfi_offset %ebp, -8
movl %esp, %ebp
Lcfi2:
.cfi_def_cfa_register %ebp
pushl %eax
calll L0$pb
L0$pb:
popl %eax
movl $0, -4(%ebp)
movl _var1-L0$pb(%eax), %eax
addl $4, %esp
popl %ebp
retl
.cfi_endproc
## -- End function
.section __DATA,__data
.globl _var1 ## @var1
.p2align 2
_var1:
.long 6 ## 0x6
.subsections_via_symbols
这显然使用 MOV 作为 LEA 和几乎与我的指令相同的指令,除了-L0$pb
应该是 +/- 地址的部分_var1
- 地址L0$pb
进入该.data
部分。
然而,当我尝试使用var1
和_main
标签相同的方法时,什么也没有:
.text
.global _main
_main:
xor %eax, %eax
xor %ebx, %ebx
#movl var1-_main(%ebp), %eax # EAX = 191
#movl var1-_main(%esp), %eax # EAX = 204
#movl var1-_main(%eax), %eax # segfault
ret
.data
var1: .long 6
.end
任何想法我做错了什么?
编辑:
我设法从反汇编的 C 示例中删除了任何不必要的东西,最后得到了这个:
.text
.global _main
_main:
pushl %ebp
pushl %eax
calll test
test:
popl %eax
/* var1, var2, ... */
movl var1-test(%eax), %eax
addl $4, %esp
popl %ebp
retl
/**
* how var1(label) - test(label) skips this label
* if it's about address subtracting?
*/
blobbbb:
xor %edx, %edx
.data
var1: .long 6
var2: .long 135
这对我来说没有多大意义,因为根据本指南,调用者应该 1)将参数推送到堆栈(无)2)call
标签和被调用者实际上应该使用 ESP、EBP 和其他寄存器。另外,为什么我什至需要一个中间标签或者更好地说,没有它有什么办法吗?