|
仿真原理圖如下(proteus仿真工程文件可到本帖附件中下載)
單片機(jī)源程序如下:
- //包含頭文件
- #include<REG51.h>
- #include<intrins.h>
- //宏定義
- #define LCM_Data P0 //將P0口定義為L(zhǎng)CM_Data
- #define uchar unsigned char
- #define uint unsigned int
- //1602的控制腳
- sbit lcd1602_rs=P2^5;//RS引腳為寄存器(1-數(shù)據(jù)寄存器,0-命令寄存器)
- sbit lcd1602_rw=P2^6;//RW為讀寫操作引腳(1-讀,0-寫)
- sbit lcd1602_en=P2^7;//使能信號(hào)
- sbit Scl=P3^4; //24c02串行時(shí)鐘
- sbit Sda=P3^5; //24c02串行數(shù)據(jù)
- sbit ALAM=P2^1; //報(bào)警,蜂鳴器
- sbit KEY=P3^6; //開鎖
- bit pass=0; //密碼正確標(biāo)志
- bit ReInputEn=0; //重置輸入允許標(biāo)志
- bit s3_keydown=0; //3秒按鍵標(biāo)志位
- bit key_disable=0; //鎖定鍵盤標(biāo)志
- unsigned char countt0,second; //t0中斷次數(shù),秒計(jì)數(shù)
- //void Delay5Ms(void); //聲明延時(shí)函數(shù)
- unsigned char code a[]={0xFE,0xFD,0xFB,0xF7};//控盤掃描控制表
- //液晶顯示數(shù)據(jù)數(shù)組
- unsigned char code start_line[] = {"password: "};
- unsigned char code name[] = {"===Coded Lock==="}; //顯示名稱
- unsigned char code Correct[] = {" correct "};//輸入正確
- unsigned char code Error[] = {" error "};//輸入錯(cuò)誤
- unsigned char code codepass[] = {" pass "};
- unsigned char code LockOpen[] = {" open "};//open
- unsigned char code SetNew[] = {"SetNewWordEnable"};
- unsigned char code Input[] = {"input: "};//input
- unsigned char code ResetOK[] = {"ResetPasswordOK "};
- unsigned char code initword[] = {"Init password..."};
- unsigned char code Er_try[] = {"error,try again!"};
- unsigned char code again[] = {"input again "};
- unsigned char InputData[6];//輸入密碼暫存區(qū)
- unsigned char CurrentPassword[6]={0,0,0,0,0,0};//讀取EEPROM密碼暫存數(shù)組
- unsigned char TempPassword[6];
- unsigned char N=0;//密碼輸入位數(shù)計(jì)數(shù)
- unsigned char ErrorCont;//錯(cuò)誤次數(shù)計(jì)數(shù)
- unsigned char CorrectCont;//正確輸入計(jì)數(shù)
- unsigned char ReInputCont;//重新輸入計(jì)數(shù)
- unsigned char code initpassword[6]={0,0,0,0,0,0};//輸入管理員密碼后將密碼初始為000000
- unsigned char code adminpassword[6]={1,3,1,4,2,0};//輸入管理員密碼后將密碼初始為000000
- #include<delay.h>
- //=========================5ms延時(shí)==========================
- void Delay5Ms(void)
- {
- unsigned int TempCyc = 5552;
- while(TempCyc--);
- }
- //===========================400ms延時(shí)============================
- void Delay400Ms(void)
- {
- unsigned char TempCycA = 5;
- unsigned int TempCycB;
- while(TempCycA--)
- {
- TempCycB=7269;
- while(TempCycB--);
- }
- }
- //==========================================================
- //===========================24c02===========================
- //==========================================================
- void mDelay(uint t)//延時(shí)
- {
- uchar i;
- while(t--)
- {
- for(i=0;i<125;i++)
- {;}
- }
- }
- void Nop(void)//空操作
- {
- _nop_(); //僅作延時(shí)用一條語(yǔ)句大約1us
- _nop_();
- _nop_();
- _nop_();
- }
- /*****24c02程序參照24c02時(shí)序圖
- /*起始條件*/
- void Start(void)
- {
- Sda=1;
- Scl=1;
- Nop();
- Sda=0;
- Nop();
- }
- //停止條件
- void Stop(void)
- {
- Sda=0;
- Scl=1;
- Nop();
- Sda=1;
- Nop();
- }
- //應(yīng)答位
- void Ack(void)
- {
- Sda=0;
- Nop();
- Scl=1;
- Nop();
- Scl=0;
- }
- //反向應(yīng)答位
- void NoAck(void)
- {
- Sda=1;
- Nop();
- Scl=1;
- Nop();
- Scl=0;
- }
- //發(fā)送數(shù)據(jù)子程序,Data為要求發(fā)送的數(shù)據(jù)
- void Send(uchar Data)
- {
- uchar BitCounter=8;
- uchar temp;
- do
- {
- temp=Data;//將待發(fā)送數(shù)據(jù)暫存temp
- Scl=0;
- Nop();
- if((temp&0x80)==0x80)//將讀到的數(shù)據(jù)&0x80
- Sda=1;
- else
- Sda=0;
- Scl=1;
- temp=Data<<1;//數(shù)據(jù)左移
- Data=temp;//數(shù)據(jù)左移后重新賦值Data
- BitCounter--;//該變量減到0時(shí),數(shù)據(jù)也就傳送完成了
- }
- while(BitCounter);//判斷是否傳送完成
- Scl=0;
- }
- //讀一字節(jié)的數(shù)據(jù),并返回該字節(jié)值
- uchar Read(void)
- {
- uchar temp=0;
- uchar temp1=0;
- uchar BitCounter=8;
- Sda=1;
- do
- {
- Scl=0;
- Nop();
- Scl=1;
- Nop();
- if(Sda) //數(shù)據(jù)位是否為1
- temp=temp|0x01; //為1 temp的最低位為1(|0x01,就是將最低位變?yōu)?)
- else //如果為0
- temp=temp&0xfe; //temp最低位為0(&0xfe(11111110)最地位就是0)
- if(BitCounter-1) //BitCounter減1后是否為真
- {
- temp1=temp<<1; //temp左移
- temp=temp1;
- }
- BitCounter--; //BitCounter減到0時(shí),數(shù)據(jù)就接收完了
- }while(BitCounter); //判斷是否接收完成
- return(temp);
- }
- void WrToROM(uchar Data[],uchar Address,uchar Num)
- {
- uchar i;
- uchar *PData;
- PData=Data;
- for(i=0;i<Num;i++)
- {
- Start();
- Send(0xa0);
- Ack();
- Send(Address+i);
- Ack();
- Send(*(PData+i));
- Ack();
- Stop();
- mDelay(20);
- }
- }
- void RdFromROM(uchar Data[],uchar Address,uchar Num)
- {
- uchar i;
- uchar *PData;
- PData=Data;
- for(i=0;i<Num;i++)
- {
- Start();
- Send(0xa0);
- Ack();
- Send(Address+i);
- Ack();
- Start();
- Send(0xa1);
- Ack();
- *(PData+i)=Read();
- Scl=0;
- NoAck();
- Stop();
- }
- }
- //=============================LCD1603==============================
- #define yi 0x80 //LCD第一行的初始位置,因?yàn)長(zhǎng)CD1602字符地址首位D7恒定為1
- #define er 0x80+0x40 //LCD第二行初始位置(因?yàn)榈诙械谝粋(gè)字符位置地址是0x40)
- //-----------------延時(shí)函數(shù),后面經(jīng)常調(diào)用-------------------------------
- void delay(uint xms)//延時(shí)函數(shù),有參函數(shù)
- {
- uint x,y;
- for(x=xms;x>0;x--)
- for(y=110;y>0;y--);
- }
- //===============寫指令======================
- void write_1602com(uchar com) //液晶寫入指令函數(shù)
- {
- lcd1602_rs=0; //數(shù)據(jù)/指令選擇置為指令
- lcd1602_rw=0; //讀寫選擇置為寫
- P0=com; //送入數(shù)據(jù)
- delay(1);
- lcd1602_en=1; //拉高使能端,為制造有效的下降沿做準(zhǔn)備
- delay(1);
- lcd1602_en=0; //en由高變低。產(chǎn)生下降沿,液晶執(zhí)行命令
- }
- //=============================寫數(shù)據(jù)==================================
- void write_1602dat(uchar dat) //液晶寫入數(shù)據(jù)函數(shù)
- {
- lcd1602_rs=1; //數(shù)據(jù)/指令選擇置為數(shù)據(jù)
- lcd1602_rw=0; //讀寫選擇置為寫
- P0=dat; //送入數(shù)據(jù)
- delay(1);
- lcd1602_en=1; //en置高電平,為制造下降沿做準(zhǔn)備
- delay(1);
- lcd1602_en=0; //en由高變低,產(chǎn)生下降沿,液晶執(zhí)行命令
- }
- //===================================初始化=========================================
- void lcd_init(void)
- {
- write_1602com(0x38); //設(shè)置液晶工作模式,意思:16*2行顯示,5*7點(diǎn)陣,8位數(shù)據(jù)
- write_1602com(0x0c); //開顯示不顯示光標(biāo)
- write_1602com(0x06); //整屏不移動(dòng),光標(biāo)自動(dòng)右移
- write_1602com(0x01); //清顯示
- }
- //========================================
- //=================================================================
- //===============將按鍵值編碼為數(shù)值============================
- unsigned char coding(unsigned char m)
- {
- unsigned char k;
- switch(m)
- {
- case(0x11):k=1;break;
- case(0x21):k=2;break;
- case(0x41):k=3;break;
- case(0x81):k='A';break;
- case(0x12):k=4;break;
- case(0x22):k=5;break;
- case(0x42):k=6;break;
- case(0x82):k='B';break;
- case(0x14):k=7;break;
- case(0x24):k=8;break;
- case(0x44):k=9;break;
- case(0x84):k='C';break;
- case(0x18):k='*';break;
- case(0x28):k=0;break;
- case(0x48):k='#';break;
- case(0x88):k='D';break;
- }
- return(k);
- }
- //====================按鍵檢測(cè)并返回按鍵值========================
- unsigned char keynum(void)
- {
- unsigned char row,col,i;
- P1=0xf0; //所有行線為0,列線為輸入
- if((P1&0xf0)!=0xf0) //一旦有鍵被按下,列線上的四位便不再全為1
- {
- Delay5Ms();
- Delay5Ms();
- if((P1&0xf0)!=0xf0) //如果有鍵被按下
- {
- col=P1^0xf0; //確定列線,任何一個(gè)數(shù)(0,1)與1異或,結(jié)果取反,與0異或,結(jié)果為其本身
- i=0;
- P1=a[i]; //精確定位
- while(i<4) //逐行掃描
- {
- if((P1&0xf0)!=0xf0)
- {
- row=~(P1&0xff); //確定行線
- break; //已定位后提前退出
- }
- else
- {
- i++;
- P1=a[i];
- }
- }
- }
- else
- {
- return 0;
- }
- while((P1&0xf0)!=0xf0);
- return(col|row); //行線與列線組合后返回
- }
- else return 0; //無(wú)鍵按下時(shí)返回0
- }
- //===================一聲提示音,表示有效輸入=============================
- void OneAlam(void)
- {
- ALAM=0;
- Delay5Ms();
- ALAM=1;
- }
- //====================兩聲提示音,表示操作成功===============================
- void TwoAlam(void)
- {
- ALAM=0;
- Delay5Ms();
- ALAM=1;
- Delay5Ms();
- ALAM=0;
- Delay5Ms();
- ALAM=1;
- }
- //====================三聲提示音,表示錯(cuò)誤=============================
- void ThreeAlam(void)
- {
- ALAM=0;
- Delay5Ms();
- ALAM=1;
- Delay5Ms();
- ALAM=0;
- Delay5Ms();
- ALAM=1;
- Delay5Ms();
- ALAM=0;
- Delay5Ms();
- ALAM=1;
- }
- //====================一直響,表示錯(cuò)誤=============================
- void Alam_KeyUnable(void){
- ALAM=0; //提示音一直響
- }
- //==================顯示提示輸入==============================
- void DisplayChar(void)
- {
- unsigned char i;
- if(pass==1)
- {
- write_1602com(er); //在二行開始顯示
- for(i=0;i<16;i++)
- {
- write_1602dat(LockOpen[i]); //顯示open,開鎖成功
- }
- }
- else
- {
- if(N==0) //輸入密碼位數(shù)
- {
- write_1602com(er);
- for(i=0;i<16;i++)
- {
- write_1602dat(Error[i]); //顯示錯(cuò)誤
- }
- }
- else
- {
- write_1602com(er);
- for(i=0;i<16;i++)
- {
- write_1602dat(start_line[i]); //顯示開始輸入
- }
- }
- }
- }
-
- //========================確認(rèn)鍵,并通過(guò)相應(yīng)標(biāo)志位執(zhí)行相應(yīng)功能
- void Ensure(void)
- {
- unsigned char i,j;
- RdFromROM(CurrentPassword,0,6); //從24c02里讀出存儲(chǔ)密碼
- if(N==6)
- {
- if(ReInputEn==0) //重置密碼功能未開啟
- {
- if((CurrentPassword[0]==InputData[0])&&(CurrentPassword[1]==InputData[1])&&(CurrentPassword[2]==InputData[2])&&(CurrentPassword[3]==InputData[3])&&(CurrentPassword[4]==InputData[4]))
- {
- ErrorCont=0; //只要密碼正確了就將錯(cuò)誤次數(shù)清零
- CorrectCont++; //輸入正確變量++
- if(CorrectCont==1)
- {
- write_1602com(er);
- for(j=0;j<16;j++)
- {
- write_1602dat(LockOpen[j]); //顯示open開鎖畫面
- }
- TwoAlam(); //操作成功提示音
- KEY=0; //開鎖
- pass=1; //密碼正確標(biāo)志位置1
- for(j=0;j<6;j++) //將輸入清除
- {
- InputData[i]=0; //開鎖后將輸入位清零
- }
- }
- else //當(dāng)兩次輸入正確時(shí),開啟重置密碼功能
- {
- write_1602com(er);
- for(j=0;j<16;j++)
- {
- write_1602dat(SetNew[j]); //顯示重置密碼界面
- }
- TwoAlam(); //操作成功提示
- ReInputEn=1; //允許重置密碼輸入
- CorrectCont=0; //正確計(jì)數(shù)器清零
- }
- }
- else if((InputData[0]==adminpassword[0])&&(InputData[1]==adminpassword[1])&&(InputData[2]==adminpassword[2])&&(InputData[3]==adminpassword[3])&&(InputData[4]==adminpassword[4]))
- {
- WrToROM(initpassword,0,6); //強(qiáng)制將初始密碼寫入24c02存儲(chǔ)
- write_1602com(er);
- for(j=0;j<16;j++)
- {
- write_1602dat(initword[j]); //顯示初始化密碼
- }
- TwoAlam(); //成功提示音
- Delay400Ms(); //延時(shí)400ms
- N=0;
- }
- else //密碼錯(cuò)誤時(shí)
- {
- CorrectCont=0; //正確計(jì)數(shù)器清零,密碼一旦錯(cuò)誤,對(duì)密碼正確次數(shù)重新清零
- ErrorCont++; //錯(cuò)誤次數(shù)++
- write_1602com(er);
- for(j=0;j<16;j++)
- {
- write_1602dat(Error[j]); //顯示錯(cuò)誤信息
- }
- pass=0;
- TR0=1; //開啟定時(shí)
- key_disable=1; //鎖定鍵盤
- KEY=1; //關(guān)閉鎖
- if(ErrorCont==3) //錯(cuò)誤輸入連續(xù)達(dá)3次時(shí),報(bào)警并鎖定鍵盤
- {
- write_1602com(er);
- for(i=0;i<16;i++)
- {
- write_1602dat(Error[i]);
- }
- do
- Alam_KeyUnable();
- while(1); //死循環(huán),超過(guò)三次密碼錯(cuò)誤則一直報(bào)警并鎖定鍵盤,直到重新運(yùn)行或按下復(fù)位鍵
- }
- }
- }
- else //當(dāng)已經(jīng)開啟重置密碼功能時(shí),而按下確認(rèn)鍵
- {
- write_1602com(er);
- for(j=0;j<16;j++)
- {
- write_1602dat(Er_try[j]); //錯(cuò)誤,請(qǐng)重新輸入
- }
- ThreeAlam(); //錯(cuò)誤提示音
- }
- }
- else //密碼沒有輸入到6位時(shí),按下確認(rèn)鍵
- {
- write_1602com(er);
- for(j=0;j<16;j++)
- {
- write_1602dat(Error[j]); //顯示錯(cuò)誤
- }
- ThreeAlam(); //錯(cuò)誤提示音
- pass=0;
- }
- N=0; //將輸入數(shù)據(jù)計(jì)數(shù)器清零,為下一次輸入做準(zhǔn)備
- }
- //===========================重置密碼===================================================================================
- void ResetPassword(void)
- {
- unsigned char i;
- unsigned char j;
- if(pass==0) //沒開鎖時(shí)
- {
- pass=0;
- DisplayChar(); //顯示開始輸入password
- ThreeAlam(); //沒開鎖時(shí)按下重置密碼報(bào)警3聲
- }
- else //開鎖狀態(tài)下才能進(jìn)行密碼重置
- {
- if(ReInputEn==1) //開鎖狀態(tài)下,ReInputEn置1,重置密碼允許
- {
- if(N==6) //輸入6位密碼
- {
- ReInputCont++; //重置密碼次數(shù)計(jì)數(shù)
- if(ReInputCont==1) //輸入一次密碼時(shí)
- {
- OneAlam();
- for(i=0;i<6;i++)
- {
- TempPassword[i]=InputData[i]; //將第一次輸入的數(shù)據(jù)暫存起來(lái)
- }
- write_1602com(er);
- for(j=0;j<16;j++)
- {
- write_1602dat(again[j]); //顯示再輸入一次
- }
- }
- if(ReInputCont==2) //輸入兩次密碼時(shí)
- {
- if((TempPassword[0]==InputData[0])&&(TempPassword[1]==InputData[1])&&(TempPassword[2]==InputData[2])&&(TempPassword[3]==InputData[3])&&(TempPassword[4]==InputData[4]))
- {
- write_1602com(er);
- for(j=0;j<16;j++)
- {
- write_1602dat(ResetOK[j]); //密碼修改成功,顯示
- }
- TwoAlam(); //操作成功顯示
- WrToROM(TempPassword,0,6); //將新密碼寫入24c02存儲(chǔ)
- ReInputEn=0; //關(guān)閉重置功能
- }
- else //如果兩次的密碼不同
- {
- write_1602com(er);
- for(j=0;j<16;j++)
- {
- write_1602dat(Error[j]); //顯示錯(cuò)誤Error
- }
- ThreeAlam(); //錯(cuò)誤提示
- pass=0; //關(guān)鎖
- ReInputEn=0; //關(guān)閉重置功能
- ReInputCont=0; //重置密碼次數(shù)清零
- KEY=1; //關(guān)閉鎖
- DisplayChar();
- }
- ReInputCont=0;
- CorrectCont=0;
- }
- N=0; //輸入數(shù)據(jù)位數(shù)計(jì)數(shù)器清零
- }
- else //密碼沒有輸入到6位時(shí),按下重置鍵時(shí)
- {
- write_1602com(er);
- for(j=0;j<16;j++)
- {
- write_1602dat(Error[j]); //顯示錯(cuò)誤
- }
- ThreeAlam(); //錯(cuò)誤提示音
- N=0;
- }
- }
- }
- }
- //===========================輸入密碼錯(cuò)誤超過(guò)三次,報(bào)警并鎖死鍵盤=====
- //========================取消所有操作===
- void Cancel(void)
- {
- unsigned char i;
- unsigned char j;
- //DisplayListChar(0,1,start_line);
- write_1602com(er);
- for(j=0;j<16;j++)
- {
- write_1602dat(start_line[j]); //顯示開機(jī)輸入密碼界面
- }
- TwoAlam(); //提示音
- for(i=0;i<6;i++)
- {
- InputData[i]=0; //將輸入密碼清零
- }
- KEY=1; //關(guān)閉鎖
- ALAM=1; //報(bào)警關(guān)
- pass=0; //密碼正確標(biāo)志清零
- ReInputEn=0; //重置輸入允許標(biāo)志清零
- ErrorCont=0; //密碼錯(cuò)誤輸入次數(shù)清零
- CorrectCont=0; //密碼正確輸入次數(shù)清零
- ReInputCont=0; //重置密碼輸入次數(shù)清理
- s3_keydown=0;
- key_disable=0; //鎖定鍵盤標(biāo)志清零
- N=0; //輸入位數(shù)計(jì)數(shù)器清零
- }
- //=========================主函數(shù)==
- void main(void)
- {
- unsigned char KEY,NUM;
- unsigned char i,j;
- P1=0xFF; //P1口復(fù)位
- TMOD=0x01; //定義工作方式
- TL0=0xB0;
- TH0=0x3C; //定時(shí)器賦初值,定時(shí)50ms
- EA=1; //打開中斷總開關(guān)
- ET0=1; //打開中斷允許開關(guān)
- TR0=1; //打開定時(shí)器開關(guān)
- Delay400Ms(); //啟動(dòng)等待,等LCM講入工作狀態(tài)
- lcd_init(); //LCD初始化
- write_1602com(yi); //日歷顯示固定符號(hào)從第一行第0個(gè)位置之后開始顯示
- for(i=0;i<16;i++)
- {
- write_1602dat(name[i]); //向液晶屏寫開機(jī)畫面
- }
- write_1602com(er);
- for(i=0;i<16;i++)
- {
- write_1602dat(start_line[i]); //寫輸入密碼等待界面
- }
- write_1602com(er+9); //設(shè)置光標(biāo)位置
- write_1602com(0x0f); //設(shè)置光標(biāo)為閃爍
- Delay5Ms(); //延時(shí)片刻(可不要)
- N=0; //初始化數(shù)據(jù)輸入位數(shù)
- while(1) //進(jìn)入循環(huán)
- {
- if(key_disable==1) //鎖定鍵盤標(biāo)志為1時(shí)
- Alam_KeyUnable(); //報(bào)警鍵盤鎖
- else
- ALAM=1; //關(guān)報(bào)警
- KEY=keynum(); //讀按鍵的位置碼
- if(KEY!=0) //當(dāng)有按鍵按下時(shí)
- {
- if(key_disable==1) //鎖定鍵盤標(biāo)志為1時(shí)
- {
- second=0; //秒清零
- }
- else //沒有鎖定鍵盤時(shí)
- {
- NUM=coding(KEY); //根據(jù)按鍵的位置將其編碼,編碼值賦值給NUM
- {
- switch(NUM) //判斷按鍵值
- {
- case ('A'): ; break;
- case ('B'): ; break;
- case ('C'): ; break; //ABC是無(wú)意義按鍵
- case ('D'):ResetPassword(); break; //重新設(shè)置密碼
- case ('*'):Cancel(); break; //取消當(dāng)前輸入
- case ('#'):Ensure(); break; //確認(rèn)鍵
- default: //如果不是功能鍵按下時(shí),就是數(shù)字鍵按下
- {
- if(N<6) //當(dāng)輸入密碼少于6位時(shí),接受輸入并保存,大于6位時(shí)則無(wú)效
- {
- write_1602com(er);
- for(i=0;i<16;i++)
- {
- write_1602dat(Input[i]); //顯示輸入畫面
- }
- OneAlam(); //按鍵提示音
- for(j=0;j<=N;j++)
- {
- write_1602com(er+6+j); //顯示位數(shù)隨輸入增加而增加
- write_1602dat('*'); //但不顯示實(shí)際數(shù)字,用*代替
- }
- InputData[N]=NUM; //將數(shù)字鍵的碼賦值給InputData【】數(shù)組暫存
- N++; //密碼位數(shù)加
- }
- else //輸入數(shù)據(jù)位數(shù)大于6后,忽略輸入
- {
- N=6; //密碼輸入大于6位時(shí),不接受輸入
- break;
- }
- }
- }
- }
- }
- }
- }
- }
- //=====中斷服務(wù)函數(shù)======
- void time0_int(void) interrupt 1 //定時(shí)器T0
- {
- TL0=0xB0;
- TH0=0x3C; //定時(shí)器重新賦初值
- countt0++; //計(jì)時(shí)器變量加,加1次時(shí)50ms
-
-
復(fù)制代碼
Keil代碼與Proteus仿真和word下載:
仿真程序文檔.7z
(1.33 MB, 下載次數(shù): 107)
2023-6-10 00:00 上傳
點(diǎn)擊文件名下載附件
相關(guān)文件
|
-
電子密碼鎖仿真
評(píng)分
-
查看全部評(píng)分
|