網上流傳比較多的C51版本是龐波的《25045操作標準子程序集41.c》,但是經我實際使用過后發(fā)現(xiàn)有一些錯誤,如&和&&的區(qū)別及一些邏輯的問題。還有總是有人問電路的接法,由于X25045的datasheet寫的不清楚,的確很容易弄錯,這次就把接法的說明也貼上來了。希望能夠對大家有幫助。
分為兩個文件:X25045.h和X25045.C內容如下
1、X25045.h
#ifndef __X25045_H__
#define __X25045_H__
/*
;軟 件 標 題:25045操作標準子程序集
;軟 件 說 明:X25045已經停產,替代產品為X5045
;_________________________________________
;原作者:龐波
;程序修改人:貓賊
;版本號:
;_________________________________________
;貓賊:感謝原作者提供參考程序,雖然其中有一些錯誤,但是其公開源碼的方式值得敬佩。
;我沒有測試連續(xù)讀和連續(xù)寫程序,不知道會不會有問題,其他的函數(shù)都已經正確了。
*/
/*
參考電路說明:X25045/X5045
PIN1-CS :直接接到MCU的IO口
2-SO :直接接到MCU的IO口
3-WP :電阻上拉5V, 個人感覺接到MCU的IO口沒什么意義
4-VSS:GND
5-SI :直接接到MCU的IO口
6-SCK:直接接到MCU的IO口
7-RST:電阻上拉5V,該引腳直接與MCU的RST腳相連,注意該引腳不能接下拉電阻
8-VCC:+5V
應用范例:
環(huán)境:AT89C2051,+5V供電,6MHz晶振頻率
編譯器:偉福,Keil C
1.設置狀態(tài)寄存器(設置/開啟Watchdog)。狀態(tài)寄存器的值請設置STATUS_REG
void main(void){
wrsr_cmd();//復位時間位和數(shù)據保護位寫入狀態(tài)寄存器
//該語句最好放在起始位置,執(zhí)行完畢后Watchdog就設置為需要的時間參數(shù)。
//Watchdog其實在上電的時候就已經開始計時了,默認的時間為1.4秒。
//狀態(tài)寄存器值默認為00
//注意不要忘記喂狗rst_wdog();
........
}
2.寫/讀一個字節(jié)到EEPROM。
void main(void){
..........
byte_write(0x55,0x01);//寫一個字節(jié)數(shù)據0x55到的EEPROM,地址0x01
..........
Array_Vector = byte_read(0x01);//讀取0x01地址的內容,如果之前沒有做過改變,則應該為0X55
..........
}
*/
sbit SO =P1^3;/*25045輸出*/
sbit SI =P1^5;/*25045輸入*/
sbit SCK=P1^6;/*25045時鐘*/
sbit CS =P1^4;/*25045片選*/
#define STATUS_REG 0X00
/* Status register,設置DOG時間設置為1.4秒,無寫保護
這是狀態(tài)寄存器的值,他的意義在于第5,第4位為WDI1,WDI0代表DOG的時間,00為1.4秒,01為600毫秒,10為200毫秒,00為dISAbLED
第3位和第2位為BL1,BL0,是寫保護設置位,00為無保護,01為保護180-1FF,10為保護100-1FF,11為保護000-1FF.第1位為WEL,
當他為1時代表已經"寫使能"設置了,現(xiàn)在可以寫了,只讀位.第0位為WIP,當他為1時代表正在進行寫操作,是只讀*/
#define MAX_POLL0x99
/* Maximum number of polls
最大寫過程時間,確定25045的最大的寫入過程的時間*/
void wren_cmd(void);/*寫使能子程序*/
void wrdi_cmd(void);/*寫使能復位*/
void wrsr_cmd(void);/*復位時間位和數(shù)據保護位寫入狀態(tài)寄存器*/
unsigned char rdsr_cmd(void);/*讀狀態(tài)寄存器*/
void byte_write(unsigned char aa,unsigned int dd);/*字節(jié)寫入,aa為寫入的數(shù)據,dd為寫入的地址*/
unsigned char byte_read(unsigned int dd);/*字節(jié)讀出,dd為讀出的地址,返回讀出的數(shù)據*/
void page_write(unsigned char aa1,unsigned char aa2,unsigned char aa3,unsigned char aa4,unsigned int dd);/*頁寫入*/
void sequ_read(void);/*連續(xù)讀出*/
void rst_wdog(void);/*DOG復位*/
#endif/* __X25045_H__ */
2、X25045.C
#include
#include
#include "X25045.H"
#define uchar unsigned char
#define uint unsigned int
uchar code WREN_INST=0X06;
/* Write enable latch instruction (WREN)*/
uchar code WRDI_INST=0X04;
/* Write disable latch instruction (WRDI)*/
uchar code WRSR_INST=0X01;
/* Write status register instruction (WRSR)*/
uchar code RDSR_INST=0X05;
/* Read status register instruction (RDSR)*/
uchar code WRITE_INST=0X02;
/* Write memory instruction (WRITE)*/
/*寫入25045的先導字,應當為0000A010,其中的A為寫入25045的高位地址
將此WRITE_INST和寫入高位地址相或后即為正確的寫先導字*/
uchar code READ_INST=0X03;
/* Read memory instruction (READ)*/
/*讀出25045的先導字,應當為0000A011,其中的A為讀出25045的高位地址
將此READ_INST和讀出高位地址相或后即為正確的讀先導字*/
uint code BYTE_ADDR=0X55;
/* Memory address for byte mode operations*/
uchar code BYTE_DATA=0X11;
/*Data byte for byte write operation*/
uintcode PAGE_ADDR=0X1F;
/* Memory address for page mode operations*/
/*頁面寫入的其始地址*/
uchar code PAGE_DATA1=0X22;
/* 1st data byte for page write operation*/
uchar code PAGE_DATA2=0X33;
/* 2nd data byte for page write operation*/
uchar code PAGE_DATA3=0X44;
/* 3rd data byte for page write operation*/
uchar code INIT_STATE=0x09;
/* Initialization value for control ports*/
uint code SLIC=0x30;
/* AddressLOCation of SLIC*/
void outbyt(uchar aa);/*輸出一個字節(jié)到25045中,不包括先導字等*/
uchar inputbyt();/*由25045輸入一個字節(jié),不包括先導字等額外的東西*/
void wip_poll(void);/*檢查寫入過程是否結束*/
void delay(char n){
char a;
for(a=0;a
/*25045操作子程序集*/
/*;**********************************************************
*
;* Name: WREN_CMD
;* Description: Set write enable latch
;* Function: This routine sends the command to enable writes to the EEPROM memory array or
;* status register
;* Calls: outbyt
;* Input: None
;* Outputs: None
;* Register Usage: A
;**************************************************************
*/
/*寫使能子程序*/
void wren_cmd(void)
{
uchar aa;
SCK=0;/* Bring SCK low */
CS=0;/* Bring /CS low */
aa=WREN_INST;
outbyt(aa);/* Send WREN instruction */
delay(1);
SCK=0;/* Bring SCK low */
CS=1;/* Bring /CS high */
}
/*;********************************************************************
*
;* Name: WRDI_CMD
;* Description: Reset write enable latch
;* Function: This routine sends the command to disable writes to the EEPROM memory array or
;* status register
;* Calls: outbyt
;* Input: None
;* Outputs: None
;* Register Usage: A
;********************************************************************
*/
/*寫使能復位子程序*/
void wrdi_cmd(void)
{
uchar aa;
SCK=0;/* Bring SCK low */
CS=0;/* Bring /CS low */
aa=WRDI_INST;
outbyt(aa);/* Send WRDI instruction */
delay(1);
SCK=0;/* Bring SCK low */
CS=1;/* Bring /CS high */
}
/*;***********************************************************************
*
;* Name: WRSR_CMD
;* Description: Write Status Register
;* Function: This routine sends the command to write the WD0, WD1, BP0 and BP0 EEPROM
;* bits in the status register
;* Calls: outbyt, wip_poll
;* Input: None
;* Outputs: None
;* Register Usage: A
;**********************************************************************
*/
/*寫狀態(tài)寄存器子程序*/
void wrsr_cmd(void)
{
uchar aa;
wren_cmd();//寫使能子程序
SCK=0;/* Bring SCK low */
CS=0;/* Bring /CS low */
aa=WRSR_INST;
outbyt(aa) ;/* Send WRSR instruction */
aa=STATUS_REG;
outbyt(aa);/* Send status register */
delay(1);
SCK=0;/* Bring SCK low */
CS=1;/* Bring /CS high */
wip_poll();/* Poll for completion of write cycle */
wrdi_cmd();//寫使能復位,其實這句可以省略,每寫一次就自動復位
}
/*;*************************************************************************
*
;* Name: RDSR_CMD
;* Description: Read Status Register
;* Function: This routine sends the command to read the status register
;* Calls: outbyt, inputbyt
;* Input: None
;* Outputs: A = status registerXicor Application Note AN21
;* Register Usage: A
;********************************************************************
*/
/*讀狀態(tài)寄存器,讀出的數(shù)據放入到aa中*/
uchar rdsr_cmd (void)
{
uchar aa;
SCK=0;
CS=0;
aa=RDSR_INST;
outbyt(aa);
aa=inputbyt();
SCK=0;
CS=1;
return aa;
}
/*;**********************************************************************
*
;* Name: BYTE_WRITE
;* Description: Single Byte Write
;* Function: This routine sends the command to write a single byte to the EEPROM memory
array
;* Calls: outbyt, wip_poll
;* Input: None
;* Outputs: None
;* Register Usage: A, B
;***************************************************************
*/
/*字節(jié)寫入,aa為寫入的數(shù)據,dd為寫入的地址,對于25045而言為000-1FF*/
void byte_write(aa,dd)
uchar aa;
uint dd;
{
uchar tmp;
wren_cmd();//寫使能子程序
SCK=0;
CS=0;
if(dd>0xff)
tmp =WRITE_INST | 0x08;
else
tmp = WRITE_INST;
outbyt(tmp);/* Send WRITE instruction including MSB of address */
/*將高位地址左移3位與寫入先導字相或,得到正確的先導字寫入25045*/
outbyt((uchar)(dd&0xff));
/*輸出低位地址到25045*/
outbyt(aa);
/*寫入數(shù)據到25045的對應單元*/
SCK=0;
CS=1;
wip_poll();
/*檢測是否寫完*/
wrdi_cmd();//寫使能復位,其實這句可以省略,每寫一次就自動復位
}
/*;***********************************************************************
*
;* Name: BYTE_READ
;* Description: Single Byte Read
;* Function: This routine sends the command to read a single byte from the EEPROM memory
array
;* Calls: outbyt, inputbyt
;* Input: None
;* Outputs: A = read byte
;* Register Usage: A, BXicor Application Note AN21
;*********************************************************************
*/
/*字節(jié)讀出,其中dd為讀出的地址,返回的值為讀出的數(shù)據*/
uchar byte_read(dd)
uint dd;
{
uchar cc,tmp;
SCK=0;
CS=0;
if(dd>0xff)
tmp =READ_INST | 0x08;
else
tmp = READ_INST;
outbyt(tmp);/* Send READ_INST instruction including MSB of address */
/*將高位地址左移3位與讀出先導字相或,得到正確的先導字寫入25045*/
outbyt((uchar)(dd&0xff));
/*輸出低位地址到25045*/
cc=inputbyt();/*得到讀出的數(shù)據*/
SCK=0;
CS=1;
return cc;
}
/*;**********************************************************************
*
;* Name: PAGE_WRITE
;* Description: Page Write
;* Function: This routine sends the command to write three consecutive bytes to the EEPROM
;* memory array using page mode
;* Calls: outbyt, wip_poll
;* Input: None
;* Outputs: None
;* Register Usage: A, B
;**************************************************************************
*/
/*頁面寫入,其中aa1,aa2,aa3,aa4為需要寫入的4個數(shù)據(最大也就只能一次寫入4個字,dd為寫入的首地址*/
void page_write(aa1,aa2,aa3,aa4,dd)
uchar aa1,aa2,aa3,aa4;
uint dd;
{
SCK=0;
CS=0;
outbyt((((uchar)(dd-0XFF))<<3)|WRITE_INST);// Send WRITE instruction including MSB of address
//將高位地址左移3位與寫入先導字相或,得到正確的先導字寫入25045
outbyt((uchar)(dd));
//寫入低位地址到25045
outbyt(aa1);
//寫入數(shù)據1到25045的對應單元
outbyt(aa2);
//寫入數(shù)據2到25045的對應單元
outbyt(aa3);
//寫入數(shù)據3到25045的對應單元
outbyt(aa4);
//寫入數(shù)據4到25045的對應單元
SCK=0;
CS=1;
wip_poll();
}
/*;**********************************************
*
;* Name: SEQU_READ
;* Description: Sequential Read
;* Function: This routine sends the command to read three consecutive bytes from the EEPROM
;* memory array using sequential mode
;* Calls: outbyt, inputbyt
;* Input: None
;* Outputs: A = last byte read
;* Register Usage: A, B
;************************************************************
*/
/*連續(xù)讀出,由于函數(shù)的返回值只能為1個,對于連續(xù)讀出的數(shù)據只能使用指針作為函數(shù)的返回值才能做到返回一系列的數(shù)組*/
//sequ_read:
unsigned int *page_read(n,dd)
uchar n;//n是希望讀出的數(shù)據的個數(shù),n<=11
unsigned int dd;//dd是讀出數(shù)據的首地址
{
uchar i;
uchar pp[10];
unsigned int *pt=pp;
SCK=0;
CS=0;
outbyt((((uchar)(dd-0XFF))<<3)|READ_INST);
for (i=0;i
pp[i]=inputbyt();
}
return (pt);
}
/*調用的方法如下
unsigned int *p;
p=page_read(4,100);
a=*(p)
b=*(p+1)
c=*(p+2)
d=*(p+3)
//abcd中存放25045中由100地址開始的4個數(shù)據
*/
/* Send WRITE
mov DPTR, #PAGE_ADDR ; Set address of 1st byte to be read
clr sck ; Bring SCK low
clr cs ; Bring /CS low
mov A, #READ_INST
mov B, DPH
mov C, B.0
mov ACC.3, C
lcall outbyt ; Send READ instruction with MSB of address
mov A, DPL
lcall outbyt ; Send low order address byte
lcall inputbyt ; Read 1st data byte
lcall inputbyt ; Read 2nd data byte
lcall inputbyt ; Read 3rd data byte
clr sck ; Bring SCK low
setb cs ; Bring /CS high
ret*/
/*;*********************************************************************
*
;* Name: RST_WDOG
;* Description: Reset Watchdog Timer
;* Function: This routine resets the watchdog timer without sending a command
;* Calls: None
;* Input: None
;* Outputs: None
;* Register Usage: None
;***********************************************************************
*/
/*復位DOG*/
void rst_wdog (void)
{
CS=0;
CS=1;
}
/*;************************************************************************
*
;* Name: WIP_POLL
;* Description: Write-In-Progress Polling
;* Function: This routine polls for completion of a nonvolatile write cycle by examining the
;* WIP bit of the status register
;* Calls: rdsr_cmdXicor Application Note AN21
;* Input: None
;* Outputs: None
;* Register Usage: R1, A
;************************************************************************
*/
/*檢測寫入的過程是否結束*/
void wip_poll(void)
{
uchar aa;
uchar my_flag;
for (aa=0;aa
my_flag=rdsr_cmd();
if ((my_flag&0x01)==0) {aa=MAX_POLL;}/*判斷是否WIP=0,即判斷是否寫入過程已經結束,若結束就跳出,否則繼續(xù)等待直到達到最大記數(shù)值*/
}
//aa=1;
//while(aa){
//my_flag=rdsr_cmd();
//if ((my_flag&0x01)==0){
//aa=0;/*判斷是否WIP=0,即判斷是否寫入過程已經結束,若結束就跳出,否則繼續(xù)等待直到達到最大記數(shù)值*/
//}
//}
aa = 0;
}
/*;**************************************************************************
*
;* Name: OUTBYT
;* Description: Sends byte to EEPROM
;* Function: This routine shifts out a byte, starting with the MSB, to the EEPROM
;* Calls: None
;* Input: A = byte to be sent
;* Outputs: None
;* Register Usage: R0, A
;***********************************************************************
*/
/*輸出一個數(shù)據到25045,此數(shù)據可能為地址,先導字,寫入的數(shù)據等*/
void outbyt(aa)
uchar aa;
{
uchar my_flag1,my_flag2,i;
my_flag1=aa;
for (i=0;i<8;i++)
{
my_flag2=my_flag1&0x80 ;
SI=my_flag2>>7;
delay(1);
SCK=0;
SCK=1; delay(3);
my_flag1 <<= 1 ;
}
SI=0;/*使SI處于確定的狀態(tài)*/
}
/*;***************************************************************************
*
;* Name: INPUTBYT
;* Description: Recieves byte from EEPROM
;* Function: This routine recieves a byte, MSB first, from the EEPROM
;* Calls: None
;* Input: None
;* Outputs: A = recieved byte
;* Register Usage: R0, A
;**********************************************************************
*/
/*得到一個數(shù)據,此數(shù)據可能為狀態(tài)寄存器數(shù)據,讀出的單元數(shù)據等*/
uchar inputbyt(void)
{
uchar aa,my_flag;
char i;
aa = 0;
for (i=7;i>=0;i--)
{
SCK=0;
//delay(1);
my_flag=SO;
SCK=1;
//delay(1);
my_flag <<= i;
aa |= my_flag ;
my_flag=0x00;
}
return aa;
}