Keil C51內存分配與優化
C51內存常見的兩個誤區:
本文引用地址:http://cqxgywz.com/article/201611/322594.htm(1)
其實,只要不超過256字節,都可以用SMALL模式
(2)
其實,由于C51尋址的不同,高128字節也可以用來存儲變量,雖與SFR地址相同,但尋址的方式不同。
下面通過幾個程序來看內存的分配。
*******************************************************************************
//程序1:
#include
void main()
{}
Program Size: data=9.0 xdata=0 code=16
TYPE
*******************************************************************************
從上面可以看到,即使程序內部無任何變量和函數data也會為9.0。這9個字節內存分別為R0-R8和一個堆棧指針(C51的堆棧是“grow up”,即使堆棧中沒有內容,也會有一個棧底指針)。data區中由于R0-R8占有8個存儲空間,因此data區最大為120字節(棧在所有的變量空間 之后),如果超過120個字節則由idata顯式的指定為間接尋址。對于整個內部256字節的RAM,在極端的情況下,最大的變量為247字節。
當定義全局變量時
*******************************************************************************
//程序2:
#include
#define uint unsigned int
#define uchar unsigned char
uchar a;
uint b;
void main()
{}
Program Size: data=12.0 xdata=0 code=16
TYPE
*******************************************************************************
存在全局變量時,根據全局變量的類型分配相應的存儲空間。
看下面的程序
*****************************************************************
//程序3:
#include
#define uint unsigned int
#define uchar unsigned char
uchar a;
uint b;
uint sum(uint c)
{
}
void main()
{}
Program Size: data=12.0 xdata=0 code=17
TYPE
********************************************************************
與上面的程序想比較,發現內存并沒有任何的變化。
看下面的程序
***************************************************************************
//程序4:
#include
#define uint unsigned int
#define uchar unsigned char
uchar a;
uint b;
uint sum(uint c)
{
}
void main()
{
}
Program Size: data=12.0 xdata=0 code=28
TYPE
****************************************************************************
這與上面的內存使用相同,在這個程序中通過反匯編,查看編譯后的匯編程序可以發現,參數的傳遞通過通用寄存器完成,沒有占用新的內存。編譯器將其優 化的通用寄存器(寄存器一般傳遞3個參數,超過3個參數時,多余的參數通過分配空間地址的方式來訪問。但是分配的內存空間包含了寄存器傳遞的3個參數在內 的所有參數的空間。詳見《Parameter And Local Variable 》和《Parameter and Register》)和棧中(程序7),但是如果參數或局部變量過多,則情況就完全不同(程序6)。
再看下面的程序
*******************************************************************
//程序5
#include
#define uint unsigned int
#define uchar unsigned char
uchar a;
uint b;
uint sum(uint c)
{
}
void main()
{
}
Program Size: data=16.0 xdata=0 code=21
TYPE
******************************************************************
同樣的程序,內部RAM的使用情況又發生了變化。函數未被調用,且參數與局部變量都沒有使用。
源文件經過編譯后形成obj文件,各個obj文件就是一個模塊,每個模塊中都含有代碼段和數據段,也就是,代碼在ROM占有多少CODE空間,數據在RAM里占用多少空間等信息。


評論