当前位置:首页 > 公众号精选 > strongerHuang
[导读]关注星标公众号,不错过精彩内容转自| 嵌入式大杂烩本次给大家分享一位大佬写的应用于单片机内存管理模块mem_malloc,这个mem_malloc的使用不会产生内存碎片,可以高效利用单片机ram空间。mem_malloc代码仓库:❝https://github.com/chenq...

关注 星标公众,不错过精彩内容

转自 | 嵌入式大杂烩


本次给大家分享一位大佬写的应用于单片机内存管理模块mem_malloc,这个mem_malloc的使用不会产生内存碎片,可以高效利用单片机ram空间。

mem_malloc代码仓库:

https://github.com/chenqy2018/mem_malloc

mem_malloc介绍

一般单片机的内存都比较小,而且没有MMU,malloc 与free的使用容易造成内存碎片。而且可能因为空间不足而分配失败,从而导致系统崩溃,因此应该慎用,或者自己实现内存管理。

mem_malloc就是一个不会产生内存碎片的、适合单片机使用的内存管理模块。其与使用malloc的区别如:

「算法原理:」

定义一个数组作为动态分配的堆空间,低地址空间保存管理数据,高地址空间实际分配给用户的缓存(类似堆栈使用,分配是往中间靠拢),free时移动高地址用户空间(以时间换空间),使得未使用的空间都是连续的。

mem_malloc测试验证

下面以小熊派IOT开发板来做实验。

实验过程很简单。准备一份开发板带串口打印的工程,下载mem_malloc,把mem_malloc.c、mem_malloc.h复制到工程目录下,并添加到工程里:

然后进行编译,编译过程可能会报错:

..\Src\mem_malloc.c(119): error:  #852: expression must be a pointer to a complete object type
这份代码在不同编译器下编译情况不同。gcc下编译不会报错,在keil下编译报如上错误。

keil编译器更严格些。报错原因是对mem_block结构体的mem_ptr成员进行操作,而mem_ptr成员的类型是void*,而mem_ptr成员参与运算时的增、减偏移量取决于mem_ptr的类型,所以这里我们需要指定类型。

我们把相关报错代码修改如:

再次编译就正常了。

下面简单看一下mem_malloc的代码。

「mem_malloc.h:」

#ifndef __MEM_MALLOC_H__
#define __MEM_MALLOC_H__

#ifdef __cplusplus
extern "C" {
#endif

#include  
#include 
#include 
#include 
#include 

#pragma pack(1)
typedef struct mem_block
{
 
    void   *mem_ptr;  
    unsigned int mem_size; 
    unsigned int mem_index;    
}mem_block;
#pragma pack()

#define MEM_SIZE        128


void print_mem_info(void);
void print_hex(char *data, int len);
void print_mem_hex(int size);
int mem_malloc(unsigned int msize);
int mem_realloc(int id, unsigned int msize);
void *mem_buffer(int id);
int mem_free(int id);


#ifdef __cplusplus
}
#endif

#endif
「mem_malloc.c:」

暂不贴出,感兴趣的小伙伴可以在上面的仓库地址自行下载阅读。在本公众号后台回复:mem_malloc,进行获取。

下面对mem_malloc进行测试验证。

测试代码作者也有给出,这里我们简单测试即可,进行了一些删减及增加了一些注释:

#include "mem_malloc.h"

char mem_id[10]={0};  // 10块内存块

void test_malloc(int i, int size)
{
 printf("------test_malloc-------\n");
 mem_id[i] = mem_malloc(size);
 if(mem_id[i] == 0)
 {
  printf("malloc --- fail\n");
  printf("size=%d\n", size);
 }
 else
 {
  char *p = mem_buffer(mem_id[i]);
        memset(p, i, size);
        printf("p = 0x%x, i=%d, id=%d, size=%d\n", (int)p, i, mem_id[i], size);
 }
 print_mem_hex(MEM_SIZE);
}

void test_buffer(int i, int size)
{
 printf("------test_buffer-------\n");
 printf("i=%d, id = %d, size=%d\n", i, mem_id[i], size);
 char *p = mem_buffer(mem_id[i]);
    if(p != NULL)
 {
  memset(p, 0xf0 i, size);
        print_mem_hex(MEM_SIZE);
 }
 else
 {
  printf("test_buffer---fail\n");
 }
}

void test_realloc(int i, int size)
{
 printf("------test_realloc-------\n");
    printf("i=%d, id = %d, size=%d\n", i, mem_id[i], size);
 int ret = mem_realloc(mem_id[i], size);
 if(ret)
 {
  char *p = mem_buffer(mem_id[i]);
  memset(p, 0xa0 i, size);
        print_mem_hex(MEM_SIZE);
 }
 else
 {
  printf("test_realloc---fail\n");
 }
}

void test_free(int i)
{
 printf("------test_free-------\n");
 printf("i=%d, id = %d\n", i, mem_id[i]);
 if(mem_free(mem_id[i]))
  print_mem_hex( MEM_SIZE);
}

void main(void)
{
 print_mem_info();   // 打印内存信息
 test_malloc(110); // 给申请一块10个字节的内存,标记内存块id为1
 test_malloc(28); // 给申请一块8个字节的内存,标记内存块id为2
 test_malloc(320); // 给申请一块20个字节的内存,标记内存块id为2

 test_free(2);  // 释放id为2的内存块的内存

 test_malloc(470); // 申请一块70个字节的内存

 test_free(1);       // 释放id为1的内存块内存

 test_buffer(320); // 获取id为3的内存块地址,并往这个内存块重新写入0xf0 i的数据

 test_realloc(310); // 重新分配内存,并往这个内存块重新写入0xa0 i的数据
 
 for(int i=0; i<10; i )  // 释放所有内存块内存,已释放的不再重新释放
  test_free(i);

运行结果及解析:

这里设定一个128字节的数组作为堆空间使用。其中数组前面存放的是申请到的内存块的信息,包括内存块地址、大小、索引信息,这三个数据各占4个字节,共12个字节。这里有设计到一个大小端模式的问题,STM32平台为小端模式,即数据的低位存储在内存的低地址中。

申请的内存块从128字节的尾部开始分配,再次申请的内存块依次往前移,释放的内存,则整体内存块往后移动,内存块之前不留空隙,即不产生内存碎片。

以上就是本次的分享,如有错误,欢迎指出,谢谢!


------------ END ------------

●嵌入式专栏精选教程●精选汇总 | ST工具、下载编程工具●精选汇总 | 嵌入式软件设计与开发
●精选汇总 | STM32、MCU、单片机
欢迎关注我的公众号回复“加群”按规则加入技术交流群,回复“1024”查看更多内容。单片机、开发工具、编程技术、行业资讯等相关内容(关注并回复“1024”查看更多精彩内容)。" data-from="0">欢迎关注我的视频号:


点击“阅读原文”查看更多分享,欢迎点分享、收藏、点赞、在看。

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

单片机是一种嵌入式系统,它是一块集成电路芯片,内部包含了处理器、存储器和输入输出接口等功能。

关键字: 单片机 编写程序 嵌入式

在现代电子技术的快速发展中,单片机以其高度的集成性、稳定性和可靠性,在工业自动化、智能家居、医疗设备、航空航天等诸多领域得到了广泛应用。S32单片机,作为其中的佼佼者,其引脚功能丰富多样,是实现与外部设备通信、控制、数据...

关键字: s32单片机引脚 单片机

在微控制器领域,MSP430与STM32无疑是两颗璀璨的明星。它们各自凭借其独特的技术特点和广泛的应用领域,在市场上占据了重要的位置。本文将深入解析MSP430与STM32之间的区别,探讨它们在不同应用场景下的优势和局限...

关键字: MSP430 STM32 单片机

该系列产品有助于嵌入式设计人员在更广泛的系统中轻松实现USB功能

关键字: 单片机 嵌入式设计 USB

单片机编程语言是程序员与微控制器进行交流的桥梁,它们构成了单片机系统的软件开发基石,决定着如何有效、高效地控制和管理单片机的各项资源。随着微控制器技术的不断发展,针对不同应用场景的需求,形成了丰富多样的编程语言体系。本文...

关键字: 单片机 微控制器

单片机,全称为“单片微型计算机”或“微控制器”(Microcontroller Unit,简称MCU),是一种高度集成化的电子器件,它是现代科技领域的关键组件,尤其在自动化控制、物联网、消费电子、汽车电子、工业控制等领域...

关键字: 单片机 MCU

STM32是由意法半导体公司(STMicroelectronics)推出的基于ARM Cortex-M内核的32位微控制器系列,以其高性能、低功耗、丰富的外设接口和强大的生态系统深受广大嵌入式开发者喜爱。本文将详细介绍S...

关键字: STM32 单片机

在当前的科技浪潮中,单片机作为嵌入式系统的重要组成部分,正以其强大的功能和广泛的应用领域受到越来越多行业的青睐。在众多单片机中,W79E2051以其卓越的性能和稳定的工作特性,成为市场上的明星产品。本文将深入探讨W79E...

关键字: 单片机 w79e2051单片机

单片机,又称为微控制器或微处理器,是现代电子设备中的核心部件之一。它集成了中央处理器、存储器、输入输出接口等电路,通过外部信号引脚与外部设备进行通信,实现对设备的控制和管理。本文将详细介绍单片机的外部信号引脚名称及其功能...

关键字: 单片机 微控制器 中央处理器

随着科技的飞速发展,单片机和嵌入式系统在现代电子设备中的应用越来越广泛。它们不仅提高了设备的智能化水平,还推动了各行各业的创新与发展。在单片机和嵌入式系统的开发中,编程语言的选择至关重要。本文将深入探讨单片机和嵌入式系统...

关键字: 单片机 嵌入式系统 电子设备
关闭
关闭