keil c51的内部RAM(idata)动态内存管理程序
来源:互联网 发布:在淘宝开店怎么取消 编辑:程序博客网 时间:2024/05/22 03:11
程序比较简单,但感觉比较有意思,个人认为有一定应用价值,希望大家有更好的思路和方法,互相促进。
程序的基本思路是;
在CPU堆栈指针SP以上的RAM区域,通过把堆栈指针SP上移若干个字节,把空出的RAM区域供用户使用,当用户在使用完后又可以把该RAM区域释放。
头文件dmalloc51.h
/*
*********************************************************************************************************
* C51内部RAM动态内存申请函数 ,动态内存释放函数
* (c) Copyright 2004.6, LM7556,China
* All Rights Reserved
*
*
* 文件 : dmalloc51.h
*********************************************************************************************************
*/
//动态内存申请函数DMEM8U *dmalloc(DMEM8U dmsize,StkSize)从RAM堆栈中开辟一段空间做变量区.
//动态内存释放函数DMEM8U freedmalloc(DMEM8U *dmp,DMEM8U dmsize)把动态内存分配的空间从堆栈中释放。
/******************************** Define Messages *****************************************/
#define CPU_PCLEN 2 //CPU 程序指针长度(字节数).
#define RAM_SIZE 0x100 //CPU内部RAM字节数。
#define MEM_OVER 0xff //CPU内部RAM内存不够。
#define NO_MEM_DEL 0xfe //试图释放不存在的内存空间。
#define MEM_DELETED 0xfd //内存空间释放成功。
typedef unsigned char idata DMEM8U;
/************* 动态内存申请函数DMEM8U *dmalloc(DMEM8U dmsize,StkSize) *********************/
//该函数把堆栈SP向上移动dmsize字节空间,再返回指向该空间起始地址的指针,
//入口参数;dmsize是需要分配的空间大小,以DMEM8U为单位。
// StkSize 需要给CPU预留的堆栈空间.
//出口参数: 返回 STACK_OVER --- CPU内部RAM堆栈溢出,分配的空间不存在.
// 如果成功则返回一个指针,该指针指向所分配空间(大小为dmsize)的起始地址。
DMEM8U *dmalloc(DMEM8U dmsize,StkSize);
/************ 动态内存释放函数 freedmalloc(DMEM8U *dmp,DMEM8U dmsize) *********************/
//该函数把起始地址为dmp的空间(大小为dmsize)从堆栈中释放。
//入口参数;dmsize是需要释放的空间大小,以DMEM8U为单位。
// dmp指针指向需要释放空间的起始地址.
//出口参数: 返回MEM_DELETE --- 内存空间释放成功.
// 返回NO_MEM_DEL --- 试图释放不存在的内存空间.
DMEM8U freedmalloc(DMEM8U *dmp,DMEM8U dmsize);
主文件 : dmalloc51.c
/*
*********************************************************************************************************
* C51内部RAM动态申请函数 ,动态释放函数
* (c) Copyright 2004.6, LM7556,China
* All Rights Reserved
*
*
* 文件 : dmalloc51.c
*********************************************************************************************************
*/
#include "dmalloc51.h"
sfr SP = 0x81;
/************* 动态内存申请函数DMEM8U *dmalloc(DMEM8U dmsize,StkSize) *********************/
//动态内存申请函数DMEM8U *dmalloc(DMEM8U dmsize,StkSize)从RAM堆栈中开辟一段空间做变量区.
//该函数把堆栈SP向上移动dmsize字节空间,再返回指向该空间起始地址的指针,
//入口参数;dmsize是需要分配的空间大小,以DMEM8U为单位。
// StkSize 需要给CPU预留的堆栈空间.
//出口参数: 返回 STACK_OVER --- CPU内部RAM堆栈溢出,分配的空间不存在.
// 如果成功则返回一个指针,该指针指向所分配空间(大小为dmsize)的起始地址。
DMEM8U *dmalloc(DMEM8U dmsize,StkSize)
{DMEM8U *p1,*p2,spbuf; unsigned int StkChk;
StkChk=SP+dmsize+StkSize;
if (StkChk>(RAM_SIZE-1)) {return MEM_OVER;} //检查要申请的内存是否存在.
spbuf=SP;
spbuf-=CPU_PCLEN;
p1=(DMEM8U*)spbuf;
p2=p1+dmsize;
spbuf=(DMEM8U)p2+CPU_PCLEN;
SP=spbuf;
//把上一级函数的返回地址移到该空间的上面,使本函数自己在执行完后可以正确的返回到上一级函数.
p1++;p2++;
*p2++=*p1++;
*p2=*p1;
p1--;
return p1;
}
/************ 动态内存释放函数 freedmalloc(DMEM8U *dmp,DMEM8U dmsize) *********************/
//动态内存释放函数DMEM8U freedmalloc(DMEM8U *dmp,DMEM8U dmsize)把动态内存分配的空间从堆栈中释放。
//该函数把起始地址为dmp的空间(大小为dmsize)从堆栈中释放。
//入口参数;dmsize是需要释放的空间大小,以DMEM8U为单位。
// dmp指针指向需要释放空间的起始地址.
//出口参数: 返回MEM_DELETE --- 内存空间释放成功.
// 返回NO_MEM_DEL --- 试图释放不存在的内存空间.
DMEM8U freedmalloc(DMEM8U *dmp,DMEM8U dmsize)
{DMEM8U *p1,*p2,spbuf,i;
spbuf=(DMEM8U)dmp+dmsize;
if (spbuf<=(DMEM8U)dmp) {return NO_MEM_DEL;}
if (spbuf>SP-CPU_PCLEN) {spbuf=SP-CPU_PCLEN;}
i=SP-spbuf;
p1=(DMEM8U*)spbuf;
p1++;
p2=dmp;
spbuf=(SP-spbuf)+p2;
spbuf--;
while (i)
{
*p2++=*p1++; //把上一级函数的返回地址移到该空间的底部,使程序可以返回到上一级函数.
i--;
}
SP=spbuf;
return MEM_DELETED;
}
应用范例;
文件 : main.c
#include <reg52.h>
#include <stdio.h>
#include <intrins.h>
#define STACK_SIZE 0x30 //为程序预留的最小堆栈.
#include "dmalloc51.h"
void initsio(void);
void TDelay(unsigned int t);
void Fn_dRamA(void);
void Fn_dRamB(void);
void Fn_dRamC(void);
void main(void)
{
initsio();
while (1)
{
TDelay(500);
Fn_dRamA();
printf("\n");
TDelay(500);
Fn_dRamB();
printf("\n");
TDelay(500);
Fn_dRamC();
printf("\n");
}
}
#define dmSIZE_A 0x20
void Fn_dRamA(void)
{unsigned char i,*mp,*mpp;
if ((mpp=dmalloc(dmSIZE_A,STACK_SIZE))==MEM_OVER) {return;};
mp=mpp;
for (i=0;i<dmSIZE_A-1;i++)
{
*mp++=i+0x20;
}
*mp=0;
printf(mpp);
printf("\n");
Fn_dRamB();
freedmalloc(mpp,dmSIZE_A); //后面加nop()为了防止c51把freedmalloc(mpp)做ljmp 调用。
_nop_();
}
#define dmSIZE_B 0x20
void Fn_dRamB(void)
{unsigned char i,*mp,*mpp;
if ((mpp=dmalloc(dmSIZE_B,STACK_SIZE))==MEM_OVER) {return;};
mp=mpp;
for (i=0;i<dmSIZE_B-1;i++)
{
*mp++=i+0x40;
}
*mp=0;
printf(mpp);
printf("\n");
Fn_dRamC();
freedmalloc(mpp,dmSIZE_B); //后面加nop()为了防止c51把freedmalloc(mpp)做ljmp 调用。
_nop_();
}
#define dmSIZE_C 0x20
void Fn_dRamC(void)
{unsigned char i,*mp,*mpp;
if ((mpp=dmalloc(dmSIZE_C,STACK_SIZE))==MEM_OVER) {return;};
mp=mpp;
for (i=0;i<dmSIZE_C-1;i++)
{
*mp++=i+0x60;
}
*mp=0;
printf(mpp);
printf("\n");
freedmalloc(mpp,dmSIZE_C); //后面加nop()为了防止c51把freedmalloc(mpp)做ljmp 调用。
_nop_();
}
void TDelay(unsigned int t)
{unsigned int i,j;
for (i=0;i<t;i++)
{
for (j=0;j<t;j++) {}
}
}
void initsio(void)
{
TMOD=TMOD&0x0F;
TMOD=TMOD|0x20;
TL1=0xFD,TH1=0xFD;//19200 , 22.1184MHz
SCON=0x50;PCON=0x00;
TR1=1;
TI = 1; /* TI: set TI to send first char of UART */
}
http://www.61ic.com/Technology/embed/200604/3329.html
- keil c51的内部RAM(idata)动态内存管理程序
- Keil C51对C语言的关键词扩展之九: idata
- Keil C51内存模型
- Keil C51内存
- KEIL查看内存ram
- keil c51中的data idata xdata code详解
- C51 keil中data,idata,xdata,pdata,code
- 80c51内部RAM空间分配
- Keil C51 的printf
- 关于C51内的code,idata,xdata
- C51和C52的RAM
- 【转】浅谈C51内存优化(data idata xdata)
- KEIL C51的XBYTE关键字
- Keil C51 的命令行调用
- 万恶的keil(c51)
- Keil C51 的printf sprintf
- KEIL MDK C51的区别
- C51中的IDATA
- leetcode之路002 Add Two Numbers
- iOS前期OC训练OC_06字典
- 二分 hdu2141 Can you find it?
- 数据结构:图的存储结构之邻接矩阵(摘自网易云课堂)
- 【POJ】【2019】
- keil c51的内部RAM(idata)动态内存管理程序
- 动态规划 钢条切割问题 两种方法 自底而上 自上而下的方法
- 编写可变参数c函数
- 真是他们知道的哪个胆怯怯懦的龙景吗
- Should I normalize/standardize/rescale the data
- Codeforces Round #313 (Div. 2)D
- 15/7/23/文件输入输出/DOM/SAX
- Python Split函数的用法总结
- curl学习(实例不断总结)