stm32HAL库+stm32F407zgt6+外部SRAM+malloc内存管理

总结一下STM32 外部sram内存分配以及管理
开发平台:stm32F407ZGT6
外部SRAM:IS62WV51216
使用STM32CubeMX 进行配置
IS62WV51216简介:
在这里插入图片描述
先简单介绍一下FSMC总线地址(若需要详细介绍请自行查阅资料)
在这里插入图片描述

关于sram如何配置请看下图:
在这里插入图片描述
本文参考的是正点原子的 内存管理章节进行简单修改即可使用

malloc.h 头文件的内容

#ifndef __MALLOC_H
#define __MALLOC_H


#include "main.h"

 
 
#ifndef NULL
#define NULL 0
#endif

//定义两个内存池
#define SRAMIN	 0		//内部内存池
#define SRAMEX   1		//外部内存池 

#define SRAMBANK 	2	  //定义支持的SRAM块数.	


//mem1内存参数设定.mem1完全处于内部SRAM里面.
#define MEM1_BLOCK_SIZE			32  	  						              //内存块大小为32字节
#define MEM1_MAX_SIZE			  40*1024  						              //最大管理内存 40K
#define MEM1_ALLOC_TABLE_SIZE	MEM1_MAX_SIZE/MEM1_BLOCK_SIZE 	//内存表大小


//mem2内存参数设定.mem2的内存池处于外部SRAM里面
#define MEM2_BLOCK_SIZE			32  	  						              //内存块大小为32字节
#define MEM2_MAX_SIZE			  512 *1024  					              //最大管理内存960K
#define MEM2_ALLOC_TABLE_SIZE	MEM2_MAX_SIZE/MEM2_BLOCK_SIZE 	//内存表大小 
		 
 
 
//内存管理控制器
struct _m_mallco_dev
{
	void    ( * init ) ( uint8_t );				 //初始化
	uint8_t ( * perused ) ( uint8_t );		 //内存使用率
	uint8_t  * membase [ SRAMBANK ];		   //内存池 管理SRAMBANK个区域的内存
	uint16_t * memmap [ SRAMBANK ]; 		   //内存管理状态表
	uint8_t    memrdy [ SRAMBANK ]; 			 //内存管理是否就绪
};
extern struct _m_mallco_dev mallco_dev;	 //在mallco.c里面定义


void mymemset(void *s,uint8_t c,uint32_t count);	    //设置内存
void mymemcpy(void *des,void *src,uint32_t n);        //复制内存     
void my_mem_init(uint8_t memx);				                //内存管理初始化函数(外/内部调用)
uint32_t my_mem_malloc(uint8_t memx,uint32_t size);	  //内存分配(内部调用)
uint8_t my_mem_free(uint8_t memx,uint32_t offset);		//内存释放(内部调用)
uint8_t my_mem_perused(uint8_t memx);				          //获得内存使用率(外/内部调用) 


//用户调用函数
void myfree(uint8_t memx,void *ptr);  			           //内存释放(外部调用)
void *mymalloc(uint8_t memx,uint32_t size);			       //内存分配(外部调用)
void *myrealloc(uint8_t memx,void *ptr,uint32_t size); //重新分配内存(外部调用)



#endif

malloc.c 文件的内容

#include "malloc.h"	    



//内存池(32字节对齐)
__align(32) uint8_t mem1base[MEM1_MAX_SIZE];													//内部SRAM内存池
__align(32) uint8_t mem2base[MEM2_MAX_SIZE] __attribute__((at(0X68000000)));					//外部SRAM内存池


//内存管理表
uint16_t mem1mapbase[MEM1_ALLOC_TABLE_SIZE];													//内部SRAM内存池MAP
uint16_t mem2mapbase[MEM2_ALLOC_TABLE_SIZE] __attribute__((at(0X68000000+MEM2_MAX_SIZE)));	//外部SRAM内存池MAP


//内存管理参数	   
const uint32_t memtblsize[SRAMBANK]={MEM1_ALLOC_TABLE_SIZE,MEM2_ALLOC_TABLE_SIZE};			//内存表大小
const uint32_t memblksize[SRAMBANK]={MEM1_BLOCK_SIZE,MEM2_BLOCK_SIZE};						      //内存分块大小
const uint32_t memsize[SRAMBANK]={MEM1_MAX_SIZE,MEM2_MAX_SIZE};								          //内存总大小


//内存管理控制器
struct _m_mallco_dev mallco_dev=
{
	my_mem_init,				             //内存初始化
	my_mem_perused,				           //内存使用率
	mem1base,mem2base,			         //内存池
	mem1mapbase,mem2mapbase,	       //内存管理状态表
	0,0,  		 				               //内存管理未就绪
};



//复制内存
//*des:目的地址
//*src:源地址
//n:需要复制的内存长度(字节为单位)
void mymemcpy(void *des,void *src,uint32_t n)  
{  
	uint8_t *xdes=des;
	uint8_t *xsrc=src; 
	
	
	while(n--)*xdes++=*xsrc++;  
	
}  


//设置内存
//*s:内存首地址
//c :要设置的值
//count:需要设置的内存大小(字节为单位)
void mymemset(void *s,uint8_t c,uint32_t count)  
{  
	uint8_t *xs = s;  


	while(count--)*xs++=c;  
	
}	   


//内存管理初始化  
//memx:所属内存块
void my_mem_init(uint8_t memx)  
{  
	mymemset(mallco_dev.memmap[memx], 0,memtblsize[memx]*2);//内存状态表数据清零  
	mymemset(mallco_dev.membase[memx], 0,memsize[memx]);	//内存池所有数据清零  
	mallco_dev.memrdy[memx]=1;								//内存管理初始化OK  
}  


//获取内存使用率
//memx:所属内存块
//返回值:使用率(0~100)
uint8_t my_mem_perused(uint8_t memx)  
{  
	uint32_t used=0;  
	uint32_t i;  
	
	
	for(i=0;i<memtblsize[memx];i++)  
	{  
			if(mallco_dev.memmap[memx][i])used++; 
	} 
	
	return (used*100)/(memtblsize[memx]);  
	
}  


//内存分配(内部调用)
//memx:所属内存块
//size:要分配的内存大小(字节)
//返回值:0XFFFFFFFF,代表错误;其他,内存偏移地址 
uint32_t my_mem_malloc(uint8_t memx,uint32_t size)  
{  
	signed long offset=0;  
	uint32_t nmemb;	//需要的内存块数  
	uint32_t cmemb=0;//连续空内存块数
	uint32_t i;  
	
	
	if(!mallco_dev.memrdy[memx])mallco_dev.init(memx);//未初始化,先执行初始化 
	
	if(size==0)
		return 0XFFFFFFFF;//不需要分配
	
	nmemb=size/memblksize[memx];  	//获取需要分配的连续内存块数
	
	if(size%memblksize[memx])nmemb++; 
	
	for(offset=memtblsize[memx]-1;offset>=0;offset--)//搜索整个内存控制区  
	{     
		if(!mallco_dev.memmap[memx][offset])cmemb++;//连续空内存块数增加
		else cmemb=0;								//连续内存块清零
		
		if(cmemb==nmemb)							//找到了连续nmemb个空内存块
		{
			for(i=0;i<nmemb;i++)  					//标注内存块非空 
			{  
					mallco_dev.memmap[memx][offset+i]=nmemb;  
			}  
			return (offset*memblksize[memx]);//返回偏移地址  
		}
		
	}  
	
	return 0XFFFFFFFF;//未找到符合分配条件的内存块  
	
}  


//释放内存(内部调用) 
//memx:所属内存块
//offset:内存地址偏移
//返回值:0,释放成功;1,释放失败;  
uint8_t my_mem_free(uint8_t memx,uint32_t offset)  
{  
	int i;  
	
	
	if(!mallco_dev.memrdy[memx])//未初始化,先执行初始化
	{
	mallco_dev.init(memx);    
			return 1;//未初始化  
	}  
	
	if(offset<memsize[memx])//偏移在内存池内. 
	{  
			int index=offset/memblksize[memx];			//偏移所在内存块号码  
			int nmemb=mallco_dev.memmap[memx][index];	//内存块数量
			for(i=0;i<nmemb;i++)  						//内存块清零
			{  
					mallco_dev.memmap[memx][index+i]=0;  
			}  
			return 0;  
	}
	else 
		return 2;//偏移超区了.  
	
}  


//释放内存(外部调用) 
//memx:所属内存块
//ptr:内存首地址 
void myfree(uint8_t memx,void *ptr)  
{  
	uint32_t offset;   
	
	
	if(ptr==NULL)return;//地址为0.  
	offset=(uint32_t)ptr-(uint32_t)mallco_dev.membase[memx];     
	my_mem_free(memx,offset);	//释放内存     
	
}  


//分配内存(外部调用)
//memx:所属内存块
//size:内存大小(字节)
//返回值:分配到的内存首地址.
void *mymalloc(uint8_t memx,uint32_t size)  
{  
	uint32_t offset;   
	
	
	offset=my_mem_malloc(memx,size);  	   	 
	
	if(offset==0XFFFFFFFF)return NULL;  
	else return (void*)((uint32_t)mallco_dev.membase[memx]+offset);  
	
}  


//重新分配内存(外部调用)
//memx:所属内存块
//*ptr:旧内存首地址
//size:要分配的内存大小(字节)
//返回值:新分配到的内存首地址.
void *myrealloc(uint8_t memx,void *ptr,uint32_t size)  
{  
	uint32_t offset;    
	
	
	offset=my_mem_malloc(memx,size);   	
	if(offset==0XFFFFFFFF)return NULL;     
	else  
	{  									   
		mymemcpy((void*)((uint32_t)mallco_dev.membase[memx]+offset),ptr,size);	//拷贝旧内存内容到新内存   
		myfree(memx,ptr);  											  		//释放旧内存
		return (void*)((uint32_t)mallco_dev.membase[memx]+offset);  				//返回新内存首地址
	}  
}

需要注意的是
在这里插入图片描述
根据所选的外部SRAM芯片来修改此处的内存大小(本实验选用的是512kb的外部SRAM)

uint32_t *pArraySram;//定义内存指针的地址变量
//  my_mem_init(SRAMIN);//初始化内部内存池
	my_mem_init(SRAMEX);//初始化外部内存池
	pArraySram = (uint32_t *)mymalloc(SRAMEX,50);//分配内存

	if(pArraySram == NULL)
	{
		printf("malloc fail\r\n");
	
	}
	else
	{
			for(char i = 0; i < 50; i++ )
			{
					 pArraySram[i] = i;

			}
			
			for(char i = 0; i < 50; i++ )
			{
				printf("*pArraySram%d value is:%d\r\n",i,pArraySram[i]);
				HAL_Delay(1);
			}
	}
	myfree(SRAMEX,pArraySram);//释放内存

然如何使用内存分配函数

实验现象:(截取部分)

在这里插入图片描述
至此完结

  • 1
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值