将大型数组写入内存 x86 程序集 - 使用堆栈空间的段错误

问题描述

我正在尝试使用 AT&T (GAS) 语法将数组写入 x86 程序集中的堆栈。我遇到了一个问题,我可以将大约 837 万个条目写入堆栈,然后如果我再尝试写入,则会得到 Segmentation fault。我不知道这是为什么,这是我正在使用的代码

    mov %rsp,%rbp
    mov $8378658,%rdx
writeDataLoop:
    sub %rdx,%rbp
    movb $0b1,(%rbp)

    add %rdx,%rbp

    sub $1,%rdx
    cmp $0,%rdx
    jg writeDataLoop

我发现的另一件奇怪的事情是,每次运行时,我可以写入数据的限制会发生很小的变化(大约为 8378658,这在十六进制 (0x7fd922) 中也没什么意义)。谁能告诉我如何将更多数据写入堆栈,并可能解释这个任意堆栈写入限制是什么?谢谢。

解决方法

首先,默认堆栈大小为 8MB,这是我达到的堆栈限制。这可以通过 ulimit -a 命令找到。但是,不应该将堆栈用于大量数据(通常是数组)。相反,应使用 .space 指令,该指令使用 AT&T 语法,以字节为单位存储要存储的数据量:.space <buffersize>。这可以标记,例如:

buffer: .space 1024

这会为标签 buffer 分配 1024 字节的存储空间。此标签应位于程序的 .bss 部分,以允许读取和写入访问。

最后,要访问此数据(读取或写入),可以使用 buffer(%rax),其中 rax 是偏移量。

编辑: 使用 .bss 部分在文件大小方面比使用 .data 部分更有效,您只需手动初始化数组。

相关问答

Selenium Web驱动程序和Java。元素在(x,y)点处不可单击。其...
Python-如何使用点“。” 访问字典成员?
Java 字符串是不可变的。到底是什么意思?
Java中的“ final”关键字如何工作?(我仍然可以修改对象。...
“loop:”在Java代码中。这是什么,为什么要编译?
java.lang.ClassNotFoundException:sun.jdbc.odbc.JdbcOdbc...