最近在用1820單線溫度傳感器時(shí),老出現(xiàn)初始化不成功,請問一般是什么情況引起來的???
該芯片對時(shí)序要求比較嚴(yán)格,最好用匯編試試
我也一直懷疑是時(shí)序的問題,現(xiàn)在我換成12M的晶振后問題解決!多謝提示,也給大家?guī)c(diǎn)經(jīng)驗(yàn),1820的時(shí)序有點(diǎn)嚴(yán)格!希望大家注意!
#include #include #include #include #define uchar unsigned char #define uint unsigned int // 函數(shù)聲明區(qū) void init(void); //復(fù)位 void delay(uchar t); //延時(shí)time=(t+2)*5*Tm void delay_ms(uchar t); void write(uchar dat); //寫18b20 子程序 uchar read(void); //讀18b20 子程序 void ck(uchar *s, uchar len); //串口發(fā)送程序 uchar readtemp(uchar n); // 參數(shù)是DS18B20的序號 void initTimer(void); void timerInt(void); // 中斷服務(wù)程序 void LCD_C_D(bit flag,uchar ldata); void dis_one_zi(uchar x_ass,uchar y_add,uchar code *po); void CLRLCD(uchar number); // 變量定義區(qū) sbit P07 = P0^7; sbit P22 = P2^2; static uchar rom[8]; uchar nowdsp=0; // 當(dāng)前顯示溫度所對應(yīng)的傳感器編號 uchar timer=0; // 全局時(shí)鐘 uchar *out; uchar xiaoshu; uchar displayvalue[4]={'1','2','3','4'}; static uchar temp[3]; uchar code RomCode[8][8]={40,69,62,167,0,0,0,137 , 40,134,186,194,0,0,0,138, 0x28 ,0x0d,0x2e,0xfd,0x00,0x00,0x00,0xe2, 0x28 ,0x48,0x3c,0xfd,0x00,0x00,0x00,0x83, 0x28 ,0x92,0x64,0xfd,0x00,0x00,0x00,0x23, 0x28 ,0xb7,0xf6,0xd9,0x00,0x00,0x00,0x34, 0x28 ,0x74,0x6b,0xfd,0x00,0x00,0x00,0xd7, 0x28 ,0xff,0x61,0xfd,0x00,0x00,0x00,0x78 }; //8個(gè)傳感器的64為序列號
// 主函數(shù) main() { /* uchar i; init(); write(0x33); for(i=0;i<8;i++) rom[i]=read(); ck(rom ,8); //讀序列號 */ initTimer(); while(1) { for(nowdsp=0;nowdsp<8;nowdsp++) { init(); // DS18B20復(fù)位子程序 write(0xcc); // 跳過ROM write(0x44); // 啟動溫度轉(zhuǎn)換 temp[0]=readtemp(nowdsp); // 讀取溫度值 displayvalue[0]=temp[0]/10+0x30; displayvalue[1]=temp[0]%10+0x30; displayvalue[2]=xiaoshu+0x30; displayvalue[3]=nowdsp+0x31; ck(displayvalue,4); } } }
// 函數(shù)定義區(qū) void init(void) // 復(fù)位 { //display(temp[0]); EA=0; // 關(guān)閉中斷 P22=0; //總線為低電平 delay(94); //總線復(fù)位電平保持500us P22=1; //釋放總線 delay(12); //保持60us while(1) { if(P22==0) // 等于說明存在 { break; } else { P0=0x99; P2=0x00; } } /// 檢測到后的處理 delay(80); P22=1; EA=1; // 完成任務(wù)后允許中斷
} void write(uchar dat) //寫一個(gè)字節(jié)子程序 { uchar i=0; bit b; EA=0; // 關(guān)閉中斷 for(i=8;i>0;i--) { //display(temp[0]); b=dat&0x01; if(b==1) { // 寫1 P22=0; // 拉低總線6us _nop_(); _nop_(); _nop_(); _nop_(); _nop_(); _nop_(); P22=1; delay(11);//64us } else { // 寫0 P22=0; //拉低60us delay(10); P22=1; delay(0); //10us } dat>>=1; } EA=1; // 完成任務(wù)后允許中斷 }
uchar read(void) //讀一個(gè)字節(jié)子程序 { uchar dat=0 ; uchar i=0; EA=0; // 關(guān)閉中斷 for(i=0;i<8;i++) { P22=0; //拉低總線6us _nop_(); _nop_(); _nop_(); _nop_(); _nop_(); _nop_(); P22; // 釋放總線9us _nop_(); _nop_(); _nop_(); _nop_(); _nop_(); _nop_(); _nop_(); _nop_(); _nop_(); dat>>=1; if(P22) { dat=dat|0x80; } delay(9); } EA=1; // 完成任務(wù)后允許中斷 return(dat); }
void ck(uchar *s , uchar len) // 串口初始化發(fā)送子程序 { uchar j; TMOD=0x21; SCON=0x50; // PCON=0x80; TH1=0xF3; TL1=0xF3; TR1=1; if(len==0)len=strlen(s); //display(temp[0]); for(j=0;j { SBUF=s[j]; while(!TI); TI=0; } }
void delay(uchar t) //延時(shí)time=(t+2)*5*Tm { while(t--); }
void delay_ms(uchar t) { uchar i, k; for(i=0;i { k=165; while(k--); } }
uchar readtemp(uchar n) {
uchar i=0; EA=0; // 關(guān)閉中斷 init(); write(0x55); // 匹配rom for(i=0;i<8;i++) { write(RomCode[n][i]); } write(0xbe); // 讀入暫存 for(i=0;i<9;i++) { temp[i]=read(); } i = ((temp[0]>>4)|(temp[1]<<4)); temp[0]=temp[0]&0x0f; switch(temp[0]) { case 0 : xiaoshu=0;break; case 1 : xiaoshu=0;break; case 2 : xiaoshu=1;break; case 3 : xiaoshu=1;break; case 4 : xiaoshu=2;break; case 5 : xiaoshu=3;break; case 6 : xiaoshu=3;break; case 7 : xiaoshu=4;break; case 8 : xiaoshu=5;break; case 9 : xiaoshu=5;break; case 10 : xiaoshu=6;break; case 11 : xiaoshu=6;break; case 12 : xiaoshu=7;break; case 13 : xiaoshu=8;break; case 14 : xiaoshu=8;break; case 15 : xiaoshu=9;break; } EA=1; // 完成任務(wù)后允許中斷 return i; }
void initTimer(void) { TMOD=0x21; TH0=0x3c; TL0=0xb0; EA=1; ET0=1; TR0=1; }
void timerInt(void) interrupt 1 { if(timer==20) //每秒鐘更換一次顯示 { nowdsp=(nowdsp==7)?0:(nowdsp+1); timer=0; } TH0=0x3c; TL0=0xb0; TR0=1; timer++; }