前言

C语言的过程调用机制(即函数之间的调用)的一个关键特性(起始大多数编程语言也是如此)都是使用了栈数据结构提供的后进先出的内存管理原则。每一个函数的栈空间被称为栈帧,一个栈帧上包含了保存的寄存器、分配给局部变量的空间以及传递给要调用函数的参数等等。一个基本的栈结构如下图所示:

但是,有一点需要引起注意的是,过程调用的参数是通过栈来传递的,并且分配的局部变量也在栈上,那么对于不同字节长度的参数或变量,是如何在栈上为它们分配空间的?这里所涉及的就是我们要探讨的字节对齐。

本文示例用到的环境如下:

  • Ubuntu x86_64 GNU/Linux
  • gcc 7.4.0

数据对齐

许多计算机系统对基本数据类型的合法地址做了一些限制,要求某种类型对象的地址必须是某个值K的倍数,其中K具体如下图。这种对齐限制简化了形成处理器和内存系统之间接口的硬件设计。举个实际的例子:比如我们在内存中读取一个8字节长度的变量,那么这个变量所在的地址必须是8的倍数。如果这个变量所在的地址是8的倍数,那么就可以通过一次内存操作完成该变量的读取。倘若这个变量所在的地址并不是8的倍数,那么可能就需要执行两次内存读取,因为该变量被放在两个8字节的内存块中了。

K 类型
1 char
2 short
4 int, float
8 long,double,char*

无论数据是否对齐,x86_64硬件都能正常工作,但是却会降低系统的性能,所以我们的编译器在编译时一般会为我们实施数据对齐。

栈的字节对齐

栈的字节对齐,实际是指栈顶指针必须须是16字节的整数倍。栈对齐帮助在尽可能少的内存访问周期内读取数据,不对齐堆栈指针可能导致严重的性能下降。

上文我们说,即使数据没有对齐,我们的程序也是可以执行的,只是效率有点低而已,但是某些型号的Intel和AMD处理器对于有些实现多媒体操作的SSE指令,如果数据没有对齐的话,就无法正确执行。这些指令对16字节内存进行操作,在SSE单元和内存之间传送数据的指令要求内存地址必须是16的倍数。

因此,任何针对x86_64处理器的编译器和运行时系统都必须保证分配用来保存可能会被SSE寄存器读或写的数据结构的内存,都必须是16字节对齐的,这就形成了一种标准:

  • 任何内存分配函数(alloca, malloc, calloc或realloc)生成的块起始地址都必须是16的倍数。
  • 大多数函数的栈帧的边界都必须是16直接的倍数。

如上,在运行时栈中,不仅传递的参数和局部变量要满足字节对齐,我们的栈指针(%rsp)也必须是16的倍数。

三个示例

我们用三个实际的例子来看一看为了实现数据对齐和栈字节对齐,栈空间的分配具体是怎样的。

如下是CSAPP上的一个示例程序。

  1. void proc(long a1, long *a1p,
  2. int a2, int *a2p,
  3. short a3, short *a3p,
  4. char a4, char *a4p) {
  5. *a1p += a1;
  6. *a2p += a2;
  7. *a3p += a3;
  8. *a4p += a4;
  9. }
  10. long call_proc()
  11. {
  12. long x1 = 1; int x2 = 2;
  13. short x3 = 3; char x4 = 4;
  14. proc(x1, &x1, x2, &x2, x3, &x3, x4, x4);
  15. return (x1+x2)*(x3+x4);
  16. }

使用如下命令进行编译和反编译:

  1. $ gcc -Og -fno-stack-protector -c call_proc.c
  2. $ objdump -d call_proc.o

其中-fno-stack-protector参数指示编译器不添加栈保护者机制

生成的汇编代码如下,这里我们仅看call_proc()中的栈空间分配

  1. 0000000000000015 <call_proc>:
  2. 15: 48 83 ec 10 sub $0x10,%rsp
  3. 19: 48 c7 44 24 08 01 00 movq $0x1,0x8(%rsp)
  4. 20: 00 00
  5. 22: c7 44 24 04 02 00 00 movl $0x2,0x4(%rsp)
  6. 29: 00
  7. 2a: 66 c7 44 24 02 03 00 movw $0x3,0x2(%rsp)
  8. 31: c6 44 24 01 04 movb $0x4,0x1(%rsp)
  9. 36: 48 8d 4c 24 04 lea 0x4(%rsp),%rcx
  10. 3b: 48 8d 74 24 08 lea 0x8(%rsp),%rsi
  11. 40: 48 8d 44 24 01 lea 0x1(%rsp),%rax
  12. 45: 50 push %rax
  13. 46: 6a 04 pushq $0x4
  14. 48: 4c 8d 4c 24 12 lea 0x12(%rsp),%r9
  15. 4d: 41 b8 03 00 00 00 mov $0x3,%r8d
  16. 53: ba 02 00 00 00 mov $0x2,%edx
  17. 58: bf 01 00 00 00 mov $0x1,%edi
  18. 5d: e8 00 00 00 00 callq 62 <call_proc+0x4d>
  19. ...

15行(我们具体以代码中给出的行号,其实这些数字应该是指令的起始位置,姑且就这样叫吧)中先将%rsp减去0x10,为4个局部变量共分配了16个字节的空间,并且在45和46行,程序将%rax和$0x4入栈,联系该函数的C语言程序和汇编程序中的具体操作,不难知,栈上的具体空间分配如下图所示:

图中,为了使栈字节对齐,4单独占用了一个8字节的空间,并且栈中的每一个类型的变量,都符合数据对齐的要求。

如果我们的参数8占用的字节数减少,会不会减少栈空间的占用呢?我们将上面的C语言程序的稍微改一改,如下:

  1. void proc(long a1, long *a1p,
  2. int a2, int *a2p,
  3. short a3, short *a3p,
  4. char a4, char a5) { // char *a4p改为了char a5
  5. *a1p += a1;
  6. *a2p += a2;
  7. *a3p += a3;
  8. a5 += a4;
  9. }
  10. long call_proc()
  11. {
  12. long x1 = 1; int x2 = 2;
  13. short x3 = 3; char x4 = 4;
  14. proc(x1, &x1, x2, &x2, x3, &x3, x4, x4); // 相应的改变了最后一个参数
  15. return (x1+x2)*(x3+x4);
  16. }

call_proc()的汇编如下:

  1. 000000000000000a <call_proc>:
  2. a: 48 83 ec 10 sub $0x10,%rsp
  3. e: 48 c7 44 24 08 01 00 movq $0x1,0x8(%rsp)
  4. 15: 00 00
  5. 17: c7 44 24 04 02 00 00 movl $0x2,0x4(%rsp)
  6. 1e: 00
  7. 1f: 66 c7 44 24 02 03 00 movw $0x3,0x2(%rsp)
  8. 26: 48 8d 4c 24 04 lea 0x4(%rsp),%rcx
  9. 2b: 48 8d 74 24 08 lea 0x8(%rsp),%rsi
  10. 30: 6a 04 pushq $0x4
  11. 32: 6a 04 pushq $0x4
  12. 34: 4c 8d 4c 24 12 lea 0x12(%rsp),%r9
  13. 39: 41 b8 03 00 00 00 mov $0x3,%r8d
  14. 3f: ba 02 00 00 00 mov $0x2,%edx
  15. 44: bf 01 00 00 00 mov $0x1,%edi
  16. 49: e8 00 00 00 00 callq 4e <call_proc+0x44>
  17. ...

对照程序,栈的空间结构编程的如下如所示:

我们发现,栈空间的占用并没有减少,为了能够达到栈字节对齐的目的,参数8和参数7各占一个8字节的空间,该过程调用浪费了1 + 7 + 7 = 15字节的空间。但为了兼容性和效率,这是值得的。

我们再看另一个程序,当我们在栈中分配字符串时又是怎样的呢?

  1. void function(int a, int b, int c) {
  2. char buffer1[5];
  3. char buffer2[10];
  4. strcpy(buffer2, buffer1);
  5. }
  6. void main() {
  7. function(1,2,3);

使用gcc -fno-stack-protector -o foo foo.cobjdump -d foo进行编译和反编译后,function()的汇编代码如下:

  1. 000000000000064a <function>:
  2. 64a: 55 push %rbp
  3. 64b: 48 89 e5 mov %rsp,%rbp
  4. 64e: 48 83 ec 20 sub $0x20,%rsp
  5. 652: 89 7d ec mov %edi,-0x14(%rbp)
  6. 655: 89 75 e8 mov %esi,-0x18(%rbp)
  7. 658: 89 55 e4 mov %edx,-0x1c(%rbp)
  8. 65b: 48 8d 55 fb lea -0x5(%rbp),%rdx
  9. 65f: 48 8d 45 f1 lea -0xf(%rbp),%rax
  10. 663: 48 89 d6 mov %rdx,%rsi
  11. 666: 48 89 c7 mov %rax,%rdi
  12. 669: e8 b2 fe ff ff callq 520 <strcpy@plt>
  13. 66e: 90 nop
  14. 66f: c9 leaveq
  15. 670: c3 retq

该过程共在栈上分配了32个字节的空间,其中包括两个字符串的空间和三个函数的参数的空间,这里需要提一下的是,尽管再x64下,函数的前6个参数直接用寄存器进行传递,但是有时候程序需要用到参数的地址,这个时候程序就不的不在栈上为参数分配内存并将参数拷贝到内存上,来满足程序对参数地址的操作。

联系程序,该过程的栈结构如下:

图中,因为char类型的地址可以从任意地址开始(地址为1的倍数),所以buffer1和buffer2是连续分配的,而三个int型变量则分配在了两个单独的8字节空间中。

小结

以上,我们看到,为了满足数据对齐和栈字节对齐的要求,或者说规范,编译器不惜牺牲了部分内存,这使得程序提高了兼容性,也提高了程序的性能。


参考:

x86_64 Linux 运行时栈的字节对齐的更多相关文章

  1. Java虚拟机运行时栈帧结构--《深入理解Java虚拟机》学习笔记及个人理解(二)

    Java虚拟机运行时栈帧结构(周志明书上P237页) 栈帧是什么? 栈帧是一种数据结构,用于虚拟机进行方法的调用和执行. 栈帧是虚拟机栈的栈元素,也就是入栈和出栈的一个单元. 2018.1.2更新(在 ...

  2. 深入理解java虚拟机(十) Java 虚拟机运行时栈帧结构

    运行时栈帧结构 栈帧(Stack Frame) 是用于虚拟机执行时方法调用和方法执行时的数据结构,它是虚拟栈数据区的组成元素.每一个方法从调用到方法返回都对应着一个栈帧入栈出栈的过程. 每一个栈帧在编 ...

  3. 计算机是如何计算的、运行时栈帧分析(神奇i++续)

    关于i++的疑问 通过JVM javap -c 查看字节码执行步骤了解了i++之后,衍生了一个问题: int num1=50; num1++*2执行的是imul(将栈顶两int类型数相乘,结果入栈), ...

  4. java虚拟机规范-运行时栈帧

    前言 java虚拟机是java跨平台的基石,本文的描述以jdk7.0为准,其他版本可能会有一些微调. 引用 java虚拟机规范 java虚拟机规范-运行时数据区 java内存运行时的栈帧结构 java ...

  5. 【转载】深入理解Java虚拟机笔记---运行时栈帧结构

    栈帧(Stack Frame)是用于支持虚拟机进行方法调用和方法执行的数据结构,它是虚拟机运行时数据区的虚拟机栈(Virtual Machine Stack)的栈元素.栈帧存储了方法的局部变量表,操作 ...

  6. linux 运行时加载不上动态库 解决方法(转)

    1. 连接和运行时库文件搜索路径到设置     库文件在连接(静态库和共享库)和运行(仅限于使用共享库的程序)时被使用,其搜索路径是在系统中进行设置的.一般 Linux 系统把 /lib 和 /usr ...

  7. cortexm内核 栈的8字节对齐及关键字PRESERVE8

    一.什么是栈对齐? 栈的字节对齐,实际是指栈顶指针须是某字节的整数倍.因此下边对系统栈与MSP,任务栈与PSP,栈对齐与SP对齐 这三对概念不做区分.另外下文提到编译器的时候,实际上是对编译器汇编器连 ...

  8. Intermediate_JVM 20180306 : 运行时数据区域

    Java比起C++一个很大的进步就在于Java不用再手动控制指针的delete与free,统一交由JVM管理,但也正因为如此,一旦出现内存溢出异常,不了解JVM,那么排查问题将会变成一项艰难的工作. ...

  9. Java运行时内存划分与垃圾回收--以及类加载机制基础

    ----JVM运行时内存划分----不同的区域存储的内容不同,职责因为不同1.方法区:被线程共享,存储被JVM加载的类的信息,常量,静态变量等2.运行时常量池:属于方法区的一部分,存放编译时期产生的字 ...

随机推荐

  1. Vue2.0仿饿了么webapp单页面应用

    Vue2.0仿饿了么webapp单页面应用 声明: 代码源于 黄轶老师在慕课网上的教学视频,我自己用vue2.0重写了该项目,喜欢的同学可以去支持老师的课程:http://coding.imooc.c ...

  2. java课堂_动手动脑4

    1.请运行以下示例代码StringPool.java,查看其输出结果.如何解释这样的输出结果?从中你能总结出什么? 答:在Java中,内容相同的字串常量(“Hello”)只保存一份以节约内存,所以s0 ...

  3. 【Java例题】1.1计算n的阶乘

    package study; import java.util.*; import java.math.*; public class myClass { public static void mai ...

  4. idea中的springboot项目如何不用重新编译,自动热部署

    两步走:引入依赖,配置idea 在pom.xml中引入如下依赖,关键字:devtools 第二步,修改idea两处配置 2.1 windows下,ctl+alt+s打开idea配置菜单 左上角输入框搜 ...

  5. byte数组和正数BigInteger之间的相互转换

    旧代码 public static void main(String[] args) { SecureRandom random = new SecureRandom(); byte[] key = ...

  6. Spark 系列(十四)—— Spark Streaming 基本操作

    一.案例引入 这里先引入一个基本的案例来演示流的创建:获取指定端口上的数据并进行词频统计.项目依赖和代码实现如下: <dependency> <groupId>org.apac ...

  7. Elasticsearch Lucene 数据写入原理 | ES 核心篇

    前言 最近 TL 分享了下 <Elasticsearch基础整理>https://www.jianshu.com/p/e8226138485d ,蹭着这个机会.写个小文巩固下,本文主要讲 ...

  8. 【模板】质数判断(Miller_Rabin)

    题意简述 给定一个范围N,你需要处理M个某数字是否为质数的询问(每个数字均在范围1-N内) 题解思路 费马小定理: n是一个奇素数,a是任何整数(\(1≤ a≤n-1\)) ,则\(a^{p-1}≡1 ...

  9. 循环while和for

    1.循环语句的基本操作 #while循环使用,其中break是用来结束当前循环的 count = 0 while True: print(count) count += 1 if count == 3 ...

  10. c# 将dwg文件转化为pdf

    https://blog.csdn.net/mywaster/article/details/50220379 最近做一个项目,要求将dwg文件转化为pdf,开发工具VS2010 + AutoCad ...