?? smbus_app_s.c
字號(hào):
///////////應(yīng)用SMBUS實(shí)現(xiàn)內(nèi)存互訪從機(jī)程序////////////////
////////Smbus_APP_R.c/////////////////////////////////
#include "smbus.h"
#include "absacc.h"
#define FLOATADR 0x0300
#define SLAVE_FLOATADR 0x0000
#define IIC_R 0x01
#define RETRY_MAX 50
/* IIC_R為SMBUS讀控制*/
#define IIC_W 0xfe
#define ISREAD 0x01
/* IIC_W為SMBUS寫控制*/
union int16{//采用聯(lián)合體實(shí)現(xiàn)16位與8位混合操作
uint word;
struct {uchar hi;uchar low;}split;
};
/*注意不能寫成如下代碼:
union int16{
uint word;
uchar hi;
uchar low;
};
這樣的結(jié)果是int16.hi與int16.low占用同一個(gè)地址,指向的均是int16.word高字節(jié)
*/
struct smbus_mes{
uchar id;
union int16 target_adr;
uint self_adr;
uchar len;
uchar isread;
uchar adr_end;
uchar isbuserror;
};
xdata struct smbus_mes mes_M,mes_S;//將結(jié)構(gòu)體定義到外部?jī)?nèi)存上
xdata float floattest _at_ FLOATADR;
xdata uchar *p;
void config(){
//看門狗禁止
WDTCN = 0x07;
WDTCN = 0xDE;
WDTCN = 0xAD;
SFRPAGE = 0x0F;
//交叉開關(guān)配置,SMBUS配置到P0.0和P0.1上
XBR0 = 0x01;
XBR1 = 0x00;
XBR2 = 0x40;
XBR3 = 0x01;
//管腳輸出配置,P0口為開漏輸出,其中P0.6接上拉電阻,P0為數(shù)字輸入口
SFRPAGE = 0x0F;
P0MDOUT = 0x00;
P1MDIN = 0xFF;
//晶振配置,采用內(nèi)部晶振8分頻
SFRPAGE = 0x0F;
CLKSEL = 0x00;
OSCXCN = 0x00;
OSCICN = 0x84;
}
void master_mes_creat(uchar id,uint target_adr,uint self_adr,uchar len,uchar dir){
mes_M.id=id<<1;//id數(shù)據(jù)在7-1位,須右移1位
mes_M.id&=IIC_W;//主機(jī)要發(fā)送從機(jī)的內(nèi)存地址,故起始必然是寫操作
mes_M.target_adr.word=target_adr;
mes_M.self_adr=self_adr;
mes_M.len=len;
mes_M.isread=dir;
mes_M.isbuserror=0;
}
void main(){
config();
mes_S.len=0x79;
smbus_cfg(0x44,0xf1,0x6f);
/*
smbus使能,AA=1,SCL高電平和低電平超時(shí)禁止
smbus時(shí)鐘頻率為100khz
自身從機(jī)地址為0x6e/2,即為55(十進(jìn)制),且廣播尋址使能
*/
EA=1;//打開全局中斷
SFRPAGE=0x00;
floattest=1.234567;
while(1);
}
void int0() interrupt 0{
}
void int1() interrupt 1{
}
void int2() interrupt 2{
}
void int3() interrupt 3{
}
void int4() interrupt 4{
}
void int5() interrupt 5{
}
void int6() interrupt 6{
}
void smbusInt() interrupt 7{//smbus中斷
//比較完善的SMBUS中斷處理程序,但是由于對(duì)各種狀態(tài)均作出處理,降低處理速度
//讀者可根據(jù)需要?jiǎng)h減一些狀態(tài)處理
static uchar retry_time=0;
SFRPAGE=0x00;
if(SMB0STA==0x08){//起始位發(fā)送成功,主機(jī)模式下才會(huì)出現(xiàn)這種狀態(tài)
SMB0DAT=mes_M.id;//將地址和讀寫控制裝入發(fā)送緩沖區(qū)
STA=0;//將STA清零,注意,若不清零則將一直為重復(fù)起始狀態(tài)
mes_M.adr_end=0;//內(nèi)存地址沒有發(fā)送完畢
goto bus_end;
}
if(SMB0STA==0x10){//重復(fù)起始條件處理
SMB0DAT=mes_M.id;
STA=0;
mes_M.adr_end=0;//內(nèi)存地址沒有發(fā)送完畢
goto bus_end;
}
if(SMB0STA==0x18){//從地址+W發(fā)出,收到從機(jī)ACK應(yīng)答
SMB0DAT=mes_M.target_adr.split.hi;//發(fā)送內(nèi)存高地址
mes_M.adr_end=0;//內(nèi)存地址沒有發(fā)送完畢
retry_time=0;//將重試計(jì)數(shù)器清零
goto bus_end;
}
if(SMB0STA==0x20){//從地址+W發(fā)出,收到從機(jī)NACK應(yīng)答
STO=1;
retry_time++;
if(retry_time<RETRY_MAX){
STA=1;//若重試次數(shù)小于最大限值,則產(chǎn)生重復(fù)起始條件
}
else{
mes_M.isbuserror=1;//isbuserror在初始化時(shí)須清零
retry_time=0;//若大于重試次數(shù),由于沒有將STA置1,不會(huì)產(chǎn)生重復(fù)起始條件
//且須將retry_time即時(shí)清零,以便下一次SMBUS中斷時(shí),重試次數(shù)從零開始計(jì)數(shù)
}
goto bus_end;
}
if(SMB0STA==0x28){//SMBUS數(shù)據(jù)成功發(fā)送,且收到從機(jī)ACK應(yīng)答
if(mes_M.len>0){
if(mes_M.adr_end==0){//若內(nèi)存地址沒有發(fā)送完畢,繼續(xù)發(fā)送內(nèi)存地址
SMB0DAT=mes_M.target_adr.split.low;//發(fā)送內(nèi)存低地址
mes_M.adr_end=2;
}
else{
if(mes_M.isread==ISREAD){
//更換角色,對(duì)從機(jī)寫完內(nèi)存地址后,讀取從機(jī)發(fā)送過來的數(shù)據(jù)
mes_M.id|=IIC_R;//將id中寫控制位改成讀控制位
STA=1;//此時(shí)不需要將STO置1
}
else{
SMB0DAT=XBYTE[mes_M.self_adr];
//取自身外部?jī)?nèi)存相應(yīng)地址上的數(shù)據(jù)發(fā)送到SMBUS總線上
mes_M.target_adr.word++;//對(duì)從機(jī)操作的內(nèi)存地址自增1
mes_M.self_adr++;//自身內(nèi)存地址自增1
mes_M.len--;//數(shù)據(jù)長(zhǎng)度減1
/*
主從機(jī)內(nèi)存地址自增能保證當(dāng)產(chǎn)生重復(fù)起始條件時(shí),對(duì)于成功發(fā)送的數(shù)據(jù)不再重發(fā)
*/
}
}
}
else{//若數(shù)據(jù)長(zhǎng)度等于0,則停止發(fā)送
STO=1;
retry_time=0;
}
goto bus_end;
}
if(SMB0STA==0x30){//SMBUS數(shù)據(jù)成功發(fā)送,但接收到從機(jī)NACK應(yīng)答
STO=1;
STA=1;//產(chǎn)生重復(fù)起始條件
goto bus_end;
}
if(SMB0STA==0x38){//總線競(jìng)爭(zhēng)失敗。
STO=1;//發(fā)送停止幀,并把總線錯(cuò)誤標(biāo)志位置1。
mes_M.isbuserror=1;
goto bus_end;
}
if(SMB0STA==0x40){//從機(jī)地址+R發(fā)送成功,接收到從機(jī)ACK應(yīng)答信號(hào)
if(mes_M.len>1)
AA=1;//下一接收數(shù)據(jù)不是最后數(shù)據(jù),故AA置1,以應(yīng)答從機(jī)
else{
if(mes_M.len==1)//下一數(shù)據(jù)為最后數(shù)據(jù)
AA=0;
}
retry_time=0;//將重試計(jì)數(shù)器清零
goto bus_end;
}
if(SMB0STA==0x48){//從機(jī)地址+R發(fā)送成功,但接收到從機(jī)NACK應(yīng)答信號(hào)
retry_time++;
STO=1;
if(retry_time<RETRY_MAX){//若小于重試最大限值,則繼續(xù)重試
STA=1;
}
else{
mes_M.isbuserror=1;//若大于重試最大限值,則停止總線,且將總線錯(cuò)誤標(biāo)志位置1
retry_time=0;
}
goto bus_end;
}
if(SMB0STA==0x50){//接收到從機(jī)所發(fā)送的數(shù)據(jù),并發(fā)送ACK應(yīng)答信號(hào)
if(mes_M.len>1)
AA=1;//下一接收數(shù)據(jù)不是最后數(shù)據(jù),故AA置1,以應(yīng)答從機(jī)
else{
if(mes_M.len==1)//下一數(shù)據(jù)為最后數(shù)據(jù)
AA=0;
}
XBYTE[mes_M.self_adr]=SMB0DAT;
mes_M.self_adr++;
mes_M.target_adr.word++;
mes_M.len--;
goto bus_end;
}
if(SMB0STA==0x58){//接收到從機(jī)所發(fā)送的最后一個(gè)數(shù)據(jù),并發(fā)送NACK應(yīng)答信號(hào)
XBYTE[mes_M.self_adr]=SMB0DAT;
mes_M.len--;
STO=1;//結(jié)束總線傳輸
goto bus_end;
}
if(SMB0STA==0x60){//在從機(jī)模式下,自身地址+W被尋址,并發(fā)送ACK應(yīng)答
//此時(shí)該節(jié)點(diǎn)為從機(jī)接收器
mes_S.adr_end=0;//被尋址后,首先接收的是內(nèi)存地址
AA=1;
goto bus_end;
}
if(SMB0STA==0x68){//在主機(jī)模式下發(fā)送從機(jī)地址+R/W時(shí)仲裁競(jìng)爭(zhēng)失敗,且被另一主機(jī)尋址
//此時(shí)該節(jié)點(diǎn)為從機(jī)接收器
mes_S.adr_end=0;
AA=1;
goto bus_end;
/*mes_M.len,沒有被改寫,所以在主程序中檢查到總線BUSY=0,且mes_M.len大于0,則發(fā)送起始位,將該節(jié)點(diǎn)變?yōu)橹鳈C(jī)*/
}
if(SMB0STA==0x70){//在從機(jī)模式下,被廣播尋址,并發(fā)送ACK應(yīng)答
//此時(shí)該節(jié)點(diǎn)為從機(jī)接收器
mes_S.adr_end=0;
AA=1;
goto bus_end;
}
if(SMB0STA==0x78){//在主機(jī)模式下發(fā)送從機(jī)地址+R/W時(shí)仲裁競(jìng)爭(zhēng)失敗,且被廣播尋址
//此時(shí)該節(jié)點(diǎn)為從機(jī)接收器
mes_S.adr_end=0;
AA=1;
goto bus_end;
/*處理方式與SMB0STA==0x68相同*/
}
if(SMB0STA==0x80){//在從機(jī)模式下,成功接收主機(jī)發(fā)送的數(shù)據(jù),并發(fā)送ACK應(yīng)答
if(mes_S.adr_end==0){//接收的數(shù)據(jù)為內(nèi)存高地址
mes_S.target_adr.split.hi=SMB0DAT;
mes_S.adr_end=1;
AA=1;
goto bus_end;
}
if(mes_S.adr_end==1){//接收的數(shù)據(jù)為內(nèi)存低地址
mes_S.target_adr.split.low=SMB0DAT;
mes_S.adr_end=2;
mes_S.self_adr=mes_S.target_adr.word;
AA=1;
goto bus_end;
}
if(mes_S.adr_end>=2){//接收數(shù)據(jù)
XBYTE[mes_S.self_adr]=SMB0DAT;
mes_S.len--;
mes_S.self_adr++;
/*判斷是否為最后一個(gè)數(shù)據(jù),一般來說從機(jī)mes_S.len可為0x7f,則數(shù)據(jù)長(zhǎng)度由主機(jī)決定,否則數(shù)據(jù)長(zhǎng)度由mes_S.len和對(duì)方主機(jī)mes_M.len的最小值決定*/
if(mes_S.len>1)
AA=1;
else
AA=0;
//////////////////////////////////////////
///////////////////////////////////////
//需要調(diào)試
goto bus_end;
}
}
if(SMB0STA==0x88){//在從機(jī)模式下,成功接收主機(jī)發(fā)送的數(shù)據(jù),且發(fā)送NACK應(yīng)答
XBYTE[mes_S.self_adr]=SMB0DAT;
mes_S.len=0;
STO=1;
goto bus_end;
}
if(SMB0STA==0x90){//在從機(jī)模式下,成功接收廣播主機(jī)發(fā)送的數(shù)據(jù),并發(fā)送ACK應(yīng)答
if(mes_S.adr_end==0){//接收的數(shù)據(jù)為內(nèi)存高地址
mes_S.target_adr.split.hi=SMB0DAT;
mes_S.adr_end=1;
AA=1;
goto bus_end;
}
if(mes_S.adr_end==1){//接收的數(shù)據(jù)為內(nèi)存低地址
mes_S.target_adr.split.low=SMB0DAT;
mes_S.adr_end=2;
mes_S.self_adr=mes_S.target_adr.word;
AA=1;
goto bus_end;
}
if(mes_S.adr_end>=2){//接收數(shù)據(jù)
XBYTE[mes_S.self_adr]=SMB0DAT;
mes_S.len--;
mes_S.self_adr++;
if(mes_S.len>1)
AA=1;
else
AA=0;
goto bus_end;
}
}
if(SMB0STA==0x98){//在從機(jī)模式下,成功接收廣播主機(jī)發(fā)送的數(shù)據(jù),且發(fā)送NACK應(yīng)答
XBYTE[mes_S.self_adr]=SMB0DAT;
STO=1;
goto bus_end;
}
if(SMB0STA==0xa0){//在從機(jī)模式下,接收到停止位或重復(fù)起始位
mes_S.len=0x7f;//以便在下一次被主機(jī)尋址時(shí)按主機(jī)長(zhǎng)度發(fā)送或接收
goto bus_end;
}
if(SMB0STA==0xa8){//在從機(jī)模式下,自身地址+R被尋址,并發(fā)送ACK應(yīng)答
SMB0DAT=XBYTE[mes_S.self_adr];
mes_S.self_adr++;
goto bus_end;
}
if(SMB0STA==0xb0){//在主機(jī)模式下發(fā)送從機(jī)地址+R/W時(shí)仲裁競(jìng)爭(zhēng)失敗,且被另一主機(jī)尋址
//此時(shí)該節(jié)點(diǎn)為從機(jī)發(fā)送器
SMB0DAT=XBYTE[mes_S.self_adr];
mes_S.self_adr++;
goto bus_end;
}
if(SMB0STA==0xb8){//在從機(jī)模式下,成功發(fā)送數(shù)據(jù),并接收到主機(jī)ACK應(yīng)答
SMB0DAT=XBYTE[mes_S.self_adr];
mes_S.self_adr++;
goto bus_end;
}
if(SMB0STA==0xc0){//在從機(jī)模式下,成功發(fā)送數(shù)據(jù),并接收到主機(jī)NACK應(yīng)答
goto bus_end;//數(shù)據(jù)發(fā)送完畢,等待停止位
}
if(SMB0STA==0x00){//出現(xiàn)總線錯(cuò)誤
mes_M.isbuserror=1;
mes_S.isbuserror=1;
STO=1;
}
bus_end:
SI=0;
}
?? 快捷鍵說明
復(fù)制代碼
Ctrl + C
搜索代碼
Ctrl + F
全屏模式
F11
切換主題
Ctrl + Shift + D
顯示快捷鍵
?
增大字號(hào)
Ctrl + =
減小字號(hào)
Ctrl + -