详解C语言在STM32中的内存分配问题

01、前言

不说废话,先上示例代码

uint8_t num_byte[4];
uint32_t num_word;
const uint32_t num_word_const = 0x1234;
uint32_t *point_heap;
int main(void)
{
  uint8_t num_byte_stack;
  static uint8_t num_byte_static;

  point_heap = (uint32_t *)malloc(4);
  *point_heap = 0x3421;
  free(point_heap);

  num_byte_stack = 0x11;

#pragma section = "CSTACK"
  char *pbeginstk = __section_begin("CSTACK");
#pragma section = "HEAP"
  char *pbeginheap = __section_begin("HEAP");    

  printf("CSTACK addr is 0x%x\r\n",pbeginstk);
  printf("HEAP addr is 0x%x\r\n",pbeginheap);

  printf("num_byte addr is 0x%x\r\n",&num_byte);
  printf("num_word addr is 0x%x\r\n",&num_word);
  printf("num_word_const addr is 0x%x\r\n",&num_word_const);
  printf("point_heap addr is 0x%x\r\n",&point_heap);
  printf("point_heap is 0x%x\r\n",point_heap);
  printf("num_byte_stack addr is 0x%x\r\n",&num_byte_stack);
  printf("num_byte_static addr is 0x%x\r\n",&num_byte_static);
}

打印如下

STACK addr is 0x20000320

HEAP addr is 0x20000720

num_byte addr is 0x20000308

num_word addr is 0x2000030c

num_word_const addr is 0x8002a44

point_heap addr is 0x20000310

point_heap is 0x20000728

num_byte_stack addr is 0x200006f8

num_byte_static addr is 0x20000318

先说结论:

num_byte、num_word、num_byte_static和point_heap存储在内部RAM中。

num_byte_stack存贮在栈中。

point_heap申请到的内存在堆中。

num_word_const在内部flash中。

如果是有同学对这个了然于胸,可以出门左转了,如果有些同学有兴趣,可以进一步往下看。

02、大小端

因为后面的内容涉及到大小端问题,这里先说下大小端问题。

大端(Big-endian):数据的高位字节存放在地址的低端低位字节存放在地址高端;

小端(Little-endian):数据的高位字节存放在地址的高端低位字节存放在地址低端;

例如:

数据0x12345678存储格式

大端格式

低地址<----0x12|0x34|0x56|0x78---->高地址

小端格式

低地址<----0x78|0x56|0x34|0x12---->高地址

其中的地址,一般由编译器分配,也可在程序中自行指定。从上表中,可以清晰的看到,大小端是以字节为单位进行数据储存的方式。大端通俗的理解就是赋值数从左自右;小端则是从右自左。

我们常用的X86结构是小端模式,而KEILC51则为大端模式。很多的ARM,DSP都为小端模式,本文使用的平台STM32F207就是小段模式。

03、逐步分析

如果有同学对这部分不是很熟悉,建议先看一下我之前的推文《C语言的内存分配》,先把C语言的堆栈,内存等概念先熟悉下。

先说关于堆栈的问题,下面代码可以打印出IAR平台下STM32的堆栈起始位置。

#pragma section = "CSTACK"
  char *pbeginstk = __section_begin("CSTACK");
#pragma section = "HEAP"
  char *pbeginheap = __section_begin("HEAP");

打印的结果如下

STACK addr is 0x20000320

HEAP addr is 0x20000720

这个地址是否正确,我们可以在IARdebug时,使用Disassembly窗口查看。

关于堆栈大小问题,如下

可以查到栈的终止位置是0x20000720,堆的终止位置是0x20000920。注意:这里计算牵扯到大小端的问题。

通过计算:

栈的大小=0x20000720-0x20000320=0x400。

堆的大小=0x20000920-0x20000720=0x200。

这和我们在IAR中的堆栈配置是一样的。

接下来就先说一下分配在内存的变量。

通过打印看出,num_byte、num_word、num_byte_static和point_heap并不在堆栈中,它们存储在内部RAM中。

使用Disassembly窗口查看如下

这也验证了static关键字,在修饰函数内的局部变量时,这个变量将和全局变量一样存储在内部ram中。

同时也说明了,STM32内部分配内存时候,是先分配全局变量(和static修饰的局部变量),再分配栈,最后再分配堆的。

对于栈的内存分配,局部变量,也就是num_byte_stack是存储在栈的范围内。

num_byte_stack addr is 0x200006f8

它的地址空间在栈中。因为在代码中num_byte_stack =0x11;使用Disassembly窗口查看到对应的地址数值是0x11。

关于栈,再说一句,栈不仅仅保存了局部变量,它会在函数切换,中断发生时保存现场,保存ARM内核的寄存器,这些不是这篇文章的讨论重点,这里先挖个坑,等以后有空再写篇文章专门说说这个部分。

堆的问题,简单来说:malloc申请的内存都在堆中。point_heap指针指向的内存地址就在堆的范围内。

point_heap is 0x20000728

代码中*point_heap= 0x3421;在Disassembly窗口查看到对应的地址数值是0x3421。

最后一个num_word_const,const修饰的变量是存储在内部flash中的,它的地址在内部flash范围内。

在代码中也有对应的赋值操作,constuint32_t num_word_const = 0x1234;在Disassembly窗口查看到对应的地址数值是0x1234。

点击查看本文所在的专辑,STM32F207教程

到此这篇关于C语言在STM32中的内存分配的文章就介绍到这了,更多相关C语言内存分配内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • C程序中Ubuntu、stm32的内存分配问题

    目录 一.内存分区概念介绍 1.1.C/C++编译程序的内存占用 1.2.栈和堆.全局/静态存储区和常量存储区的对比 1.3.图片说明  二.C语言编程论证 1.1.Ubuntu测试代码实现  1.2.STM32验证代码实现 1.3.keil下stm32存储观察 三.总结 四.参考资料 一.内存分区概念介绍 1.1.C/C++编译程序的内存占用 1.栈区(stack) 由编译器自动分配释放,存放函数的参数值,局部变量的值等.其操作方式类似于数据结构中的栈. 2.堆区(heap) 一般由程序员分配

  • 详解C语言在STM32中的内存分配问题

    01.前言 不说废话,先上示例代码 uint8_t num_byte[4]; uint32_t num_word; const uint32_t num_word_const = 0x1234; uint32_t *point_heap; int main(void) { uint8_t num_byte_stack; static uint8_t num_byte_static; point_heap = (uint32_t *)malloc(4); *point_heap = 0x3421;

  • 详解C语言-二级指针三种内存模型

    二级指针相对于一级指针,显得更难,难在于指针和数组的混合,定义不同类型的二级指针,在使用的时候有着很大的区别 第一种内存模型char *arr[] 若有如下定义 char *arr[] = {"abc", "def", "ghi"}; 这种模型为二级指针的第一种内存模型,在理解的时候应该这样理解:定义了一个指针数组(char * []),数组的每个元素都是一个地址. 在使用的时候,若要使用中间量操作元素,那么此时中间量应该定义为 char *tm

  • 详解C语言结构体中的char数组如何赋值

    目录 前景提示 一.char数组类型的处理 1.结构体初始化 2.结构体内数据赋值(简单法) 二.char数组指针类型的处理 1.结构体初始化 2.结构体内数据赋值 3.结构体内输出数据 三.全部代码 1.char数组 2.char数组指针 总结 前景提示 定义一个结构体,结构体中有两个变量,其中一个是char类型的数组,那么,怎么向这个数组中插入数据,打印数据呢? typedef struct SequenceList { // 数组的元素 char element[20]; // 数组的长度

  • 详解C语言结构体中的函数指针

    结构体是由一系列具有相同类型或不同类型的数据构成的数据集合.所以,标准C中的结构体是不允许包含成员函数的,当然C++中的结构体对此进行了扩展.那么,我们在C语言的结构体中,只能通过定义函数指针的方式,用函数指针指向相应函数,以此达到调用函数的目的. 函数指针 函数类型 (*指针变量名)(形参列表):第一个括号一定不能少. "函数类型"说明函数的返回类型,由于"()"的优先级高于"*",所以指针变量名外的括号必不可少.  注意指针函数与函数指针表示

  • 详解C语言面向对象编程中的封装

    目录 前言 一.面向对象基本概念 什么是对象? 对象与类 面向对象的编程方式 二.C语言实现面向对象 面向对象的三大特征 面向对象之封装 简介 代码实现–基础版 代码实现-进阶版 总结 前言 面向对象是一种思维方式,基本上用什么语言都是可以实现的.C语言的编程方式一般是面向过程的,但是也是可以实现面向对象的.对象是什么?什么又是面向对象?面向对象的三大特性又怎么实现,且听我细细道来. 一.面向对象基本概念 什么是对象? 此对象非彼对象,虽然有时候此对象又可以是你脑袋中的对象,那让我们从我们误解的

  • 详解C 语言项目中.h文件和.c文件的关系

    详解C 语言项目中.h文件和.c文件的关系 在编译器只认识.c(.cpp))文件,而不知道.h是何物的年代,那时的人们写了很多的.c(.cpp)文件,渐渐地,人们发现在很多.c(.cpp)文件中的声明语句就是相同的,但他们却不得不一个字一个字地重复地将这些内容敲入每个.c(.cpp)文件.但更为恐怖的是,当其中一个声明有变更时,就需要检查所有的.c(.cpp)文件. 于是人们将重复的部分提取出来,放在一个新文件里,然后在需要的.c(.cpp)文件中敲入#include XXXX这样的语句.这样即

  • 详解Go语言中关于包导入必学的 8 个知识点

    1. 单行导入与多行导入 在 Go 语言中,一个包可包含多个 .go 文件(这些文件必须得在同一级文件夹中),只要这些 .go 文件的头部都使用 package 关键字声明了同一个包. 导入包主要可分为两种方式: 单行导入 import "fmt" import "sync" 多行导入 import( "fmt" "sync" ) 如你所见,Go 语言中 导入的包,必须得用双引号包含,在这里吐槽一下. 2. 使用别名 在一些场

  • 详解R语言中生存分析模型与时间依赖性ROC曲线可视化

    R语言简介 R是用于统计分析.绘图的语言和操作环境.R是属于GNU系统的一个自由.免费.源代码开放的软件,它是一个用于统计计算和统计制图的优秀工具. 人们通常使用接收者操作特征曲线(ROC)进行二元结果逻辑回归.但是,流行病学研究中感兴趣的结果通常是事件发生时间.使用随时间变化的时间依赖性ROC可以更全面地描述这种情况下的预测模型. 时间依赖性ROC定义 令 Mi为用于死亡率预测的基线(时间0)标量标记. 当随时间推移观察到结果时,其预测性能取决于评估时间 t.直观地说,在零时间测量的标记值应该

  • 详解R语言中的多项式回归、局部回归、核平滑和平滑样条回归模型

    在标准线性模型中,我们假设 .当线性假设无法满足时,可以考虑使用其他方法. 多项式回归 扩展可能是假设某些多项式函数, 同样,在标准线性模型方法(使用GLM的条件正态分布)中,参数  可以使用最小二乘法获得,其中  在  . 即使此多项式模型不是真正的多项式模型,也可能仍然是一个很好的近似值 .实际上,根据 Stone-Weierstrass定理,如果  在某个区间上是连续的,则有一个统一的近似值  ,通过多项式函数. 仅作说明,请考虑以下数据集 db = data.frame(x=xr,y=y

  • 详解R语言中的表达式、数学公式、特殊符号

      在R语言的绘图函数中,如果文本参数是合法的R语言表达式,那么这个表达式就被用Tex类似的规则进行文本格式化. y <- function(x) (exp(-(x^2)/2))/sqrt(2*pi) plot(y, -5, 5, main = expression(f(x) == frac(1,sqrt(2*pi))*e^(-frac(x^2,2))), lwd = 3, col = "blue") library(ggplot2) x <- seq(0, 2*pi, b

随机推荐