单片机内存管理和启动文件

server/2024/11/28 17:39:42/

一、常见存储器介绍

FLASH又称为闪存,不仅具备电子可擦除可编程(EEPROM)的性能,还不会断电丢失数据同时可以快速读取数据,U盘和MP3里用的就是这种存储器。在以前的嵌入式芯片中,存储设备一直使用ROM(EPROM),随着技术的进步,现在嵌入式中基本都是FLASH,用作存储Bootloader以及操作系统或者程序代码或者直接当硬盘使用(U盘)。

RAM随机存储器(Random Access Memory)表示既可以从中读取数据,也可以写入数据。当机器电源关闭时,存于其中的数据就会丢失。比如电脑的内存条。

RAM有两大类,一种称为静态RAM(Static RAM/SRAM),SRAM速度非常快,是目前读写最快的存储设备了,但是它也非常昂贵,所以只在要求很苛刻的地方使用,譬如CPU的一级缓冲,二级缓冲。另一种称为动态RAM(Dynamic RAM/DRAM),DRAM保留数据的时间很短,速度也比SRAM慢,不过它还是比任何的ROM都要快,但从价格上来说DRAM相比SRAM要便宜很多,计算机内存就是DRAM的。

二、STM32 的存储器映射分析

STM32存储器映射表(选用的是STM32F103VE的,不同的型号Flash 和 SRAM 的地址空间不同,起始地址都是一样的):

STM32 的内存管理主要就是对0X0800 0000 开始的 Flash 部分 和0x2000 0000 开始的 SRAM 部分使用管理

三、C/C++ 程序编译后的存储数据段

.data

数据段,储存已初始化且不为0的全局变量和静态变量(全局静态变量和局部静态变量)。
static声明的变量放在data段。

需要在 RAM里面运行,但是起初需要保存在 Flash里面,程序运行后复制到 RAM里面运行,需要占用Flash空间

.BSS

Block Started by Symbol。储存未初始化的,或初始化为0的全局变量和静态变量。
BSS段属于静态内存分配,所以放在RAM里。

.text(CodeSegment/Text Segment)

代码段,储存程序代码。也就是存放CPU执行的机器指令(machineinstructions)。这部分区域的大小在程序运行前就已经确定,并且内存区域通常属于只读(某些架构也允许代码段为可写,即允许修改程序)。
在代码段中,也有可能包含一些只读的常数变量,例如字符串常量等。
放在Flash里。

.constdata

储存只读常量。const修饰的常量,不管是在局部还是全局放在Flash 里。
所以为了节省 RAM,把常量的字符串,数据等 用const声明

heap(堆)

堆是用于存放进程运行中被动态分配的内存段。他的大小并不固定,可动态扩张或者缩减,由程序员使用malloc()和free()函数进行分配和释放。当调用malloc等函数分配内存时,新分配的内存就被动态添加到堆上(堆被扩张);当利用free等函数释放内存时,被释放的内存从堆中被剔除(堆被缩减)。
放在RAM里
其可用大小定义在启动文件startup_stm32fxx.s中。

stack(栈)

栈又称堆栈,是用户存放程序临时创建的局部变量,由系统自动分配和释放。可存放局部变量、函数的参数和返回值(但不包括static声明的变量,static意味着 放在 data 数据段中)。

 

四、单片机启动过程

  1. 设置堆栈指针 SP = _initial_sp
  2. 设置PC指针 = Reset_Handler
  3. 配置系统时钟SystemInit
  4. 配置外部 SRAM 用于程序变量等数据存储(可选)
  5. 调用C库的 _main 函数,最终调用main函数

五、start_xxx.s启动文件讲解

5.1 开辟栈空间和堆空间

代码的开始,就是开辟栈空间,用于局部变量,函数调用,函数参数等。

接下来是开辟堆空间,主要用于动态内存分配,使用malloc,calloc等函数分配的变量空间是在堆上的。

5.2 中断向量表部分

5.3 Reset_Handler 系统启动

系统上电或者复位后首先执行的代码就是复位中断服务函数 Reset_Handler:

图中的 Reset_Handler 中断服务函数使用了WEAK,说明在外部可以自定义Reset_Handler 函数
PROC、ENDP这一对伪指令把程序分为若干个过程,是程序结构更加清晰
_main 标号表示 C/C++标准实时库函数里的一个初始化子程序 _main的入口地址。该程序的一个主要作用是初始化堆栈(跳转_user_initial_stackheap标号进行初始化堆栈),并初始化映像文件,最后跳转到C程序中的main函数。这也正解释了为什么所有的C程序必须有一个main函数作为程序的起点,因为这是由C/C++标准实时库所规定的。


http://www.ppmy.cn/server/137694.html

相关文章

Gitee push 文件

1、背景 想将自己的plecs仿真放到git中管理,以防丢失,以防乱改之后丢失之前版本仿真。此操作说明默认用户已下载git。 2、操作步骤 2.1 开启Git Bash 在文件夹中右键,开启Git Bash。 2.2 克隆文件 在Git Bash中打git clone git地址&#…

如何在算家云搭建GFP-GAN(图像生成)

一、GFP-GAN简介 GFP-GAN是腾讯在人像复原、超分等方面的佳作 ,其基于 FFHQ 上训练,由 70000 张高质量图像组成。在训练过程中,将所有图像的大小调整为5122。 可用于真实世界面部高清修复。 更多详细信息见Github仓库 二、模型搭建流程 1…

MySQL 数据库备份与恢复全攻略

MySQL 数据库备份与恢复全攻略 引言 在现代应用中,数据库是核心组件之一。无论是个人项目还是企业级应用,数据的安全性和完整性都至关重要。为了防止数据丢失、损坏或意外删除,定期备份数据库是必不可少的。本文将详细介绍 MySQL 数据库的备…

基于hive分析Flask为后端框架echarts为前端框架的招聘网站可视化大屏项目

基于hive分析Flask为后端框架echarts为前端框架的招聘网站可视化大屏项目 1. 项目概述 项目目标是构建一个大数据分析系统,包含以下核心模块: 1、数据爬取:通过request请求获取猎聘网的就业数据。 2、数据存储和分析:使用 Hive …

数据分析-36-时间序列分解之互补集合经验模态分解CEEMD

文章目录 1 时间序列模态分解1.1 模态分解的概念1.2 模态分解的作用1.3 常用的模态分解方法1.4 模态分解的常用库2 互补集合经验模态分解CEEMD2.1 CEEMD的流程2.2 加载数据集2.2.1 数据重采样2.2.2 原始数据可视化2.3 互补集合经验模态分解CEEMD2.3.1 自定义函数my_ceemd2.3.2 …

《8.3.2 前向分步算法与 AdaBoost》最小α公式如何通过简化得到的

本文是将文章《8.3.2 前向分步算法与 AdaBoost》中的公式单独拿出来做一个详细的解析,便于初学者更好的理解。 α m ∗ 1 2 log ⁡ 1 − e m e m \alpha_m^* \frac{1}{2} \log \frac{1 - e_m}{e_m} αm∗​21​logem​1−em​​ 我们从公式 ( 8.22 ) (8.22) (8.2…

LeetCode23:合并K个升序链表

原题地址:. - 力扣(LeetCode) 题目描述 给你一个链表数组,每个链表都已经按升序排列。 请你将所有链表合并到一个升序链表中,返回合并后的链表。 示例 1: 输入:lists [[1,4,5],[1,3,4],[2,6]] …

【每日C/C++问题】

一、C/C中数组定义和初始化的方式有哪些? int arr[100]; // 定义了数组arr,并未对数组进行初始化int arr[100] {1, 2}; // 定义并初始化了数组arr前两个元素,其他元素为0int arr[3] {1, 2, 3}; // 定义并初始化了数组arr所有元素int arr[]…