0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

C语言在STM32中的内存分配

GReq_mcu168 来源:CSDN技术社区  作者:CSDN技术社区  2022-02-10 14:57 次阅读

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
",pbeginstk);  printf("HEAP addr is 0x%x
",pbeginheap);    printf("num_byte addr is 0x%x
",&num_byte);  printf("num_word addr is 0x%x
",&num_word);  printf("num_word_const addr is 0x%x
",&num_word_const);  printf("point_heap addr is 0x%x
",&point_heap);  printf("point_heap is 0x%x
",point_heap);  printf("num_byte_stack addr is 0x%x
",&num_byte_stack);  printf("num_byte_static addr is 0x%x
",&num_byte_static);}

打印如下

STACK addr is 0x20000320HEAP addr is 0x20000720num_byte addr is 0x20000308num_word addr is 0x2000030cnum_word_const addr is 0x8002a44point_heap addr is 0x20000310point_heap is 0x20000728num_byte_stack addr is 0x200006f8num_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---->高地址

C语言在STM32中的内存分配

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

我们常用的X86结构是小端模式,而KEILC51则为大端模式。很多的ARMDSP都为小端模式,本文使用的平台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 0x20000320HEAP addr is 0x20000720

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

C语言在STM32中的内存分配

关于堆栈大小问题,如下

C语言在STM32中的内存分配

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

通过计算:

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

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

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

C语言在STM32中的内存分配

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

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

使用Disassembly窗口查看如下

C语言在STM32中的内存分配

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

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

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

num_byte_stack addr is 0x200006f8

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

C语言在STM32中的内存分配

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

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

point_heap is 0x20000728

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

C语言在STM32中的内存分配

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

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

C语言在STM32中的内存分配

原文标题:C语言在STM32中的内存分配

文章出处:【微信公众号:硬件攻城狮】欢迎添加关注!文章转载请注明出处。

审核编辑:汤梓红


声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 内存
    +关注

    关注

    8

    文章

    3019

    浏览量

    73998
  • STM32
    +关注

    关注

    2270

    文章

    10895

    浏览量

    355697
  • C语言
    +关注

    关注

    180

    文章

    7604

    浏览量

    136665

原文标题:C语言在STM32中的内存分配

文章出处:【微信号:mcu168,微信公众号:硬件攻城狮】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    C语言中申请的堆内存能不能自动释放

    C语言中申请的堆内存能不能自动释放?每次都要手动 free 太麻烦,也容易忘记。 学过 C++ 的同学,应该首先能想到智能指针。 但是这是C
    的头像 发表于 11-27 09:33 103次阅读

    解读版|Air780E软件C语言内存数组的神秘面纱!

    今天我们来揭开Air780E 软件 C 语言内存数组的神秘面纱,希望有所收获。
    的头像 发表于 11-17 10:00 224次阅读
    解读版|Air780E软件<b class='flag-5'>中</b><b class='flag-5'>C</b><b class='flag-5'>语言</b><b class='flag-5'>内存</b>数组的神秘面纱!

    C语言指针学习笔记

    本文从底层内存分析,彻底让读者明白C语言指针的本质。
    的头像 发表于 11-05 17:40 226次阅读
    <b class='flag-5'>C</b><b class='flag-5'>语言</b>指针学习笔记

    C语言与Java语言的对比

    C语言和Java语言都是当前编程领域中的重要成员,它们各自具有独特的优势和特点,适用于不同的应用场景。以下将从语法特性、内存管理、跨平台性、性能、应用领域等多个方面对
    的头像 发表于 10-29 17:31 314次阅读

    转载 golang内存分配

    . 线程拥有一定的 cache, 可用于无锁分配. 同时 Go 对于 GC 后回收的内存页, 并不是马上归还给操作系统, 而是会延迟归还, 用于满足未来的内存需求.    1.
    的头像 发表于 09-05 14:12 253次阅读
    转载 golang<b class='flag-5'>内存</b><b class='flag-5'>分配</b>

    ESP32S3+LVGL创建一个界面,请问能只SPIRAM分配内存,IRAM不分配吗?

    各位前辈好。ESP32S3+LVGL的开发的过程中发现,创建一个界面,会同时SPIRAM和IRAM分配相同大小的内存。请问能只SPIRAM分配
    发表于 06-06 07:45

    FreeRTOS如何在中断调用内存分配函数?

    最近在玩FreeRTOS,遇到一个问题,就是不知如何在中断调用内存分配函数。pvPortMalloc函数中会调用xTaskResumeAll,而这个函数不能再中断调用,所以请问中断
    发表于 05-08 08:25

    你知道吗? 51单片机也有动态内存分配

    一、简述其实在51单片机也可以使用动态内存,动态内存其实就是划出一块内存区域,将这块内存进行管理,称为
    的头像 发表于 04-26 08:10 1501次阅读
    你知道吗? 51单片机也有动态<b class='flag-5'>内存</b><b class='flag-5'>分配</b>

    浅谈C语言内存分区和STM32存储器分配

    RAM是与CPU直接交换数据的内部存储器,也叫主存(内存)。 它可以随时读写,而且速度很快,通常作为操作系统或其他正在运行的程序的临时数据存储媒介。 当电源关闭时RAM不能保留数据(掉电数据消失哦)如果需要保存数据,就必须把它们写入一个长期的存储设备
    的头像 发表于 04-02 09:50 1307次阅读
    浅谈<b class='flag-5'>C</b><b class='flag-5'>语言</b><b class='flag-5'>内存</b>分区和<b class='flag-5'>STM32</b>存储器<b class='flag-5'>分配</b>

    C语言内存泄漏问题原理

    内存泄漏问题只有使用堆内存的时候才会出现,栈内存不存在内存泄漏问题,因为栈内存会自动
    发表于 03-19 11:38 520次阅读
    <b class='flag-5'>C</b><b class='flag-5'>语言</b><b class='flag-5'>内存</b>泄漏问题原理

    C语言的指针用法

    C语言编程善用指针可以简化一些任务的处理,而对于一些任务(比如动态内存分配),必须要有指针才行的。也就是说精通
    发表于 03-05 14:22 346次阅读
    <b class='flag-5'>C</b><b class='flag-5'>语言</b>的指针用法

    C语言中的动态内存管理讲解

    本章将讲解 C 的动态内存管理。C 语言内存分配
    的头像 发表于 02-23 14:03 388次阅读
    <b class='flag-5'>C</b><b class='flag-5'>语言</b>中的动态<b class='flag-5'>内存</b>管理讲解

    Linux内核内存管理之内核非连续物理内存分配

    我们已经知道,最好将虚拟地址映射到连续页帧,从而更好地利用缓存并实现更低的平均内存访问时间。然而,如果对内存区域的请求并不频繁,那么考虑基于通过连续线性地址访问非连续页帧的分配方案是有意义的。该模式
    的头像 发表于 02-23 09:44 943次阅读
    Linux内核<b class='flag-5'>内存</b>管理之内核非连续物理<b class='flag-5'>内存</b><b class='flag-5'>分配</b>

    Linux内核内存管理之ZONE内存分配

    内核中使用ZONE分配器满足内存分配请求。该分配器必须具有足够的空闲页帧,以便满足各种内存大小请求。
    的头像 发表于 02-21 09:29 890次阅读

    C语言:指针内存是如何存放变量

    程序定义一个变量,那么程序编译的过程,系统会根据你定义变量的类型来分配「相应尺寸」的内存
    发表于 01-08 10:14 457次阅读