加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
文件
克隆/下载
xmem.c 32.17 KB
一键复制 编辑 原始数据 按行查看 历史
牧梦追马 提交于 2018-09-17 11:19 . 1.更新到1.0.2。
12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106
/*
* xmem implementation and interface functions
* Copyright (c) 2016, Shoowing <420260138@qq.com>
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of Apache License Version 2.0
*
* See README for more details.
*/
#include "xconfig.h"
#include "platform.h"
#include "xtypes.h"
#define XMEM_VER "1.0.2"
/***************************************************************************
X-Memory Physical Sketch
----------------------------------------------------------------------------
| Pool |
| |
| ------------------ -------------------- -------------------- |
| | SuperBlock | | Block | | Block | |
| | | | | | | |
| | 4-bytes list | | xx-bytes block | | xxx-bytes block | |
| | | | | | | |
| ------------------ -------------------- -------------------- |
| |
| ------------------ -------------------- -------------------- |
| | SuperBlock | | Block | | Block | |
| | | | | | | |
| | 8-bytes list | | xxxx-bytes block | | xxxxx-bytes block| |
| | | | | | | |
| ------------------ -------------------- -------------------- |
| |
| ------------------ |
| | SuperBlock | |
| | | ... ... |
| | 16-bytes list | |
| | | |
| ------------------ |
| |
----------------------------------------------------------------------------
****************************************************************************/
static pxMemMgrHdr xMemMgrHdrList;
static pxMemBlock xMemBlkList = NULL;
static u32 xMemMgrHdrListEnd=0;
static u32 xMemBlkPoolStart=0;
static unsigned long xMem_Addr = 0;
static size_t xMem_Size = 0;
#define XMEM_POOL_START (xMem_Addr)
#define XMEM_POOL_END (xMem_Addr+xMem_Size)
/***************************************************************************
* FUNCTION
* xMemBlockListInit
* DESCRIPTION
* Init Block List
* PARAMETERS
* void
* RETURNS
* void
* *************************************************************************/
static void xMemBlockListInit(void)
{
#if XMEM_CORRUPT_CHECK
xMemBlkList = (pxMemBlock)XMEM_POOL_START;
xMemBlkList->blksize=xMem_Size-XMEM_BLOCK_SIZE;
xMemBlkList->next=NULL;
xMemBlkList->free=1;
#else
xMemBlkPoolStart = XMEM_POOL_END;
xMemBlkList = NULL;
#endif
}
#if XMEM_CORRUPT_CHECK
/***************************************************************************
* FUNCTION
* xMemBlockListCheck
* DESCRIPTION
* check memory block boundry
* PARAMETERS
* void
* RETURNS
* void
* *************************************************************************/
const char xMemDumpFmtBlockList[]="blk:%u,blksize:%d,blknext:%u,free:%d\r\n";
void dump(unsigned char * mem,size_t size)
{
int i;
for(i=0;i<size;i++)
{
xMemInfoPrintf("%02x",mem[i]);
if((i%4)==3) xMemInfoPrintf(" ");
if((i%16)==15) xMemInfoPrintf("\r\n");
}
if((i%16)!=0) xMemInfoPrintf("\r\n");
}
void xMemBlockListCheck()
{
pxMemBlock pmemblk,preblk;
preblk=pmemblk=xMemBlkList;
while(pmemblk)
{
if(pmemblk->free>1
||(pmemblk->next&&
((u32)pmemblk+XMEM_BLOCK_SIZE+pmemblk->blksize!=(u32)pmemblk->next
||(u32)pmemblk->next>=XMEM_POOL_END
||(u32)pmemblk->next<=XMEM_POOL_START)
)
)
{
xMemInfoPrintf(xMemDumpFmtBlockList,(u32)pmemblk,pmemblk->blksize,(u32)pmemblk->next,pmemblk->free);
dump((unsigned char *)preblk,preblk->blksize+XMEM_BLOCK_SIZE);
dump((unsigned char *)pmemblk,128);
xMemAssert(0);
}
preblk=pmemblk;
pmemblk=pmemblk->next;
}
return;
}
/***************************************************************************
* FUNCTION
* xMemBlockAlloc
* DESCRIPTION
* allocate a memory bock
* PARAMETERS
* size [IN] block size that required
* RETURNS
* void * memory block address that alocated
* *************************************************************************/
static void * xMemBlockAlloc(size_t size)
{
pxMemBlock blkprev=NULL,blk=NULL,blknew=NULL,blkalloc=NULL;
s16 allocsize,remainsize;
/*
--------------------------------------------------------------
| | prev | blk | next | |
| ... |--------------|--------------|--------------| ... |
| |header| mem |header| mem |header| mem | |
--------------------------------------------------------------
*/
blk=xMemBlkList;
remainsize=xMem_Size;
blkalloc=NULL;
allocsize=size+((size%4)==0?0:(4-size%4));
while(blk)
{
if(blk->free)
{
if(blk->blksize==allocsize)
{//most fitable, block size equals to required size
blk->free=0;
return (void*)((u32)blk+XMEM_BLOCK_SIZE);
}
else if(blk->blksize>allocsize&&(blk->blksize-allocsize)<remainsize)
{// find minimal fitable size block
remainsize=blk->blksize-allocsize;
blkalloc=blk;
}
}
blkprev=blk;
blk=blk->next;
}
if(blkalloc)
{
if(remainsize>(XMEM_BLOCK_SIZE+XMEM_BALLANCE_SIZE))
{
//split into 2 blocks
blknew=(pxMemBlock)((void *)((u32)blkalloc+allocsize+XMEM_BLOCK_SIZE));
blknew->blksize=remainsize-XMEM_BLOCK_SIZE;
blknew->free=1;
blknew->next=blkalloc->next;
blkalloc->next=blknew;
blkalloc->blksize=allocsize;
}
blkalloc->free=0;
return (void*)((u32)blkalloc+XMEM_BLOCK_SIZE);
}
xMemBlockListCheck();
return NULL;
}
/***************************************************************************
* FUNCTION
* xMemBlockFree
* DESCRIPTION
* free a memory bock
* PARAMETERS
* ptr [IN] block address that be free
* RETURNS
* u8 0-success,1-failure
* *************************************************************************/
static u8 xMemBlockFree(void *ptr){
pxMemBlock blkprev=NULL,blkfree=NULL;
/*
--------------------------------------------------------------
| | prev | blk | next | |
| ... |--------------|--------------|--------------| ... |
| |header| mem |header| mem |header| mem | |
--------------------------------------------------------------
*/
blkfree = xMemBlkList;
while(blkfree)
{
if((u32)blkfree+XMEM_BLOCK_SIZE==(u32)ptr)
{
blkfree->free=1;
//merge physical neighbor blocks, previous or next, assure block will not overlap reserve space
if(blkprev&&blkprev->free)
{
blkprev->blksize += (blkfree->blksize+XMEM_BLOCK_SIZE);
blkprev->next = blkfree->next;
blkfree = blkprev;
}
if(blkfree->next&&blkfree->next->free)
{
blkfree->blksize += (blkfree->next->blksize+XMEM_BLOCK_SIZE);
blkfree->next = blkfree->next->next;
}
return 0;
}
blkprev = blkfree;
blkfree = blkfree->next;
}
return 1;
}
#else
/***************************************************************************
* FUNCTION
* xMemMgrHdrListInit
* DESCRIPTION
* init Header List
* PARAMETERS
* void
* RETURNS
* void
* *************************************************************************/
static void xMemMgrHdrListInit(void)
{
/*
-------------------------------------------------------------------------------
| Header List| xMemHdrEnd--->| ... |<---xMemBlkStart |Blocks,Super Blocks|
-------------------------------------------------------------------------------
This Structure avoid Header List overwrote by pointer which allocated from X-Memory Pool
*/
xMemMgrHdrList = NULL;
xMemMgrHdrListEnd = XMEM_POOL_START;
}
/***************************************************************************
* FUNCTION
* xMemMgrHdrGet
* DESCRIPTION
* get a Header
* PARAMETERS
* type [IN] XMEM_LIST_TYPE_BLOCK or XMEM_LIST_TYPE_SUPERBLOCK
* RETURNS
* void
* *************************************************************************/
static void * xMemMgrHdrGet(u8 type)
{
pxMemMgrHdr hdr = NULL,hdrnew = NULL, hdrprev = NULL;
u16 allocsize;
/*
-------------------------------------------
| ... | prev | blk | next | ... |
-------------------------------------------
*/
hdr = xMemMgrHdrList;
if(type == XMEM_LIST_TYPE_BLOCK)
allocsize = XMEM_NODE_SIZE(xMemBlock);
else if(type == XMEM_LIST_TYPE_SUPERBLOCK)
allocsize = XMEM_NODE_SIZE(xMemSuperBlock);
else
return NULL;
while(hdr)
{
if(hdr->type == XMEM_LIST_TYPE_FREE)
{
if(hdr->size == allocsize)
{
//most fitable, block size equals to required size
hdr->type = type;
return (void*)((u32)hdr+XMEM_HEADER_SIZE);
}
else if(hdr->size > allocsize)
{
//split into 2 hdrs
hdrnew = (pxMemMgrHdr)((void*)((u32)hdr+allocsize+XMEM_HEADER_SIZE));
hdrnew->type = XMEM_LIST_TYPE_FREE;
hdrnew->size = hdr->size - allocsize - XMEM_HEADER_SIZE;
hdrnew->next = hdr->next;
hdr->next = hdrnew;
hdr->size = allocsize;
hdr->type = type;
return (void*)((u32)hdr+XMEM_HEADER_SIZE);
}
}
hdrprev = hdr;
hdr = hdr->next;
}
if(xMemMgrHdrListEnd+allocsize+XMEM_HEADER_SIZE<xMemBlkPoolStart)
{
hdrnew = (pxMemMgrHdr)xMemMgrHdrListEnd;
if(hdrprev)
{
hdrprev->next = hdrnew;
}
else
{
xMemMgrHdrList = hdrnew;
}
hdrnew->type = type;
hdrnew->size = allocsize;
hdrnew->next = NULL;
xMemMgrHdrListEnd += allocsize+XMEM_HEADER_SIZE;
hdr = hdrnew;
return (void*)((u32)hdr+XMEM_HEADER_SIZE);
}
return NULL;
}
/***************************************************************************
* FUNCTION
* xMemMgrHdrPut
* DESCRIPTION
* put a Header
* PARAMETERS
* header [IN] the header be put
* RETURNS
* void
* *************************************************************************/
static void xMemMgrHdrPut(void * header)
{
pxMemMgrHdr hdrfree=NULL,hdrprev=NULL,hdrprev2=NULL;
u16 headersize;
/*
-------------------------------------------
| ... | prev | blk | next | ... |
-------------------------------------------
*/
hdrfree = xMemMgrHdrList;
while(hdrfree)
{
if((u32)hdrfree+XMEM_HEADER_SIZE == (u32)header)
{
xMemAssert(hdrfree->type!=XMEM_LIST_TYPE_FREE);
if(hdrfree->type == XMEM_LIST_TYPE_BLOCK)
headersize = XMEM_NODE_SIZE(xMemBlock);
else if(hdrfree->type == XMEM_LIST_TYPE_SUPERBLOCK)
headersize = XMEM_NODE_SIZE(xMemSuperBlock);
hdrfree->type = XMEM_LIST_TYPE_FREE;
if(hdrprev&&hdrprev->type == XMEM_LIST_TYPE_FREE)
{
hdrprev->next = hdrfree->next;
hdrprev->size += headersize+XMEM_HEADER_SIZE;
hdrfree = hdrprev;
hdrprev = hdrprev2;
}
if(hdrfree->next&&hdrfree->next->type == XMEM_LIST_TYPE_FREE)
{
hdrfree->size += hdrfree->next->size + XMEM_HEADER_SIZE;
hdrfree->next = hdrfree->next->next;
}
if((u32)hdrfree + hdrfree->size+XMEM_HEADER_SIZE==xMemMgrHdrListEnd)
{
xMemMgrHdrListEnd -= hdrfree->size+XMEM_HEADER_SIZE;
if(hdrprev) hdrprev->next = NULL;
else xMemMgrHdrList = NULL;
}
return;
}
hdrprev2 = hdrprev;
hdrprev = hdrfree;
hdrfree = hdrfree->next;
}
xMemAssert(0);
}
static const char xMemDumpMsgMgrHdrLst[]="-----xMemMgrHdrLst Info-----\r\n";
static const char xMemDumpFmtMgrHdrLst[]="hdr:%u,hdrsize:%d,hdrnext:%u,type:%d\r\n";
/***************************************************************************
* FUNCTION
* xMemMgrHdrListInfoDump
* DESCRIPTION
* Dump Header List information for debug or test
* PARAMETERS
* void
* RETURNS
* void
* *************************************************************************/
static void xMemMgrHdrListInfoDump(void)
{
pxMemMgrHdr phdrlst;
phdrlst=xMemMgrHdrList;
xMemInfoPrintf(xMemDumpMsgMgrHdrLst);
while(phdrlst)
{
xMemInfoPrintf(xMemDumpFmtMgrHdrLst,(u32)phdrlst,phdrlst->size,(u32)phdrlst->next,phdrlst->type);
phdrlst=phdrlst->next;
}
xMemInfoPrintf(xMemDumpMsgMgrHdrLst);
return;
}
/***************************************************************************
* FUNCTION
* xMemBlockAlloc
* DESCRIPTION
* allocate a memory bock
* PARAMETERS
* size [IN] block size that required
* RETURNS
* void * memory block address that alocated
* *************************************************************************/
static void * xMemBlockAlloc(size_t size)
{
pxMemBlock blkprev=NULL,blk=NULL,blknew=NULL,blkalloc=NULL;
u16 allocsize,remainsize;
/*
-------------------------------------------
| ... | next | blk | prev | ... |
-------------------------------------------
*/
blk = xMemBlkList;
blkprev = xMemBlkList;
remainsize = xMemBlkPoolStart-xMemMgrHdrListEnd;
blkalloc=NULL;
allocsize=size+((size%4)==0?0:(4-size%4));
while(blk)
{
if(blk->free)
{
if(blk->blksize==allocsize)
{//most fitable, block size equals to required size
blk->free=0;
return (void*)blk->addr;
}
else if(blk->blksize>allocsize&&(blk->blksize-allocsize)<remainsize)
{// find minimal fitable size block
remainsize=blk->blksize-allocsize;
blkalloc=blk;
}
}
blkprev=blk;
blk=blk->next;
}
if(blkalloc)
{
if(remainsize>=XMEM_BALLANCE_SIZE)
{
//split into 2 blocks
blknew=(pxMemBlock)xMemMgrHdrGet(XMEM_LIST_TYPE_BLOCK);
if(blknew != NULL)
{
blknew->blksize=remainsize;
blknew->free=1;
blknew->next=blkalloc->next;
blknew->addr=(void*)((u32)blkalloc->addr+allocsize);
blkalloc->next=blknew;
blkalloc->blksize=allocsize;
}
}
blkalloc->free=0;
return (void*)blkalloc->addr;
}
else if(remainsize>allocsize)
{
blknew=(pxMemBlock)xMemMgrHdrGet(XMEM_LIST_TYPE_BLOCK);
if(blknew != NULL)
{
if(blkprev)
{
blkprev->next = blknew;
}
else
{
xMemBlkList = blknew;
}
blknew->next = NULL;
blknew->blksize = allocsize;
blknew->free = 0;
xMemBlkPoolStart -= allocsize;
blknew->addr = (void *)xMemBlkPoolStart;
blkalloc = blknew;
return (void*)blkalloc->addr;
}
}
return NULL;
}
/***************************************************************************
* FUNCTION
* xMemBlockFree
* DESCRIPTION
* free a memory bock
* PARAMETERS
* ptr [IN] block address that be free
* RETURNS
* u8 0-success,1-failure
* *************************************************************************/
static u8 xMemBlockFree(void *ptr)
{
pxMemBlock blkprev=NULL,blkfree=NULL,blkprev2=NULL;
/*
-------------------------------------------
| ... | next | blk | prev | ... |
-------------------------------------------
*/
blkfree=xMemBlkList;
while(blkfree)
{
if(blkfree->addr==ptr)
{
blkfree->free = 1;
//may move this block of code to memory collection
if(blkprev&&blkprev->free)
{
blkprev->blksize += blkfree->blksize;
blkprev->next = blkfree->next;
blkprev->addr = blkfree->addr;
xMemMgrHdrPut(blkfree);
blkfree = blkprev;
blkprev = blkprev2;
}
if(blkfree->addr==(void *)xMemBlkPoolStart)
{
xMemBlkPoolStart += blkfree->blksize;
if(blkprev)
{
blkprev->next = blkfree->next;
}
else
{
blkfree->addr = NULL;
blkfree->blksize = 0;
blkfree->next = NULL;
}
xMemMgrHdrPut(blkfree);
}
else if(blkfree->next&&blkfree->next->free)
{
blkprev = blkfree;
blkfree = blkfree->next;
blkprev->blksize += blkfree->blksize;
blkprev->addr = blkfree->addr;
blkprev->next = blkfree->next;
xMemMgrHdrPut(blkfree);
}
//end
return 0;
}
blkprev2 = blkprev;
blkprev = blkfree;
blkfree = blkfree->next;
}
return 1;
}
#endif
static const char xMemDumpMsgBlock[]="-----Block Info-----\r\n";
static const char xMemDumpFmtBlock[]="blk:%u,blksize:%d,blknext:%u,free:%d\r\n";
/***************************************************************************
* FUNCTION
* xMemBLockListInfoDump
* DESCRIPTION
* Dump block lists information
* PARAMETERS
* void
* RETURNS
* void
* *************************************************************************/
static void xMemBlockListInfoDump(void)
{
pxMemBlock pmemblk;
pmemblk=xMemBlkList;
xMemInfoPrintf(xMemDumpMsgBlock);
while(pmemblk)
{
#if XMEM_CORRUPT_CHECK
xMemInfoPrintf(xMemDumpFmtBlock,(u32)pmemblk,pmemblk->blksize,(u32)pmemblk->next,pmemblk->free);
#else
xMemInfoPrintf(xMemDumpFmtBlock,(u32)pmemblk->addr,pmemblk->blksize,(u32)pmemblk->next,pmemblk->free);
#endif
pmemblk=pmemblk->next;
}
xMemInfoPrintf(xMemDumpMsgBlock);
return;
}
#if XMEM_SUPERBLOCK_ENABLE
static xMemSuperBlock xMemSuperBlockList[XMEM_SUPERBLOCK_LIST_COUNT]={0};
static const char xMemSuperBlkInitFailureFmt[]="Init Super Block [Failed], super block:%u, address:%u, blocks:%d,block size:%d\r\n";
/***************************************************************************
* FUNCTION
* xMemSuperBlockInit
* DESCRIPTION
* Init super block
* PARAMETERS
* psuperblock [IN/OUT] super block be initial
* addr [IN] meta blocks address
* nblks [IN] blocks count
* blksize [IN] block size
* RETURNS
* void
* *************************************************************************/
static void xMemSuperBlockInit (xMemSuperBlock * psuperblock,void *addr, u8 nblks, u16 blksize)
{
xMemAssert(nblks<=XMEM_SUPERBLOCK_BLKS_MAX);
if (addr == NULL||nblks < 2||blksize < sizeof(void *)||psuperblock == NULL)
{
xMemDebugPrintf(xMemSuperBlkInitFailureFmt,(u32)psuperblock,(u32)addr,nblks,blksize);
return ;
}
psuperblock->addr = addr;
psuperblock->freeList = (nblks==XMEM_SUPERBLOCK_BLKS_MAX)?0xFFFFFFFF:((1<<nblks)-1);
psuperblock->nfree = nblks;
psuperblock->nblk = nblks;
psuperblock->blksize = blksize;
psuperblock->next = NULL;
return ;
}
/***************************************************************************
* FUNCTION
* xMemSuperBlockAppend
* DESCRIPTION
* Append a new super block
* PARAMETERS
* psuperblock [IN/OUT] super block list
* RETURNS
* xMemSuperBlock * new super block
* *************************************************************************/
static xMemSuperBlock * xMemSuperBlockAppend(xMemSuperBlock * superblocklist)
{
xMemSuperBlock * pmemtail,*pmemnew=NULL;
void *blk;
pmemtail=superblocklist;
while(pmemtail->next) pmemtail=pmemtail->next;
#if XMEM_CORRUPT_CHECK
pmemnew=(xMemSuperBlock *)xMemBlockAlloc(XMEM_LIST_TYPE_SUPERBLOCK);
#else
pmemnew=(xMemSuperBlock *)xMemMgrHdrGet(XMEM_LIST_TYPE_SUPERBLOCK);
#endif
if(pmemnew)
{
blk=xMemBlockAlloc(superblocklist->blksize*(superblocklist->nblk/2));
if(blk)
{
xMemSuperBlockInit(pmemnew,blk,(superblocklist->nblk/2),superblocklist->blksize);
pmemtail->next=pmemnew;
}else
{
#if XMEM_CORRUPT_CHECK
xMemBlockFree(pmemnew);
#else
xMemMgrHdrPut(pmemnew);
#endif
pmemnew=NULL;
}
}
return pmemnew;
}
/***************************************************************************
* FUNCTION
* xMemSuperBlockListInit
* DESCRIPTION
* Init super block list
* PARAMETERS
* void
* RETURNS
* void
* *************************************************************************/
static void xMemSuperBlockListInit(void)
{
void * blk;
#if XMEM_8META_ENABLE
blk=xMemBlockAlloc(XMEM_SUPERBLOCK_1META_MIN_SIZE+XMEM_SUPERBLOCK_2META_MIN_SIZE+XMEM_SUPERBLOCK_4META_MIN_SIZE+XMEM_SUPERBLOCK_8META_MIN_SIZE);
#else
blk=xMemBlockAlloc(XMEM_SUPERBLOCK_1META_MIN_SIZE+XMEM_SUPERBLOCK_2META_MIN_SIZE+XMEM_SUPERBLOCK_4META_MIN_SIZE);
#endif
if(blk)
{
xMemSuperBlockInit(&xMemSuperBlockList[0],
blk,
XMEM_SUPERBLOCK_1META_CNT,
XMEM_META_BLOCK_SIZE);
xMemSuperBlockInit(&xMemSuperBlockList[1],
(void *)((u32)blk+XMEM_SUPERBLOCK_1META_MIN_SIZE),
XMEM_SUPERBLOCK_2META_CNT,
XMEM_2META_BLOCK_SIZE);
xMemSuperBlockInit(&xMemSuperBlockList[2],
(void *)((u32)blk+XMEM_SUPERBLOCK_1META_MIN_SIZE+XMEM_SUPERBLOCK_2META_MIN_SIZE),
XMEM_SUPERBLOCK_4META_CNT,
XMEM_4META_BLOCK_SIZE);
#if XMEM_8META_ENABLE
xMemSuperBlockInit(&xMemSuperBlockList[3],
(void *)((u32)blk+XMEM_SUPERBLOCK_1META_MIN_SIZE+XMEM_SUPERBLOCK_2META_MIN_SIZE+XMEM_SUPERBLOCK_4META_MIN_SIZE),
XMEM_SUPERBLOCK_8META_CNT,
XMEM_8META_BLOCK_SIZE);
#endif
}
}
static const char xMemDumpMsgSuperBblock[]="-----SuperBlock Info-----\r\n";
static const char xMemDumpFmtSuperBlock[]="size:%d,free:%d,total:%d\r\n";
/***************************************************************************
* FUNCTION
* xMemSuperBlockInfoDump
* DESCRIPTION
* Dump super block lists information
* PARAMETERS
* void
* RETURNS
* void
* *************************************************************************/
static void xMemSuperBlockInfoDump(void)
{
u8 i;
xMemSuperBlock * psuperblock;
xMemInfoPrintf(xMemDumpMsgSuperBblock);
for(i=0;i<XMEM_SUPERBLOCK_LIST_COUNT;i++)
{
psuperblock=&xMemSuperBlockList[i];
while(psuperblock)
{
xMemInfoPrintf(xMemDumpFmtSuperBlock,psuperblock->blksize,psuperblock->nfree,psuperblock->nblk);
psuperblock=psuperblock->next;
}
}
xMemInfoPrintf(xMemDumpMsgSuperBblock);
}
/***************************************************************************
* FUNCTION
* xMallocMetaBlockGet
* DESCRIPTION
* Get a meta block
* PARAMETERS
* superblocklist [IN/OUT] super block list
* size [IN] Meta block size required
* RETURNS
* void * meta block
* *************************************************************************/
static void * xMallocMetaBlockGet(xMemSuperBlock * superblocklist,size_t size)
{
u32 i;
void *pblk;
xMemSuperBlock *pmemiter;
if (superblocklist == NULL||size>superblocklist->blksize) return NULL;
pmemiter=superblocklist;
while(pmemiter)
{
if (pmemiter->nfree > 0)
{
break;
}
pmemiter=pmemiter->next;
}
if(pmemiter==NULL)pmemiter=xMemSuperBlockAppend(superblocklist);
if(pmemiter)
{
for(i=0;i<pmemiter->nblk;i++)
{
if((1<<i)&(pmemiter->freeList))
{
pmemiter->freeList&=(~(1<<i));
break;
}
}
xMemAssert(i<pmemiter->nblk);
pblk = (void *)((u32)pmemiter->addr+i*(u32)(pmemiter->blksize));
//xMemDebugPrintf("get %u\r\n",pblk);
pmemiter->nfree--;
return (pblk);
}
return NULL;
}
/***************************************************************************
* FUNCTION
* xMallocMetaBlockPut
* DESCRIPTION
* put a meta block
* PARAMETERS
* superblocklist [IN/OUT] super block list
* pblk [IN] meta block be put
* RETURNS
* void
* *************************************************************************/
static void xMallocMetaBlockPut (xMemSuperBlock *superblocklist, void *pblk)
{
u8 i;
if (superblocklist == NULL||pblk == NULL||superblocklist->nfree >= superblocklist->nblk) return ;
i=((u32)pblk-(u32)superblocklist->addr)/superblocklist->blksize;
superblocklist->freeList|=(1<<i);
superblocklist->nfree++;
return;
}
/***************************************************************************
* FUNCTION
* xMallocMetaBlockAlloc
* DESCRIPTION
* Allocate a meta block
* PARAMETERS
* size [IN] meta block size that required
* RETURNS
* void * meta block
* *************************************************************************/
static void * xMallocMetaBlockAlloc(size_t size)
{
void * ptr;
#if XMEM_8META_ENABLE
if(size>XMEM_4META_BLOCK_SIZE)
{
ptr=xMallocMetaBlockGet(&xMemSuperBlockList[3],size);
}
else
#endif
if(size>XMEM_2META_BLOCK_SIZE)
{
ptr=xMallocMetaBlockGet(&xMemSuperBlockList[2],size);
}
else if(size>XMEM_META_BLOCK_SIZE)
{
ptr=xMallocMetaBlockGet(&xMemSuperBlockList[1],size);
}
else if(size>0)
{
ptr=xMallocMetaBlockGet(&xMemSuperBlockList[0],size);
}
else
{
return NULL;
}
if(ptr==NULL)
{
xMemSuperBlockInfoDump();
}
return ptr;
}
/***************************************************************************
* FUNCTION
* xMemMetaBlockFree
* DESCRIPTION
* free a meta block
* PARAMETERS
* pblk [IN] meta block be free
* RETURNS
* u8 0-success, 1-failure
* *************************************************************************/
static u8 xMemMetaBlockFree(void * pblk)
{
int i;
u32 start,end,p;
xMemSuperBlock *pmem,*pmemprev=NULL;
if (pblk == NULL) return 0;
for(i=0;i<XMEM_SUPERBLOCK_LIST_COUNT;i++)
{
pmem=&xMemSuperBlockList[i];
while(pmem)
{
p=(u32)pblk;
start=(u32)pmem->addr;
end=start+pmem->blksize*pmem->nblk;
if(p>=start&&p<end)
{
xMallocMetaBlockPut(pmem,pblk);
if(pmem->nfree==pmem->nblk&&pmemprev)
{
pmemprev->next=pmem->next;
xMemBlockFree(pmem->addr);
xMemBlockFree(pmem);
}
return 0;
}
pmemprev=pmem;
pmem=pmem->next;
}
}
return 1;
}
#endif
static u8 xmem_init_flag=0;
/***************************************************************************
* FUNCTION
* xMemInit
* DESCRIPTION
* Init X-Memory Pool
* PARAMETERS
* void
* RETURNS
* void
* *************************************************************************/
void xMemInit(unsigned long addr ,size_t size)
{
xMemDebugPrintf("xMem Version: %s\r\n",XMEM_VER);
xMem_Addr = addr;
xMem_Size = size;
#if XMEM_CORRUPT_CHECK == 0
xMemMgrHdrListInit();
#endif
xMemBlockListInit();
#if XMEM_SUPERBLOCK_ENABLE
xMemSuperBlockListInit();
#endif
xmem_init_flag=1;
return;
}
/***************************************************************************
* FUNCTION
* xmalloc
* DESCRIPTION
* allocate a memory block
* PARAMETERS
* size [IN] block size that required
* RETURNS
* void * memory block address
* *************************************************************************/
void * xmalloc(size_t size)
{
void * ptr = NULL;
xMemAssert(xmem_init_flag==1);
XMEM_ENTER_CRITICAL();
#if XMEM_CORRUPT_CHECK
xMemBlockListCheck();
#endif
#if XMEM_SUPERBLOCK_ENABLE
#if XMEM_8META_ENABLE
if(size<=XMEM_8META_BLOCK_SIZE)
#else
if(size<=XMEM_4META_BLOCK_SIZE)
#endif
{
ptr = xMallocMetaBlockAlloc(size);
}else{
#endif
ptr = xMemBlockAlloc(size);
#if XMEM_SUPERBLOCK_ENABLE
}
#endif
XMEM_EXIT_CRITICAL();
return ptr;
}
/***************************************************************************
* FUNCTION
* xfree
* DESCRIPTION
* free a memory block
* PARAMETERS
* void * [IN] memory block pointer
* RETURNS
* void
* *************************************************************************/
void xfree(void *ptr)
{
XMEM_ENTER_CRITICAL();
#if XMEM_CORRUPT_CHECK
xMemBlockListCheck();
#endif
if(
#if XMEM_SUPERBLOCK_ENABLE
xMemMetaBlockFree(ptr)&&
#endif
xMemBlockFree(ptr))
{//Meta block first, then common block, avoid super block start addr equals common block start addr
xMemDebugPrintf("prt:%u\r\n",(u32)ptr);
xMemSuperBlockInfoDump();
}
XMEM_EXIT_CRITICAL();
return;
}
/***************************************************************************
* FUNCTION
* xMemInfoDump
* DESCRIPTION
* Dump X-Memory Information, Header List information, Block List
* information, Super Block List information
* PARAMETERS
* void
* RETURNS
* void
* *************************************************************************/
void xMemInfoDump(void)
{
#if XMEM_CORRUPT_CHECK == 0
xMemDebugPrintf("hdr end:%u,blk start:%u\r\n",xMemMgrHdrListEnd,xMemBlkPoolStart);
xMemMgrHdrListInfoDump();
#endif
xMemBlockListInfoDump();
#if XMEM_SUPERBLOCK_ENABLE
xMemSuperBlockInfoDump();
#endif
xMemDebugPrintf("\r\n");
}
Loading...
马建仓 AI 助手
尝试更多
代码解读
代码找茬
代码优化