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