C编译器剖析_1.5 结合C语言来学汇编_指针、数组和结构体

本文通过分析C代码及其编译后的汇编代码,探讨了C语言中指针、数组和结构体在内存中的表示和初始化过程。举例说明了编译器如何处理数组初始化、函数调用以及结构体成员赋值。同时,解释了C语言中函数名和数组名的特殊性,以及它们在汇编层面上的体现。通过对简单的C程序的剖析,展示了如何从C代码到汇编代码的转换,为理解编译器工作原理奠定了基础。
摘要由CSDN通过智能技术生成

    让我们再来看一份C代码,及其经UCC编译器编译后产生的主要汇编代码,如图1.33所示,其中包含了数组、指针和结构体。


图1.33 数组、指针和结构体

     按照C的语义,图1.33第9行的C代码是对局部数组number的初始化,需要把number[0]初始化为2015,而数组中的其他元素皆被初始化为0。UCC编译器采取的翻译方法是:先调用memset函数来把数组number所占的内存空间清0,然后再把number[0]设为2015,如图1.33的第17至24行所示。C库函数memset的API如下所示:

void *memset(void *s, int c, size_t n);

    按照“C调用约定 Calling Convention”,参数需要从右到左入栈,即先要把n入栈,再把c入栈,之后才是目标地址s。第17行的$64对应参数n,表示数组number所占的内存大小为64字节,因为每个int占4字节,而number共有16个整数;而第18行的$0对应参数c,表示我们要把指针s所指的大小为n字节的内存全部设为0;第19行用于取内存单元-64(%ebp)的地址,第20行把这个地址入栈,即对应参数s,而-64(%ebp)所对应的内存空间正是UCC编译器为局部数组number在栈中分配的存储空间。第21行完成了对库函数memset的调用,第22行用于把刚才入栈的3个参数出栈,此处每个参数正好都占4个字节,所以总共占了12字节的栈空间。按照“C调用约定”,这个退栈操作需要由主调函数来完成。实际上,第17至22行也演示了如何在汇编语言中调用C库函数。

    而与第10行” ptr = &number[1];”对应的汇编代码为第25至27行,第25行把number[0]的地址存到寄存器eax中,因为number[1]与number[0]之间隔了一个占4字节的整数,所以第26行对eax进行了加4的操作,这样eax中的内容就是number[1]的地址,第27行把这个地址存到了全局变量ptr所对应的内存单元中。

    与第11行” *ptr = 2016;”对应的汇编代码在图1.33的第29至30行,第29行通过movl指令把内存单元

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值