最近的一個項目中,體會了一下AVR單片機的外部RAM擴展,發(fā)文記錄于此。
本文以ATMEGA64單片機為測試平臺,外擴74HC573(資料上講用AHC系列,但是我用HC系列在16M晶振下配置成最快速度訪問并沒有發(fā)現(xiàn)什么不妥,當(dāng)然這是個不嚴(yán)謹(jǐn)?shù)脑O(shè)計)以及62256芯片。
我采用的是GCC編譯器,并且在項目中采用C++來編寫程序,根據(jù)理論上來講,只要外部存儲器使能了并且配置完成,那么變量的地址分配完全可以交給編譯器來做,但是前提是存儲器使能及配置代碼要在上電復(fù)位后最先得到執(zhí)行,而采用C或者C++來編寫程序,上電復(fù)位之后最先得到執(zhí)行的代碼是由編譯器自動增加的啟動代碼以及構(gòu)造函數(shù)代碼。我不清楚如何讓存儲器使能及配置代碼在上電復(fù)位后最先得到執(zhí)行,所以就寫了下面一段宏來控制外部存儲器的訪問。
代碼如下:
//extern_ram.h
#ifndef _H_EXTERN_RAM_H_
#define _H_EXTERN_RAM_H_
#include "type.h"
#define DEF_EX_VAR(name) EX_VAR name;
#define INIT_EX_VAR(name,ele_size,ele_n) do{
name.size_ele=ele_size;
name.n_ele=ele_n;
name.addr=p;
p+=((ele_size)*(ele_n));
}while(0)
#define WR_EX_VAR(name,pos,pdat) do {
for(UINT8 iiiii="0";iiiii
} while(0)
#define RD_EX_VAR(name,pos,pdat) do {
for(UINT8 iiiii="0";iiiii
} while(0);
#define GET_P_EX_VAR(name,pos) (name.addr+(pos)*name.size_ele)
typedef struct _tag_ex_var
{
UINT8 size_ele;
UINT16 n_ele;
INT8 *addr;
} EX_VAR;
#endif
應(yīng)用示例
//main.cpp
#include "includes.h"
INT8 *p=(INT8*)0x3100;//外部RAM的地址從0X3100開始
DEF_EX_VAR(test);//定義一個外部變量
INT16 main(void)
{
UINT8 i;
INT16 tmp;
INT16 *p_ele;
MCUCR|=0X80;//使能XRAM,并配制成最快速度訪問
INIT_EX_VAR(test,sizeof(INT16),100);//外部變量初始化,該變量有100個元素,元素大小為sizeof(INT16)
tmp="0xaaaa";
for(i=0;i<100;i++)
{
WR_EX_VAR(test,i,&tmp);//寫示例,對變量test的所有元素都賦值為0Xaaaa
}
for(i=0;i<100;i++)
{
RD_EX_VAR(test,i,&tmp);//讀示例
}
p_ele=( INT16 *)GET_P_EX_VAR(test,32);//獲取變量test的第32個元素的指針
while(1);
return 0x00;
}
后記:
1、頭文件請自行修改。
2、若要用于其它編譯器,請注意變量的內(nèi)存布局,本文所述采用大端模式。
3、局部變量的命名請不要使用iiiii,否則在宏展開的時候會被覆蓋。
4、文章比較粗糙,但是為了表示對作者勞動的尊重,轉(zhuǎn)帖請注明出處并保留版權(quán)信息