RT-Thread memheap 开启多块 SRAM的方法

news2024/12/22 19:01:33

验证环境

  • NUCLEO-L476RG 开发板,板载 STM32L476RGT6(96K SARM1 + 32K SRAM2)

  • Win10 64 位

  • Keil MDK 5.36

  • RT-Thread 5.0.1 版本(2023-05-28 master 主线)

功能描述

  • 最近在研究 RT-Thread 内存的管理,熟悉了一下 memheap 的功能实现,并且了解到, memheap 支持多块内存(物理地址不连续)的管理,当开启 memheap 后,rt_malloc 会遍历所有注册的 heap 内存块,并且进行 内存的申请与释放。

  • 当前 STM32L476RGT6 支持两块 SRAM,其中 SRAM1 96KB,还有一块 SRAM2 32KB,SRAM2 默认没有使用,如何开启 SRAM2 呢?

开启 memheap

  • STM32L476RGT6 SRAM 总共:128KB,其中 SRAM1 默认开启, SRAM2 默认没有开启

  • 首先 RT-Thread 开启 memheap

  • 通过 RT-Thread ENV 工具: menuconfig,配置使能 memheap

  • 当前测试的 BSP 为:stm32l476-st-nucleo

在这里插入图片描述
在这里插入图片描述

  • 这里注意,如果让 rt_malloc 自动在多个 Heap 内存池中申请,需要开启 RT_USING_MEMHEAP_AUTO_BINDING,也就是 勾选 [*] Use all of memheap objects as heap

  • 配置使能 memheap 后,编译下载,通过 msh 命令 free 查看

msh >free
memheap   pool size  max used size available size
-------- ---------- ------------- --------------
heap     98304      7232          91072
  • 发现 SRAM1 96KB 正常的初始化了

开启 SRAM2

  • SRAM2 需要手动初始化,首先可以在 board.h 中增加 SRAM2 的内存信息:起始地址、大小
#define HEAP_SRAM2_BEGIN                (0x10000000)
#define HEAP_SRAM2_SIZE                 (32 * 1024)
  • 需要修改 board.c ,增加 SRAM2 的 memheap 的初始化操作
int system_sram2_init(void)
{
    static struct rt_memheap memheap_sram2;
    /* Heap initialization */
#if defined(RT_USING_HEAP)
    rt_memheap_init(&memheap_sram2, "sram2", (void *)HEAP_SRAM2_BEGIN, (rt_size_t)HEAP_SRAM2_SIZE);
#endif
    return 0;
}
INIT_BOARD_EXPORT(system_sram2_init);
  • 这里使用 RT-Thread 自动初始化机制,初始化为 board 级别。

功能验证

  • RT-Thread ENV scons --target=mdk5,可以刷新 Keil MDK5 的工程

  • 使用 Keil MDK5 打开工程,编译,下载到开发板,连接开发板串口,可以查看 SRAM2 正常初始化成功

memheap   pool size  max used size available size
-------- ---------- ------------- --------------
sram2    32768      48            32720
heap     98304      7232          91072
  • 也就是 sram2 存在 memheap 的列表中了

内存申请

  • rt_malloc 底层由 memheap 实现后,并且使能 RT_USING_MEMHEAP_AUTO_BINDING,使用 rt_malloc 会自动在所有 heap 中 操作

  • 测试用例:内存申请与释放

#define MEMHEAP_BLOCK_NUM       64
static void *buf[MEMHEAP_BLOCK_NUM] = { 0 };

void memheap_alloc_test(void)
{
    for (int i = 0; i < MEMHEAP_BLOCK_NUM; i++)
    {
        buf[i] = rt_malloc(3 * 1024 - 24);
        if (!buf[i])
        {
            rt_kprintf("malloc failed, index = %d\n", i);
            return;
        }
    }
}

MSH_CMD_EXPORT(memheap_alloc_test, memheap_alloc_test);

void memheap_free_test(void)
{
    for (int i = 0; i < MEMHEAP_BLOCK_NUM; i++)
    {
        if (buf[i])
        {
            rt_memheap_free(buf[i]);
        }
    }
}
MSH_CMD_EXPORT(memheap_free_test, memheap_free_test);
  • 测试结果

在这里插入图片描述

  • 发现 rt_malloc 自动在 新增加的 sram2 中申请了内存

  • free 申请的内存,发现内存可用大小恢复了

专用 heap 内存

  • 如果不使能 RT_USING_MEMHEAP_AUTO_BINDING, 新注册的 memheap sram2,不会被系统 rt_malloc 使用到,需要用户自己定义 内存申请与释放的函数进行 专用内存的管理

  • 大概的管理思路如下:

int system_sram2_init(void)
{
    return rt_memheap_init(&memheap_sram2, "sram2", (void *)HEAP_SRAM2_BEGIN, (rt_size_t)HEAP_SRAM2_SIZE);
}
INIT_BOARD_EXPORT(system_sram2_init);

void *user_alloc(rt_size_t size)
{
    return rt_memheap_alloc(&memheap_sram2, size);
}

void user_free(void *ptr)
{
    rt_memheap_free(ptr);
}

void user_alloc_test(void)
{
    for (int i = 0; i < MEMHEAP_BLOCK_NUM; i++)
    {
        user_ptr[i] = user_alloc(1024);
        if (!user_ptr[i])
        {
            rt_kprintf("malloc failed, index = %d\n", i);
            return;
        }
    }
}
MSH_CMD_EXPORT(user_alloc_test, user_alloc_test);

void user_free_test(void)
{
    for (int i = 0; i < MEMHEAP_BLOCK_NUM; i++)
    {
        if (user_ptr[i])
        {
            user_free(user_ptr[i]);
        }
    }
}
MSH_CMD_EXPORT(user_free_test, user_free_test);
  • 这样 user_allocuser_free 只会操作指定的 memheap

在这里插入图片描述

  • 【备注】如果同时开启了 RT_USING_MEMHEAP_AUTO_BINDING,并且又自定义了某个 memheap的 内存申请与释放操作,rt_malloc 常规内存申请,依旧有可能 申请这个 特定的 memheap 的内存

小结

  • 经过测试发现,一直申请某个大小的 memheap 内存并且不释放,会出现 hardfault 死机问题,后面抽时间研究一下死机的原因。

  • 以下:每次申请 512 字节内存,62次左右的时候,死机了。
    在这里插入图片描述

  • 可以通过 RT-Thread memheap,把几块物理上地址不连续的内存管理起来,统一使用 rt_malloc 、rt_free 等 内存操作接口进行操作,也可以独立管理各个 memheap 内存块,不过需要用户自己实现相应的 内存申请与释放接口

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/579980.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Linux内核源码分析 2:Linux内核版本号和源码目录结构

一、Linux的版本 1. 稳定版和开发版 Linux内核主要分为两种版本&#xff1a; 稳定版&#xff08;长期支持版&#xff09;&#xff1a;稳定版的内核具有工业级的强度&#xff0c;可以广泛地应用和部署。而每一代新推出的稳定版内核大部分都只是修正了一些Bug或是加入了一些新的…

【网络协议详解】——FTP系统协议(学习笔记)

目录 &#x1f552; 1. 概述&#x1f552; 2. 工作原理&#x1f558; 2.1 两个连接 &#x1f552; 3. 相关命令与处理&#x1f558; 3.1 接入命令&#x1f558; 3.2 文件管理命令&#x1f558; 3.3 数据格式化命令&#x1f558; 3.4 端口定义命令&#x1f558; 3.5 文件传输命令…

计算机组成原理 期末复习笔记

&#x1f331;博客主页&#xff1a;大寄一场. &#x1f618;博客制作不易欢迎各位&#x1f44d;点赞⭐收藏➕关注 目录 前言 第一章 计算机系统概论计算机软件的发展 计算机硬件的基本组成 计算机系统的层次结构 计算机的性能指标 第二章 数据表示 与 第三章 数据运算与运…

Go语言实现JDBC

Go语言操作数据库 Go语言提供了关于数据库的操作,包下有sql/driver 该包用来定义操作数据库的接口&#xff0c;这保证了无论使用哪种数据库&#xff0c;操作方式都是相同的; 准备工作: 下载驱动 需要在代码所在文件夹下执行相应的命令 go get github.com/go-sql-driver/mys…

DAY 69 rsync远程同步

rsync介绍 rsync简介 rsync&#xff08;Remote Sync&#xff0c;远程同步&#xff09;是一个开源的快速备份工具&#xff0c;可以在不同主机之间镜像同步整个目录树&#xff0c;支持增量备份&#xff0c;并保持链接和权限&#xff0c;且采用优化的同步算法&#xff0c;传输前…

超超超级详细的画图以及代码分析各种排序的实现!

各种排序的实现 排序的概念直接插入排序基本思想实现直接插入排序的特性总结 希尔排序基本思想实现希尔排序的特性总结 简单选择排序基本思想实现直接选择排序的特性总结 堆排序实现堆排序的特性总结 冒泡排序基本思想实现冒泡排序的特性总结 快速排序基本思想hoare版本挖坑法前…

JDBC测试

JDBC是什么? JDBC是一套接口,各大厂商来实现这套接口&#xff0c;进行数据库连接操作 比如Mysql驱动,Oracle驱动,sqlServer驱动,高斯驱动 以Mysql为例: JDBC编程六步 第一步&#xff1a;注册驱动 第二步&#xff1a;获取连接 第三步&#xff1a;获取数据库操作对象 第…

为什么不用Go开发操作系统?

操作系统 (OS) 是计算机系统的心脏和灵魂&#xff0c;它管理着计算机的硬件和软件资源&#xff0c;并为用户提供与计算机交互的方式。传统上&#xff0c;C 和 Assembly 等语言因其低开销和 “接近机器码” 的特性而被用于开发操作系统。 但诸如 Go 等高级语言的兴起引入了一些…

黑客为什么不攻击赌博网站?

攻击了&#xff0c;只是你不知道而已&#xff01; 同样&#xff0c;对方也不会通知你&#xff0c;告诉你他黑了赌博网站。 攻击赌博网站的不一定是正义的黑客&#xff0c;也可能是因赌博输钱而误入歧途的法外狂徒。之前看过一个警方破获的真实案件&#xff1a;28岁小伙因赌博…

Xubuntu22.04之替换blueman-manager连接蓝牙设备(一百七十五)

简介&#xff1a; CSDN博客专家&#xff0c;专注Android/Linux系统&#xff0c;分享多mic语音方案、音视频、编解码等技术&#xff0c;与大家一起成长&#xff01; 优质专栏&#xff1a;Audio工程师进阶系列【原创干货持续更新中……】&#x1f680; 人生格言&#xff1a; 人生…

【C++】static在类中修饰成员变量成员函数

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 一、定义&#xff1a;二、特性&#xff1a;1. 静态成员为所有类对象所共享&#xff0c;不属于某个具体的对象&#xff0c;存放在静态区2. 静态成员变量必须在类外定…

C++学习笔记3:sort和priority_queue的比较器重载

1 sort 三种方法 1. 直接重载函数 #include <vector> #include <memory> #include <vector> #include <queue> #include <iostream> #include <algorithm>using namespace std;class Node{ public:int value;Node(){value 0;};explici…

【解决】sklearn-LabelEncoder遇到没在编码规则里的新值

文章目录 一、问题描述二、解决方法Reference 一、问题描述 问题&#xff1a;sklearn-LabelEncoder 遇到没在编码规则里的新值 二、解决方法 方法一&#xff1a;直接保存old_data和encoder_data和之间的映射关系&#xff0c;字典或者下面的csv格式里都可以。 for col in be…

UDS诊断实战系列-再谈19 04读取冻结帧子服务

本文框架 1. 前言2. 19 04 子服务2.1 请求某DTC快照信息2.1.1 请求报文格式及说明2.1.2 响应报文格式及说明 3. 开发注意事项3.1 快照高低字节顺序3.2 快照DID 1. 前言 19服务在整个UDS服务中非常重要&#xff0c;而19 04读取DTC冻结帧数据子服务又在0x19服务中非常重要&#…

<Linux开发>驱动开发 -之-资源的并发与竞争处理

&#xff1c;Linux开发&#xff1e;驱动开发 -之-资源的并发与竞争处理 交叉编译环境搭建&#xff1a; &#xff1c;Linux开发&#xff1e; linux开发工具-之-交叉编译环境搭建 uboot移植可参考以下&#xff1a; &#xff1c;Linux开发&#xff1e; -之-系统移植 uboot移植过…

K8s之Pod生命周期、启动停止钩子详解

文章目录 一、Pod生命周期流程二、初始化容器-initContainers三、主容器操作-containers1、启动钩子-lifecycle.postStart2、停止钩子-lifecycle.preStop 一、Pod生命周期流程 Pod生命周期整个过程 如下图&#xff1a; 1、在启动任何容器之前&#xff0c;前创建 pause 容器&am…

网络通信IO模型上

计算机组成 计算机由软件和硬件组成&#xff0c;软件包括CPU、内存等&#xff0c;硬件包括主板&#xff0c;磁盘&#xff0c;IO设备&#xff08;网卡、鼠标、键盘等&#xff09;、电源按钮。 内核程序加载过程 当接通电源的时候1、BIOS就会把它的一段代码放入了内存当中&#…

ORB SLAM3 构建Frame

1.构造Frame 为了构建一帧Frame&#xff0c;主要的步骤如下&#xff1a; 提取ORB特征点(ExtractORB)对提取的特征点进行矫正(cv::undistortPoints)计算去畸变后的图像边界(ComputeImageBounds)将特征点分配到网格中(AssignFeaturesToGrid) A.提取ORB特征点 首先需要对当前帧…

某程序员哀叹:月薪四五万,却每天极度焦虑痛苦,已有生理性不适,又不敢裸辞,该怎么办?

高薪能买来快乐吗&#xff1f; 来看看这位程序员的哀叹&#xff1a; 实在是扛不住了&#xff0c;每天都在极度焦虑和痛苦中度过&#xff0c;早上起来要挣扎着做心理建设去上班&#xff0c;已经产生生理性的头晕恶心食欲不振。有工作本身的原因&#xff0c;更多是自己心态的问…

如何在CSDN获得铁粉

文章目录 前言关于铁粉方法总结 前言 提示&#xff1a;这里可以添加本文要记录的大概内容&#xff1a; 铁粉512位即可参加此活动 完成一篇如何获得铁粉&#xff0c;或者相关的文章且质量分达到80分以上即可 关于铁粉 简单地说&#xff0c;就是在过去 N 个月内&#xff0c;一…