DIY高精度時鐘、溫度顯示器Ds3231+12864+ds18b20+89c51 硬件: 程序: #include<reg52.h> #include<intrins.h> #include<stdlib.h>
#define uchar unsigned char #define uint unsigned int /*端口定義*/ sbit LCD_RS=P2^6; sbit LCD_RW=P2^5; sbit LCD_EN=P2^4; sbit LCD_PSB=P2^3; sbit DQ=P3^7; //18b20 sbit SDA=P1^4; //ds32321 //模擬I2C數據傳送位SDA sbit SCL=P1^3; //模擬I2C時鐘控制位SCL //***按鍵功能****// ////***K1停止時間顯示****// ////***K2選擇修改位置****// ////***K3進行加1的修改****// ////***K4將修改寫入ds3231,同時啟動時間顯示****// sbit K1=P3^2; sbit K2=P3^3; sbit K3=P3^4; sbit K4=P3^5; //定義變量 uchar numbr[10]="0123456789"; //字模
uchar dis4[]=" "; // 第四行顯示 自己添加 uchar t[]=" . ℃" ; //18b20 uint sdata,xiaoshu1,xiaoshu2; //整數、小數1位、小數2位 bit fg=1; //溫度正負標志 uchar tempL=0,tempH=0; // 變量 uchar year,month,date,hour,min,sec; // ds3231 uchar a[]="2011年22月33日"; uchar b[]="11時22分33秒"; ///函數 //******************延時子程序 *******************************
//這個延時程序的具體延時時間是time=i*8+10,適用于小于2ms的延時
//************************************************************ void delay(unsigned char i) { for(i;i>0;i--); } //*********************************************************** // 延時子程序 //************************************************************ void delay1ms(uchar j) { while(j!=0) {uchar i; for(i=124;i>0;i--); //延時124*8+10=1002us j--; } } /**************************12864部分*************************************/ /**************************12864部分*************************************/ /*寫指令數據到LCD RS=L——表示DB0-DB7為顯示指令數據 RW=L——表示DB0-DB7數據被write(當E=“H-L”,指令數據被寫到IR或DR) E=高脈沖 此時DB0-DB7=指令碼 */ void write_cmd(uchar cmd) { LCD_RS=0; LCD_RW=0; LCD_EN=0; P0=cmd; delay1ms(5); LCD_EN=1; delay1ms(5); LCD_EN=0; } /*設定顯示位置*/ void lcd_pos(uchar X, uchar Y) { ucharpos;
if(X== 0) { X= 0x80; } elseif(X == 1) { X= 0x90; } elseif(X == 2) { X= 0x88; } elseif(X == 3) { X= 0x98; } pos= X + Y; write_cmd(pos); //顯示地址 }
/*寫顯示數據到LCD*/ /* RS=H——表示DB0-DB7為顯示數據 RW=L——RW=L,E='H-L',DB0-DB7數據被寫到IR或DR E=高脈沖 DB0-DB7=顯示數據 */ void write_dat(uchar dat) { LCD_RS=1; LCD_RW=0; LCD_EN=0; P0=dat; delay1ms(5); LCD_EN=1; delay1ms(5); LCD_EN=0; } /*LCD初始化*/ void lcd_init() { uinti;
LCD_PSB=1; //并口方式 write_cmd(0x30); //基本操作指令 delay1ms(5); write_cmd(0x0c); //打開顯示,光標關閉 delay1ms(5); write_cmd(0x01); //清除LCD顯示類容 delay1ms(5);
lcd_pos(3,0); i=0; while(dis4[ i]!='\0') { delay1ms(1); write_dat(dis4); delay1ms(1); i++; } } /**********************************18b20************************************************/ /**********************************18b20************************************************/ void Init_DS18B20(void) //初始化 { uchar x=0; DQ=1; //DQ先置高 delay(8); //稍延時 DQ=0; //發送復位脈沖 delay(80); //延時(>480us) DQ=1; //拉高數據線 delay(5); //等待(15~60us) x=DQ; //用X的值來判斷初始化有沒有成功,18B20存在的話X=0,否則X=1 delay(20); } //**********讀一個字節************// ReadOneChar(void) //主機數據線先從高拉至低電平1us以上,再使數據線升為高電平,從而產生讀信號 { unsigned char i=0; //每個讀周期最短的持續時間為60us,各個讀周期之間必須有1us以上的高電平恢復期 unsigned char dat=0; for (i=8;i>0;i--) //一個字節有8位 { DQ=1; delay(1); DQ=0; dat>>=1; DQ=1; if(DQ) dat|=0x80; delay(4); } return(dat); } //*********************** **寫一個字節**************************// void WriteOneChar(unsigned char dat) { unsigned char i=0; //數據線從高電平拉至低電平,產生寫起始信號。15us之內將所需寫的位送到數據線上, for(i=8;i>0;i--) //在15~60us之間對數據線進行采樣,如果是高電平就寫1,低寫0發生。 { DQ=0; //在開始另一個寫周期前必須有1us以上的高電平恢復期。 DQ=dat&0x01; delay(5); DQ=1; dat>>=1; } delay(4); } void ReadTemperature(void) //讀溫度值(低位放tempL;高位放tempH;)// { Init_DS18B20(); //初始化 WriteOneChar(0xcc); //跳過讀序列號的操作 WriteOneChar(0x44); //啟動溫度轉換 delay(125); //轉換需要一點時間,延時 Init_DS18B20(); //初始化 WriteOneChar(0xcc); //跳過讀序列號的操作 WriteOneChar(0xbe); //讀溫度寄存器(頭兩個值分別為溫度的低位和高位) tempL=ReadOneChar(); //讀出溫度的低位LSB tempH=ReadOneChar(); //讀出溫度的高位MSB if(tempH>0x7f) //最高位為1時溫度是負 { tempL=~tempL; //補碼轉換,取反加一 tempH=~tempH+1; fg=0; //讀取溫度為負時fg=0 } sdata= tempL/16+tempH*16; //整數部分 xiaoshu1= (tempL&0x0f)*10/16; //小數第一位 xiaoshu2= (tempL&0x0f)*100/16%10;//小數第二位 t[0]=numbr[sdata/10]; t[1]=numbr[sdata%10]; t[3]=numbr[xiaoshu1]; t[4]=numbr[xiaoshu2]; } /*****************************************ds3231********************************************/ #define ADDRTW 0xD0 //器件寫地址 #define ADDRTD 0xD1 //器件讀地址 #define DS3231_SEC 0x00 //秒 #define DS3231_MIN 0x01 //分 #define DS3231_HOUR 0x02 //時 #define DS3231_DAY 0x03 //星期 #define DS3231_DATE 0x04 //日 #define DS3231_MONTH 0x05 //月 #define DS3231_YEAR 0x06 //年 //鬧鈴1 #define DS3231_Al1SEC 0x07 //秒 #define DS3231_AL1MIN 0x08 //分 #define DS3231_AL1HOUR 0x09 //時 #define DS3231_AL1DAY 0x0A //星期/日 //鬧鈴2 #define DS3231_AL2MIN 0x0b //分 #define DS3231_AL2HOUR 0x0c //時 #define DS3231_AL2DAY 0x0d //星期/日 #define DS3231_CONTROL 0x0e //控制寄存器 #define DS3231_STATUS 0x0f //狀態寄存器 bit ack; uchar BCD2HEX(uchar val) //BCD轉換為Byte { uchari; i= val&0x0f; val >>= 4; val &= 0x0f; val *= 10; i+= val; return i; } uchar HEX2BCD(uchar val)//B碼轉換為BCD碼 { uchari,j,k; i=val/10; j=val%10; k=j+(i<<4); return k; } void Start() { SDA=1; //發送起始條件的數據信號 delay(1); SCL=1; delay(5); //起始條件建立時間大于4.7us,延時 SDA=0; //發送起始信號 delay(5); // 起始條件鎖定時間大于4μs SCL=0; //鉗住I2C總線,準備發送或接收數據 delay(2); } void Stop() { SDA=0; //發送結束條件的數據信號 delay(1); //發送結束條件的時鐘信號 SCL=1; //結束條件建立時間大于4us delay(5); SDA=1; //發送I2C總線結束信號 delay(4); } /********************************************************/ /******************************************************************* 字節數據發送函數 函數原型: void SendByte(uchar Dat); 功能: 將數據c發送出去,可以是地址,也可以是數據,發完后等待應答,并對 此狀態位進行操作.(不應答或非應答都使ack=0) ack=1 發送數據正常, ack=0 被控器無應答或損壞。 ********************************************************************/ void SendByte(uchar Dat) { uchar BitCnt; for(BitCnt=0;BitCnt<8;BitCnt++) //要傳送的數據長度為8位 { if((Dat<<BitCnt)&0x80) SDA=1; //判斷發送位 else SDA=0; delay(1); SCL=1; //置時鐘線為高,通知被控器開始接收數據位 delay(5); //保證時鐘高電平周期大于4μs SCL=0; } delay(2); SDA=1; //8位發送完后釋放數據線,準備接收應答位 delay(2); SCL=1; delay(3); if(SDA==1) ack=0; else ack=1; //判斷是否接收到應答信號 SCL=0; delay(2); } uchar RcvByte() //功能: 用來接收從器件傳來的數據,并判斷總線錯誤(不發應答信號),發完后請用應答函數應答從機。 { uchar retc; uchar BitCnt; retc=0; SDA=1; //置數據線為輸入方式 for(BitCnt=0;BitCnt<8;BitCnt++) { delay(1); SCL=0; //置時鐘線為低,準備接收數據位 delay(5); //時鐘低電平周期大于4.7μs SCL=1; //置時鐘線為高使數據線上數據有效 delay(3); retc=retc<<1; if(SDA==1) retc=retc+1; //讀數據位,接收的數據位放入retc中 delay(2); } SCL=0; delay(2); return(retc); } void I2CACK(bit a) // 功能: 主控器進行應答信號(可以是應答或非應答信號,由位參數a決定) { if(a==0) SDA=0; //在此發出應答或非應答信號 else SDA=1; delay(3); SCL=1; delay(5); //時鐘低電平周期大于4μs SCL=0; //清時鐘線,鉗住I2C總線以便繼續接收 delay(2); } uchar I2CRead() /************從DS3231當前地址讀一個字節************/ { uchar read_data; Start(); SendByte(ADDRTD); if(ack==0) { return(0); } read_data = RcvByte(); I2CACK(1); Stop(); return read_data; } uchar I2CReadAdd(uchar addr) /************從DS3231指定地址讀一個字節************/ { Start(); SendByte(ADDRTW); if(ack==0) { return(0); } SendByte(addr); if(ack==0) { return(0); } return(I2CRead()); } void Readtime() /*********************讀取時間**********************/ { uchar temp; temp=I2CReadAdd(DS3231_SEC);//秒 sec=BCD2HEX(temp); temp=I2CReadAdd(DS3231_MIN);//分 min=BCD2HEX(temp); temp=I2CReadAdd(DS3231_HOUR); //時 hour=BCD2HEX(temp); temp=I2CReadAdd(DS3231_DATE); //日 date=BCD2HEX(temp); temp=I2CReadAdd(DS3231_MONTH); //月 month=BCD2HEX(temp); temp=I2CReadAdd(DS3231_YEAR); //年 year=BCD2HEX(temp); } void InitDS3231() //ds3231初始化 {SCL=1; delay(5); SDA=1; delay(5); } void TimeDisplay(uchar Dhour,ucharDmin,uchar Dsec) //時分秒數組賦值 { b[0]=numbr[Dhour / 10]; // 時十位 b[1]=numbr[Dhour % 10]; // 時個位 b[4]=numbr[Dmin / 10]; // 分十位 b[5]=numbr[Dmin % 10]; // 分個位 b[8]=numbr[Dsec / 10]; // 秒十位 b[9]=numbr[Dsec % 10]; // 秒個位 } void DateDisplay(uchar Dyear,ucharDmonth,uchar Dday) //年月天數組賦值 { a[2]=numbr[Dyear / 10]; // 年十位 a[3]=numbr[Dyear % 10]; // 年個位 a[6]=numbr[Dmonth / 10]; // 月十位 a[7]=numbr[Dmonth % 10]; // 月個位 a[10]=numbr[Dday / 10]; // 天十位 a[11]=numbr[Dday % 10]; // 天個位 } void Start_I2C() { SDA=1; //發送起始條件的數據信號 delay(1); SCL=1; delay(5); //起始條件建立時間大于4.7us,延時 SDA=0; //發送起始信號 delay(5); // 起始條件鎖定時間大于4μs SCL=0; //鉗住I2C總線,準備發送或接收數據 delay(2); } void Stop_I2C() { SDA=0; //發送結束條件的數據信號 delay(1); //發送結束條件的時鐘信號 SCL=1; //結束條件建立時間大于4us delay(5); SDA=1; //發送I2C總線結束信號 delay(4); } uchar write_byte(uchar addr, ucharwrite_data) { Start_I2C(); SendByte(ADDRTW); //////*******************************************************************/////////// if (ack == 0) return 0; SendByte(addr); if (ack == 0) return 0; SendByte(write_data); if (ack == 0) return 0; Stop_I2C(); delay1ms(10); return 1; } void ModifyTime(uchar yea,uchar mon,ucharda,uchar hou,uchar min,uchar sec) { uchar temp=0; temp=HEX2BCD(yea); write_byte(DS3231_YEAR,temp); //修改年 temp=HEX2BCD(mon); write_byte(DS3231_MONTH,temp); //修改月 temp=HEX2BCD(da); ///////////////////// write_byte(DS3231_DATE,temp); //修改日 temp=HEX2BCD(hou); write_byte(DS3231_HOUR,temp); //修改時 temp=HEX2BCD(min); write_byte(DS3231_MIN,temp); //修改分 temp=HEX2BCD(sec); write_byte(DS3231_SEC,temp); //修改秒 } void xianshi(void) { {uint i; TimeDisplay(hour,min,sec); lcd_pos(1,1); //時間 i=0; while(b[ i]!='\0') { delay1ms(1); write_dat(b); delay1ms(1); i++; }
DateDisplay(year,month,date); //顯示日期 delay1ms(1); lcd_pos(0,0); i=0; while(a[ i]!='\0') { delay1ms(1); write_dat(a); delay1ms(1); i++; } ReadTemperature(); //顯示溫度 delay1ms(1); lcd_pos(2,1); delay1ms(1); i=0; while(t[ i]!='\0') { delay1ms(1); write_dat(t); delay1ms(2); i++; } } }
void shuaxin(void) { uint i; TimeDisplay(hour,min,sec); lcd_pos(1,1); //時間 i=0; while(b[ i]!='\0') { delay1ms(1); write_dat(b); delay1ms(1); i++; }
DateDisplay(year,month,date); //顯示日期 delay1ms(1); lcd_pos(0,0); i=0; while(a[ i]!='\0') { delay1ms(1); write_dat(a); delay1ms(1); i++; } }
void sotp(void) { uinti;
while(1) { duan1: if(K1==0) { delay1ms(100); if(K1==0) { Readtime(); shuaxin(); i=0; gotoduan2;
} } else {gotoduan5;}
duan2: if(K2==0) { delay1ms(100); if(K2==0) { i++; if(i>6){i=0;} shuaxin(); } } switch(i) { case 0: if(K3==0){delay1ms(100);if(K3==0){year=year+1;} if(year>=100){year=0;} shuaxin();}break; case 1: if(K3==0){delay1ms(100);if(K3==0){month=month+1;} if(month>=13){month=0;} shuaxin();}break; case 2: if(K3==0){delay1ms(100);if(K3==0){date=date+1;} if(date>=32){date=0;} shuaxin();}break; case 3: if(K3==0){delay1ms(100);if(K3==0){hour=hour+1;} if(hour>=25){hour=0;} shuaxin();}break; case 4: if(K3==0){delay1ms(100);if(K3==0){min=min+1;} if(min>=61){min=0;} shuaxin();}break; case 5: if(K3==0){delay1ms(100);if(K3==0){sec=sec+1;} if(sec>=61){sec=0;} shuaxin();} break; }
duan4: if(K4==0) { delay1ms(100); if(K4==0) { ModifyTime(year,month,date,hour,min,sec); gotoduan5; ///////////////////*******************////////// } }
else {gotoduan2;}
duan5: Readtime(); xianshi(); }
}
main() /*主程序*/ {
lcd_init(); delay1ms(5); InitDS3231(); delay1ms(5); delay1ms(5); sotp(); } |