欧美极品高清xxxxhd,国产日产欧美最新,无码AV国产东京热AV无码,国产精品人与动性XXX,国产传媒亚洲综合一区二区,四库影院永久国产精品,毛片免费免费高清视频,福利所导航夜趣136
標題:
硬件和軟件兼容i2c協議的24Cxx系列EEPROM存儲器
[打印本頁]
作者:
51黑電子迷
時間:
2017-1-14 23:45
標題:
硬件和軟件兼容i2c協議的24Cxx系列EEPROM存儲器
硬件上由于24c01的A0A1A2管腳不允許懸空,故暫時的想法是兼容24c02 ---24c16
使用一個dip8封裝的芯片插座,A0 A1 A2管腳都懸空即可,換芯片方便
軟件上24c02地址只有8位,而其他型號是大于8位的,故地址參數使用16位
256個字節作為一個大頁,即largePage,測試芯片24c04空間有512字節
上代碼,求測試和討論
#include "MY51.H"
//轉載請注明: 求測試討論
//stc89c52rc,11.0592MHz晶振
sbit sda=P2^0; //總線連接口定義
sbit scl=P2^1; //總線連接口定義
void delayus() //需要4個機器周期,大概4.34us
{
; //晶振頻率11.0592M,機器周期為1.085微秒
}
void iic_start() //啟動信號
{
sda=1;
scl=1;
delayus(); //sda和scl同為高電平保持4.7us以上
_nop_(); //1.085us,共5.78us
sda=0; //下降沿
delayus(); //sda低電平保持4us以上 ,這里是4.34us滿足要求
}
void iic_stop() //停止信號
{
sda=0;_nop_(); //準備狀態
scl=1;
delayus(); //該狀態穩定時間要求保持4us以上
sda=1; //scl高電平期間,sda來一個上升沿
delayus(); //sda保持4.7us以上,4.34加上函數返回時間大于4.7us
//注:此時scl和sda都為1
}
void iic_sendByte(u8 byteData) //mcu發送一個字節
{
u8 i;
u8 temp=byteData;
for(i=0;i<8;i++)
{
temp=temp<<1; //移動后最高位到了PSW寄存器的CY位中
scl=0; //準備
_nop_(); //穩定一下
sda=CY; //將待發送的數據一位位的放到sda上
_nop_();
scl=1; //每一個高電平期間,ic器件都會將數據取走
_nop_();
}
scl=0; //如果寫成scl=1;sda=1就是停止信號,不能這么寫
_nop_();
sda=1; //釋放總線,數據總線不用時要釋放
_nop_();
}
u8 iic_readByte() //讀一個字節
{
u8 i,temp;
scl=0; //準備讀數據
_nop_();
sda=1; //釋放總線
_nop_();
for(i=0;i<8;i++)
{
scl=1; //mcu開始取數據
delayus(); //scl為高電平后,ic器件就會將1位數據送到sda上
//總共用時不會大于4.34us,然后就可以讓mcu讀sda了
temp=(temp<<1)|sda; //讀一位保存到temp中
scl=0;
delayus();
}
return temp;
}
bool iic_checkACK() //處理應答信號
{
u8 errCounts=255; //定義超時量為255次
scl=1;
_nop_();
while(sda) //在一段時間內檢測到sda=0的話認為是應答信號
{
if(0==errCounts)
{
scl=0; //鉗住總線
_nop_();
return FALSE; //沒有應答信號
}
errCounts--;
}
scl=0; //鉗住總線,為下1次通信做準備
_nop_();
return TRUE; //成功處理應答信號
}
void iic_init() //總線初始化
{
scl=1;
sda=1;
delayus();
}
void iic_sendACK(bool b_ACK) //發送應答或非應答信號
{
scl=0; //準備
_nop_();
if(b_ACK) //ACK 發送應該信號
{
sda=0;
}
else //unACK 發送非應答信號
{
sda=1;
}
_nop_();
scl=1;
delayus(); //大于4us的延時
scl=0; //鉗住scl,以便繼續接收數據
_nop_();
}
void AT24Cxx_writeByte(u16 address,u8 dataByte)//向24cxx寫一字節數據
{
u8 largePage = address/256; //24c04是512字節(尋址范圍0~511),largePage最大值是1
u8 addressOffset = address%256; //largePage=0的話地址范圍是(0~255)
iic_start();
iic_sendByte(0xa0|(largePage<<1));//控制字,前4位固定1010,后三位是器件地址,末位0是寫
iic_checkACK(); //mcu處理應答信號
iic_sendByte(addressOffset); //指定要寫入的器件內地址在 largePage塊中的偏移
iic_checkACK();
iic_sendByte(dataByte); //寫數據
iic_checkACK();
iic_stop();
delayms(2);
//按字節寫入時,24cxx在接收到停止信號后將數據擦寫到內部,這需要時間
//并且在這段時間內不會響應總線上的任何請求,故讓mcu有2毫秒以上的等待
}
void AT24Cxx_writeData(u16 address,u8 numBytes,u8* buf)//寫入任意長度數據(最大256字節)
{
while(numBytes--)
{
AT24Cxx_writeByte(address++,*buf++);
}
}
void AT24Cxx_readData(u16 beginAddr,u8 dataSize,u8* buf)//讀取任意長度字節到緩沖區buf中
{
u8 largePage = beginAddr/256; //計算largePage,256字節為一大頁
u8 addressOffset = beginAddr%256; //計算相對于largePage的偏移
iic_start(); //起始信號
iic_sendByte(0xa0|(largePage<<1)); //控制字,寫
iic_checkACK(); //處理應答信號
iic_sendByte(addressOffset); //要讀取的目標地址偏移
iic_checkACK(); //處理應答信號
iic_start(); //發送起始信號
iic_sendByte(0xa1|(largePage<<1)); //控制字,讀
iic_checkACK(); //處理應答信號
while(dataSize--) //讀取dataSize個字節,最大256個字節
{ //dataSize用u16類型會暴掉ram的
*buf++=iic_readByte(); //讀取一個個字節并保存到緩沖區buf中
iic_sendACK(dataSize); //發送應答,當dataSize為0時mcu發送非應答
}
iic_stop(); //發送停止信號
}
void main()//測試
{
u8 buf[3]; //接受數據的緩沖區
u8 arr[7]={0x06,1,2,3,4,0x55,0x33}; //待寫入的數據
iic_init(); //總線初始化
AT24Cxx_writeData(0x00+256,sizeof(arr),arr); //向指定地址處開始寫入7字節的數據
P1=0xff; //調試代碼,用P1口的led顯示
delayms(1000); //調試代碼
AT24Cxx_readData(0x00+256,sizeof(buf),buf); //從指定地址開始讀3個字節
P1=buf[2]; //也就是2 //led燈顯示數值
while(1)
{
P1=~P1;
delayms(500);
}
}
復制代碼
//my51.h中主要用到
#include<reg52.h>
#include"mytype.h"
void delayms(u16 ms) //軟延時函數
{
u16 i,j;
for(i=ms;i>0;i--)
{
for(j=113;j>0;j--)
{}
}
}
復制代碼
對代碼進行了改進
去掉了在寫數據時的
delayms(2);
這句軟延時代碼低效 ,而且沒有保障
改成加一個檢測函數
bool check_icWriteComplete() //檢測eeprom是否對內部擦寫完成
{
iic_start();
iic_sendByte(0xa0);
return iic_checkACK();
}
復制代碼
歡迎光臨 (http://m.raoushi.com/bbs/)
Powered by Discuz! X3.1