本帖最后由 51黑黑黑 于 2016-2-13 00:39 編輯

與其它逐次逼近式的AD操作類似,有幾個點需要主要:1.有8個通道,內部還有一個通道用于測溫;2.內部可產生一個1.2V電壓基準,并能x2輸出到Vref,但必須接到Vref0上,當然也可以外接基準電壓;3.有低功耗跟蹤方式可供選擇;4.內部有運放增益,對微弱信號特別有效

一般采用向ADBUSY寫1的方式進行轉換,以下是讀取AD值的函數:
uint GetAD0value(uchar channel)//0~7對應AIN0~7,8是溫度傳感器
{//AD采樣
uint ad0_value;
AMX0SL=channel;
AD0INT=0;
AD0BUSY=1;//啟動AD轉換
while(!AD0INT);
{
ad0_value=ADC0H;
ad0_value<<=8;
ad0_value+=ADC0L;//AD轉換結果
}
return ad0_value;
}
溫度的轉換利用以下關系式:[2.4*GetAD0value(8)-3177.7]/117;
----------------以下為SMBus介紹-------------------------------------
C8051F020的SMBus兼容I2C,因此可以方便地掛接多個I2C器件,如下圖:
主發送器方式:
主接收器方式:
SMB0STA:
bit SM_BUSY;//讀和寫時的忙標志位
void SMBus_Init()
{
SMB0CN=0x44;
SMB0CR=0xec;//頻率400k
EIE1|=0x02;
SM_BUSY=0;
} void SM_Send(uchar chip_select,uintbyte_address,uchar out_byte)
{//i2c寫驅動
while(SM_BUSY);//等待SMBus空閑
SM_BUSY=1;//占用SMBus(設置為忙)
SMB0CN=0x44;
BYTE_NUMBER=2;//2地址字節
COMMAND=chip_select;
HIGH_ADD=byte_address>>8;//高8位地址
LOW_ADD=byte_address&0x00FF;//低8位地址
WORD=out_byte;//待寫數據
STA=1;//啟動傳輸過程
while(SM_BUSY);//等待傳輸結束
} uchar SM_Receive(uchar chip_select,uintbyte_address)
{//i2c讀驅動
while(SM_BUSY);//等待總線空閑
SM_BUSY=1;//占用SMBus(設置為忙)
SMB0CN=0x44;
BYTE_NUMBER=2;//2地址字節
COMMAND=chip_select|1;
HIGH_ADD=byte_address>>8;//高8位地址
LOW_ADD=byte_address&0x00FF;//低8位地址
STA=1;//啟動傳輸過程
while(SM_BUSY);//等待傳輸結束
return WORD;
} void i2c_ISR() interrupt 7
{
switch (SMB0STA)
{// SMBus 狀態碼(SMB0STA 寄存器)
case 0x08://起始條件已發出
SMB0DAT=COMMAND&0xfe;//從器件的地址+W
STA=0;//手動清除START位
break;
case 0x10://重復起始條件已發出
SMB0DAT=COMMAND;//COMMAND中應保持從地址+R
STA=0;
break;
case0x18://從地址+W已發出,收到ACK
SMB0DAT=HIGH_ADD;//如果是EEPROM,裝入待寫存儲器地址的高字節
break;
case0x20://從地址+W已發出收到NACK
STO=1;
STA=1;
break;
case0x28://數據字節已發出收到ACK
switch (BYTE_NUMBER)
{
case 2://如果BYTE_NUMBER==2,
SMB0DAT=LOW_ADD;//發送了LOW_ADD
BYTE_NUMBER--;
break;
case1://如果BYTE_NUMBER==1,LOW_ADD已發送
if(COMMAND&0x01)
{//如果R/W=READ,發送重復起始條件
STO=0;
STA=1;
}
else
{
SMB0DAT=WORD;//如果R/W=WRITE,裝入待寫字節
BYTE_NUMBER--;
}
break;
default://如果BYTE_NUMBER==0,傳輸結束
STO=1;
SM_BUSY=0;//釋放SMBus
}
break;
case0x30://數據字節已發出收到NACK
STO=1;
STA=1;
break;
case0x40://從地址+R已發出收到ACK
AA=0;//只收一個字節,清AA
break;
case0x48://從地址+R已發出收到NACK
STA=1;
break;
case 0x58://數據字節收到NACK已發出
WORD=SMB0DAT;
STO=1;
SM_BUSY=0;//釋放SMBus
break;
default:
STO=1;//通信復位
SM_BUSY=0;
break;
}
SI=0;//清除中斷標志
}
|