Keil开发的ARM程序main函数之前的汇编分析

Keil开发的ARM程序main函数之前的汇编分析

——BIN文件中RW段的数据移动

系统平台:

STM32系列STM32F103ZE,512KB内部FLASH,64KB片内存储;

FLASH地址范围0x0800 0000 ~ 0x0808 0000,用于存放代码;

片内存储地址范围0x2000 0000 ~ 0x2001 0000,用于存放数据;

 

Cortex-M3上电后来到复位中断(已将前4个字节的值存入MSP堆栈指针),转到__main标号,完成RW段的移动、ZI段的初始化,建立堆栈,初始化库函数,然后跳转到main函数,开启C程序之旅,执行流程如图1所示。

图1 main函数之前的汇编程序执行流程图

 

本文主要讨论RW段的移动,RW段就是程序中赋了初值的变量,它的搬移我看到过两种方式。在BIN文件中,RO段和RW段之间有8个双字的Region$$Table,4个双字一组,分别用于完成RW段的搬移和ZI段的初始化。

(1)       __scatterload_copy来完成

此时RW段的内容保存到内存开始的地方,本文中是0x20000000,用这一方式完成后,内存中存放的不是RW数据的内容,而是一个地址。这个地址是在FLASH中,即指向了其在RO段的地址,实际的内容是在RO段中。

(2)       通过__uncompressed1实现

RW是程序中初始化的变量,但是这些变量有可能初值是0,因此为了节省空间,实际在BIN文件中RW段是压缩过的。调用__uncompressed1解压缩RW段的数据内容,并将其保存到内存开始的地方。

 

图2 BIN文件中压缩RW段内容

   图2是BIN文件中RW段的数据内容,影印部分显示,大小是164字节。其中0x0001 C72C前面8个双字的内容是Region$$Table,将其列出如下。

0x0801 C72C         BIN文件中RW段的开始地址

0x2000 0000          RW段要存放到RAM中的地址

0x0000 0334          要存放到RAM中的RW段数据大小

0x0800 0184          执行函数__scatterload_copy或者__uncompressed1的地址

上面4个双字完成RW段的搬移。

0x0801 C7D0        BIN文件的末尾,ZI段的开始

0x2000 0334          ZI段放到RAM中的起始位置

0x0000 F4BC        ZI段的大小

0x0800 01E0         执行函数__scatterload_zeroinit的地址

这4个双字完成ZI段的初始化。

 

图3 RW段内容解压缩后在RAM中排布

将BIN文件中RW解压缩以后(这个解压缩从反汇编可以看到,但是没有看明白),实际内容如图3所示,大小变为820个字节,该补0的地方都已经补足了。

posted on   wangyw  阅读(2963)  评论(0编辑  收藏  举报

编辑推荐:
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 基于Microsoft.Extensions.AI核心库实现RAG应用
· Linux系列:如何用heaptrack跟踪.NET程序的非托管内存泄露
· 开发者必知的日志记录最佳实践
· SQL Server 2025 AI相关能力初探
阅读排行:
· winform 绘制太阳,地球,月球 运作规律
· 震惊!C++程序真的从main开始吗?99%的程序员都答错了
· AI与.NET技术实操系列(五):向量存储与相似性搜索在 .NET 中的实现
· 超详细:普通电脑也行Windows部署deepseek R1训练数据并当服务器共享给他人
· 【硬核科普】Trae如何「偷看」你的代码?零基础破解AI编程运行原理
< 2025年3月 >
23 24 25 26 27 28 1
2 3 4 5 6 7 8
9 10 11 12 13 14 15
16 17 18 19 20 21 22
23 24 25 26 27 28 29
30 31 1 2 3 4 5

导航

统计

点击右上角即可分享
微信分享提示