内存管理 malloc free 的实现
libc 中提供非常好用的 malloc free 功能,如果自己实现一个,应该怎么做。
要实现 malloc free 需要有 可以分配内存使用的堆,和记录内存使用情况的链表。
如下图所示,堆从高向低分配,链表从低向高分配,图是 ps 画的。

/* 管理堆的链表 */
typedef struct A_BLOCK_LINK
{
char *pV; /* 内存实际物理地址 */
int index; /* 内存编号 */
char empty; /* 1 空 0 非空 */
int heapLen; /* 分配长度 */
struct A_BLOCK_LINK *pxNextFreeBlock; /* 上个节点 */
struct A_BLOCK_LINK *pxPrevFreeBlock; /* 下个节点 */
} BlockLink_t;
这里的对应关系是,链表 1 对应 最后一个堆,链表 2对应 最后2个堆。
如何初始化?
1,先计算出来,共能分配多少块内存
2,分配管理链表
如何找到合适的可以分配的内存?
这里从链表1 开始向后查找,当未使用的内存长度满足要求时,将最后找到的链表的,堆地址返回给 申请者。
如何free 内存?
因为 free 只传过来一个 ,最初申请的内存地址,没有传入长度,这里也需要在 链表结构体中进行记录
详细的代码,可以去置顶的 github 项目。
经过测试在, ubuntu16.4 gcc 5.4 中运行正常,free 后的内存,可以被重新 malloc 。
目前还没有实现,内存碎片整理功能。仅有一个空实现。 后期在更新。
实现代码:
#include <string.h>
#include <stdio.h>
#include "sram.h" //以下定义大小是为了直观的看到程序运行分配结果,取10进制数
//整个SRAM大小
#define SRAM_SIZE (1000) //堆分块大小
#define HeapBlockSIZE (100) //以下代码在 GCC 中演示使用编译器分一个大的SRAM 在实际硬件中,直接写内存开始地址就可以
//#define SRAM_BASE_ADDR 0
static char SRAM_BASE_ADDR[SRAM_SIZE] = {}; /* 管理堆的链表 */
typedef struct A_BLOCK_LINK
{
char *pV; /* 内存实际物理地址 */
int index; /* 内存编号 */
char empty; /* 1 空 0 非空 */
int heapLen; /* 分配长度 */
struct A_BLOCK_LINK *pxNextFreeBlock; /* 上个节点 */
struct A_BLOCK_LINK *pxPrevFreeBlock; /* 下个节点 */
} BlockLink_t; //头节点
static char *HeapHead = NULL;
//块数量
static int BlockTotal = ; void TraceHeap(void)
{
BlockLink_t *pTempBlockLink = (BlockLink_t *)HeapHead;
printf("\r\n##########TraceHeap#############\r\n");
while(pTempBlockLink)
{
printf("index: %d empty:%d addr:%d \r\n", pTempBlockLink->index, pTempBlockLink->empty, pTempBlockLink->pV - SRAM_BASE_ADDR);
pTempBlockLink = pTempBlockLink->pxNextFreeBlock;
}
printf("\r\n##########TraceHeap#############\r\n");
} //堆 Heap 分配初始化
void InitHeap(void)
{
BlockLink_t *pBlockLink, *pTempBlockLink = NULL;
int i = ;
char *pHeapStart = (char *)SRAM_BASE_ADDR;
char *pHeapEnd = (char *)(SRAM_BASE_ADDR + SRAM_SIZE);
pHeapEnd -= HeapBlockSIZE;
BlockTotal = SRAM_SIZE / (HeapBlockSIZE + sizeof(BlockLink_t));
while(i < BlockTotal)
{
pBlockLink = (BlockLink_t *)pHeapStart;
pBlockLink->pxPrevFreeBlock = pTempBlockLink;
pBlockLink->pV = pHeapEnd;
pBlockLink->index = i;
pBlockLink->empty = ;
pBlockLink->heapLen = ;
//指针重定位
pHeapEnd -= HeapBlockSIZE;
pHeapStart += sizeof(BlockLink_t);
//双向链表的上一个
pTempBlockLink = pBlockLink;
pBlockLink->pxNextFreeBlock = (BlockLink_t *)pHeapStart;
i++;
}
pBlockLink->pxNextFreeBlock = NULL;
HeapHead = (char *)SRAM_BASE_ADDR;
} static BlockLink_t *FindHeap(char *addr)
{
BlockLink_t *pTempBlockLink = (BlockLink_t *)HeapHead;
//从低向高查找可用的内存
while(pTempBlockLink)
{
if(pTempBlockLink->pV == addr)
{
return pTempBlockLink;
}
//切换下一节点
pTempBlockLink = pTempBlockLink->pxNextFreeBlock;
}
return NULL;
} //查找可用的连续内存
static void *SramFindHeap(int size)
{
char *mem;
int seriesSize = ; //已查找到连续用的内存
BlockLink_t *pTempBlockLink; //头节点
pTempBlockLink = (BlockLink_t *)HeapHead;
//从低向高查找可用的内存
while(pTempBlockLink)
{
//如果是未使用的内存
if(pTempBlockLink->empty)
{
seriesSize += HeapBlockSIZE;
}
else
{
seriesSize = ;
}
//如果查找到连续未使用的内存
if(seriesSize >= size)
{
//返回内存堆开始地址
pTempBlockLink->heapLen = seriesSize; //设置分配堆长度
mem = pTempBlockLink->pV; //将内存标记为 已使用
while(seriesSize && pTempBlockLink)
{
seriesSize -= HeapBlockSIZE;
pTempBlockLink->empty = ;
pTempBlockLink = pTempBlockLink->pxPrevFreeBlock;
}
return mem;
}
//切换下一节点
pTempBlockLink = pTempBlockLink->pxNextFreeBlock;
}
return NULL;
} //内存碎片整理
static void SramMerge(void)
{ } void *SramMalloc(size_t xWantedSize)
{
char *mem; if(! HeapHead)
{
InitHeap();
}
//地址对齐 mem = SramFindHeap(xWantedSize);
if(mem)
{
return mem;
}
//如果没有查找到 整理内存碎片
SramMerge(); //仍然分配不成功 返回错误
mem = SramFindHeap(xWantedSize);
if(mem)
{
return mem;
}
return NULL;
} void SramFree( void *pv )
{
int heapLen = ;
//释放内存 从堆的高位开始向低位查找
BlockLink_t *pTempHeap = FindHeap(pv);
heapLen = pTempHeap->heapLen;
while(heapLen && pTempHeap)
{
//设为空闲状态
pTempHeap->empty = ;
pTempHeap->heapLen = ;
//查找上个节点
pTempHeap = pTempHeap->pxPrevFreeBlock;
heapLen -= HeapBlockSIZE;
}
}
内存管理 malloc free 的实现的更多相关文章
- 操作系统动态内存管理——malloc和free的工作机制
动态内存分配 就 是指在程序执行的过程中动态地分配或者回收存储空间的分配内存的方法. malloc()是C语言中动态存储管理 的一组标准库函数之一.其作用是在内存的动态存储区中分配一个长度为size的 ...
- 内存管理malloc 2
malloc可以在函数指针内使用.#include <stdio.h> #include <stdlib.h> char * get_string() { //char s[] ...
- Linux内存管理原理
本文以32位机器为准,串讲一些内存管理的知识点. 1. 虚拟地址.物理地址.逻辑地址.线性地址 虚拟地址又叫线性地址.linux没有采用分段机制,所以逻辑地址和虚拟地址(线性地址)(在用户态,内核态逻 ...
- Linux内存管理原理【转】
转自:http://www.cnblogs.com/zhaoyl/p/3695517.html 本文以32位机器为准,串讲一些内存管理的知识点. 1. 虚拟地址.物理地址.逻辑地址.线性地址 虚拟地址 ...
- Windows内存管理和linux内存管理
windows内存管理 windows 内存管理方式主要分为:页式管理,段式管理,段页式管理. 页式管理的基本原理是将各进程的虚拟空间划分为若干个长度相等的页:页式管理把内存空间按照页的大小划分成片或 ...
- 【转帖】linux内存管理原理深入理解段式页式
linux内存管理原理深入理解段式页式 https://blog.csdn.net/h674174380/article/details/75453750 其实一直没弄明白 linux 到底是 段页式 ...
- Linux C 堆内存管理函数malloc()、calloc()、realloc()、free()详解
C 编程中,经常需要操作的内存可分为下面几个类别: 堆栈区(stack):由编译器自动分配与释放,存放函数的参数值,局部变量,临时变量等等,它们获取的方式都是由编译器自动执行的 堆区(heap):一般 ...
- 内存管理运算符new delete与内存管理函数malloc free的区别——已经他们对对象创建的过程。
(1)内存管理函数与内存管理运算符的区别 内存管理函数有内存分配函数,malloc calloc realloc 以及内存释放函数free. 内存管理运算符有new 和delete. 两种内存管理方式 ...
- <转载>内存管理内幕-动态分配的选择、折衷和实现 对malloc内存分配有个简单的描述,对内存管理有个大致的说明
这篇文章看后感觉不错,和我在glibc下的hurdmalloc.c文件里关于malloc的实现基本意思相同,同时,这篇文章还介绍了一些内存管理方面的知识,值得推荐. 原文链接地址为:http://ww ...
随机推荐
- z-index优先级小结
z-index是深度属性,设置元素在z轴上面的堆叠顺序. 强调:z-index必须和定位元素position:absollute|relative|fixed一起使用,否则无效 1.z-index属性 ...
- cs231n spring 2017 lecture5 Convolutional Neural Networks
1. 之前课程里,一个32*32*3的图像被展成3072*1的向量,左乘大小为10*3072的权重矩阵W,可以得到一个10*1的得分,分别对应10类标签. 在Convolution Layer里,图像 ...
- [LC] 490. The Maze
There is a ball in a maze with empty spaces and walls. The ball can go through empty spaces by rolli ...
- C语言学习笔记之动态分配数组空间
本文为原创文章,转载请标明出处 高级语言写多了,再拿起C语言的时候,自己已经傻了... C语言中数组大小不能为变量,即使这个变量已经被赋过值了,应该使用malloc方法进行数组空间动态分配. 如下: ...
- 谁来破解公交WiFi的尴尬困局?
在咖啡馆.饭店坐定,菜单可以待会再看,远远看见服务员迎上来,赶紧先问一句:"这里有无线网吗,密码是多少",和笔者一样养成类似习惯的人越来越多.所以,当公交WiFi"大 ...
- 红灯区:DevOps 建设的思考和实践
点击关注"有赞coder" 获取更多技术干货哦- 作者:费解 团队:效能改进 背景 众所周知,在丰田精益生产中,核心观念包含对人的尊重.消除浪费.持续改善,只有这样,企业才能保持良 ...
- python中字典dic详解-创建,遍历和排序
原文地址:http://www.bugingcode.com/blog/python_dic_create_sort.html 在python的编程中,字典dic是最典型的数据结构,看看如下对字典的操 ...
- hexo NexT主题首页title链接的优化
在默认设置下,文章链接都会改变,不利于搜索引擎收录,也不利于分享 更改index.swig文件 文件路径是your-hexo-sitethemesnextlayout,将下面代码 1 {% block ...
- 一个自动递增生成目录和文件的cop文件类
package com.hudong.util.orther; import java.io.File; import java.io.FileInputStream; import java.io. ...
- 免密码 ssh 到其它机器
背景:在配置 hadoop 的时候这样设置会比较方便.目标:A 机器上输入 ssh root@B 可以直接访问,不需要输入密码 步骤: 首先在 A 机器上生成密钥对,一路回车 1 ssh-keygen ...