現(xiàn)在非常多的的MCU性能都還不錯,同時用戶也會去擴展一些外部RAM,這樣如果高效便捷的管理這些內(nèi)存是一個重要話題。
今天給大家分享一份源碼:基于無操作系統(tǒng)的STM32單片機開發(fā),功能強大,可申請到地址空間連續(xù)的不同大小的內(nèi)存空間,且用戶接口簡單,使用方便。
正文部分:
1
源碼說明
源碼包含memory.h 和 memory.c 兩個文件(嵌入式C/C++代碼的“標(biāo)配”),其源碼中包含重要的注釋。memory.h文件 :包含結(jié)構(gòu)體等定義,函數(shù)API申明等;memory.c文件 :是實現(xiàn)內(nèi)存管理相關(guān)API函數(shù)的原型。2
頭文件memory.h
頭文件是相關(guān)的定義和申請:
#ifndef__MEMORY_H__
#define__MEMORY_H__
#include"stdio.h"
#include"string.h"
#include"includes.h"
//用戶使用
typedefstruct
{
void*addr;//申請到的內(nèi)存的起始地址
uint32_tsize;//申請到的內(nèi)存的大小,按照塊大小分配,大于等于申請大小
uint16_ttb;//申請表序號,申請內(nèi)存時分配,釋放內(nèi)存時使用,用戶不使用
}DMEM;
//若返回空,則申請失敗
DMEM*DynMemGet(uint32_tsize);
voidDynMemPut(DMEM*pDmem);
#endif//__MEMORY_H__
這里的代碼比較簡單,也是常規(guī)的寫法,重點是要理解結(jié)構(gòu)體成員的含義。
3
源文件memory.c
源文件主要就是實現(xiàn)內(nèi)存管理的函數(shù),源碼比較多,這里才分為三部分。1、相關(guān)的定義
#include"memory.h"
#defineDMEM_BLOCK_SIZE256//內(nèi)存塊大小為128字節(jié)
#defineDMEM_BLOCK_NUM20//內(nèi)存塊個數(shù)為40個
#defineDMEM_TOTAL_SIZE(DMEM_BLOCK_SIZE*DMEM_BLOCK_NUM)//內(nèi)存總大小
staticuint8_tDMEMORY[DMEM_TOTAL_SIZE];
staticDMEM_STATEDMEMS={0};
typedefenum
{
DMEM_FREE=0,
DMEM_USED=1,
}DMEM_USED_ITEM;
typedefstruct
{
DMEM_USED_ITEMused;//使用狀態(tài)
uint16_tblk_s;//起始塊序號
uint16_tblk_num;//塊個數(shù)
}DMEM_APPLY;
typedefstruct
{
DMEM_USED_ITEMtb_blk[DMEM_BLOCK_NUM];
DMEMtb_user[DMEM_BLOCK_NUM];//用戶申請內(nèi)存信息
DMEM_APPLYtb_apply[DMEM_BLOCK_NUM];//系統(tǒng)分配內(nèi)存信息
uint16_tapply_num;//內(nèi)存申請表占用數(shù)目
uint16_tblk_num;//內(nèi)存塊占用數(shù)目
}DMEM_STATE;
2、內(nèi)存分配函數(shù)DynMemGet
DMEM*DynMemGet(uint32_tsize)
{
uint16_tloop=0;
uint16_tfind=0;
uint16_tblk_num_want=0;
DMEM*user=NULL;
DMEM_APPLY*apply=NULL;
//申請內(nèi)存大小不能為0
if(size==0){returnNULL;}
//申請內(nèi)存不可超過總內(nèi)存大小
if(size>DMEM_TOTAL_SIZE){returnNULL;}
//申請內(nèi)存不可超過剩余內(nèi)存大小
if(size>(DMEM_BLOCK_NUM-DMEMS.blk_num)*DMEM_BLOCK_SIZE){returnNULL;}
//申請表必須有空余
if(DMEMS.apply_num>=DMEM_BLOCK_NUM){returnNULL;}
//計算所需連續(xù)塊的個數(shù)
blk_num_want=(size+DMEM_BLOCK_SIZE-1)/DMEM_BLOCK_SIZE;
//尋找申請表
for(loop=0;loopif(DMEMS.tb_apply[loop].used==DMEM_FREE)
{
apply=&DMEMS.tb_apply[loop];//申請表已找到
user=&DMEMS.tb_user[loop];//用戶表對應(yīng)找到
user->tb=loop;//申請表編號記錄
user->size=blk_num_want*DMEM_BLOCK_SIZE;//分配大小計算
break;
}
}
//沒有找到可用申請表,理論上是不會出現(xiàn)此現(xiàn)象的,申請表剩余已在上面校驗
if(loop==DMEM_BLOCK_NUM){returnNULL;}
//尋找連續(xù)內(nèi)存塊
for(loop=0;loopif(DMEMS.tb_blk[loop]==DMEM_FREE)
{//找到第一個空閑內(nèi)存塊
for(find=1;(find//找到下一個空閑內(nèi)存塊
if(DMEMS.tb_blk[loop+find]!=DMEM_FREE)
{//發(fā)現(xiàn)已使用內(nèi)存塊
break;
}
}
if(find>=blk_num_want)
{//尋找到的空閑內(nèi)存塊數(shù)目已經(jīng)夠用
user->addr=DMEMORY+loop*DMEM_BLOCK_SIZE;//計算申請到的內(nèi)存的地址
apply->blk_s=loop;//記錄申請到的內(nèi)存塊首序號
apply->blk_num=blk_num_want;//記錄申請到的內(nèi)存塊數(shù)目
for(find=0;findblk_num;find++)
{
DMEMS.tb_blk[loop+find]=DMEM_USED;
}
apply->used=DMEM_USED;//標(biāo)記申請表已使用
DMEMS.apply_num+=1;
DMEMS.blk_num+=blk_num_want;
returnuser;
}
else
{//尋找到的空閑內(nèi)存塊不夠用,從下一個開始找
loop+=find;
}
}
}
//搜索整個內(nèi)存塊,未找到大小適合的空間
returnNULL;
}
3、內(nèi)存釋放函數(shù)DynMemPut
voidDynMemPut(DMEM*user)
{
uint16_tloop=0;
//若參數(shù)為空,直接返回
if(NULL==user){return;}
//釋放內(nèi)存空間
for(loop=DMEMS.tb_apply[user->tb].blk_s;looptb].blk_s+DMEMS.tb_apply[user->tb].blk_num;loop++)
{
DMEMS.tb_blk[loop]=DMEM_FREE;
DMEMS.blk_num-=1;
}
//釋放申請表
DMEMS.tb_apply[user->tb].used=DMEM_FREE;
DMEMS.apply_num-=1;
}
代碼中包含注釋,注釋描述的比較清楚,也比較容易理解。
原文標(biāo)題:一個簡單的MCU內(nèi)存管理模塊(附源碼)
文章出處:【微信公眾號:硬件攻城獅】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。
-
單片機
+關(guān)注
關(guān)注
6040文章
44604瀏覽量
637116 -
操作系統(tǒng)
+關(guān)注
關(guān)注
37文章
6862瀏覽量
123533 -
STM32
+關(guān)注
關(guān)注
2270文章
10918瀏覽量
356857
原文標(biāo)題:一個簡單的MCU內(nèi)存管理模塊(附源碼)
文章出處:【微信號:mcu168,微信公眾號:硬件攻城獅】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。
發(fā)布評論請先 登錄
相關(guān)推薦
評論