找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 3867|回复: 4
收起左侧

STC89C52单片机指纹识别程序 1602显示

[复制链接]
ID:487236 发表于 2019-8-7 21:29 | 显示全部楼层 |阅读模式
实物照片.jpg IMG_4527.JPG IMG_4526.JPG IMG_4525.JPG IMG_4524.JPG
  1. #include <reg52.h>
  2. #include <intrins.h>
  3. #include"lcd1602.h"


  4. sbit buzzer=P2^0;  //蜂鸣器        低电平打开蜂鸣器

  5. sbit red=   P1^0;  //红色灯 表示录入模式
  6. sbit green= P1^1;  //绿色灯 表示识别模式

  7. sbit k2=P3^4; //录入一次指纹
  8. sbit k1=P3^3; //模式识别转换
  9. sbit k3=P3^2; //清除所有指纹(10个指纹清除)

  10. #define Max_User 10
  11. #define error 2

  12. unsigned char         SaveNumber=0,searchnum=0;
  13. unsigned int          SearchNumber=0;               
  14. unsigned int         clk0=0;

  15. unsigned char str[3]={0,'\0','\0'};

  16. unsigned char code Identify[16]="Lock:   Idenfity";
  17. unsigned char code Input[16]   ="Lock:   Input   ";

  18. bit modeflag=0,clearallflag=0,changeflag=0;

  19. // modeflag=0 为识别模式 modeflag=1为录入指纹模式
  20. //每成功录入一次指纹 SaveNumber++

  21. //////////////////////////////////////常用指令定义/////////////////////////////

  22. //Verify  Password   :验证设备握手口令
  23. unsigned char code VPWD[16]={16,0X01 ,0Xff,0xff,0xff,0xff, 0x01,0,7,0x13,0x00,0x00,0x00,0x00,0x00,0x1b};         //回送12个

  24. //设置设备握手口令
  25. unsigned char code STWD[16]={16,0X01 ,0Xff,0xff,0xff,0xff, 0x01,0,7,0x12,0x00,0x00,0x00,0x00,0x00,0x1a};         //回送12个

  26. //GetImage           :探测手指并从传感器上读入图像
  27. unsigned char code GIMG[14]={12, 0X01 ,0Xff,0xff,0xff,0xff, 0x01, 0,3,1,0x00,0x05};        //回送12个

  28. //Gen Templet1        :根据原始图像生成指纹特征1
  29. unsigned char code GENT1[14]={13,0X01 ,0Xff,0xff,0xff,0xff,0x01,0,4,2,1,0x00,0x08};        //回送12个

  30. //Gen Templet2        :根据原始图像生成指纹特征2
  31. unsigned char code GENT2[14]={13,0X01 ,0Xff,0xff,0xff,0xff,0x01,0,4,2,2,0x00,0x09}; //回送12个        

  32.         
  33. //Search Finger      :以CharBufferA或CharBufferB中的特征文件搜索整个或部分指纹库
  34. unsigned char code SEAT[18]={17,  0X01 ,0Xff,0xff,0xff,0xff, 0x01,   0,8, 4,1,0,0,    0,0x65,  0x00,0x73};        //回送16个


  35. //Merge Templet      ;将CharBufferA与CharBufferB中的特征文件合并生成模板,结果存于ModelBuffer。
  36. unsigned char code MERG[14]={12,  0X01 ,0Xff,0xff,0xff,0xff, 0x01,  0,3,5 , 0x00,0x09};//回送12个        

  37. //Store Templet      :将ModelBuffer中的文件储存到flash指纹库中
  38. unsigned char code STOR[16]={15,  0X01 ,0Xff,0xff,0xff,0xff, 0x01,  0,6,6,2,     0x00,0x00,     0x00,0x0f}; //回送12个

  39. //Read Note
  40. unsigned char code RDNT[14]={13,0X01 ,0Xff,0xff,0xff,0xff, 0x01, 0,4,0x19,  0, 0x00,0x1e};

  41. //Clear Note
  42. unsigned char code DENT[46]={45,0X01 ,0Xff,0xff,0xff,0xff, 0x01, 0,36,0x18,0,0,0,
  43. 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0x00,0x3d};

  44. //DEL one templet
  45. unsigned char code DELE_one[16]={16, 0X01 ,0Xff,0xff,0xff,0xff, 0x01,   0,7,  0x0c,    0x00,0x00, 0,1, 0x00,0x15};

  46. //DEL templet      ;清空指纹库
  47. unsigned char code DELE_all[12]={12,0X01 ,0Xff,0xff,0xff,0xff, 0x01, 0,3,  0x0d,0x00,0x11};

  48. //////////////////////////////////////常用指令定义-------结束///////////////////////////////////////////////////////////////////////////////////////


  49. //////////////////////////////////////宏定义////////////////////////////////////////////////////////////////////////
  50. #define FALSE 0
  51. #define TURE  1

  52. //状态定义表
  53. #define on            1
  54. #define off           0

  55. #define MAX_NUMBER    63
  56. #define _Nop()  _nop_()


  57. //////////////////////////////////////宏定义------------结束////////////////////////////////////////////////////////

  58. unsigned char                   FifoNumber=0;
  59. xdata unsigned char   FIFO[MAX_NUMBER+1]={0};

  60. /*********1毫秒延时程序**********/
  61. void delay1ms(unsigned int t)
  62. {
  63.         unsigned int i,j;
  64.         for(i=0;i<t;i++)
  65.            for(j=0;j<120;j++)
  66.            ;
  67. }


  68. void TxdByte(unsigned char dat)//串口发送信息,通过查询方式发送一个字符
  69. {
  70.     TI = 0;                 //让TI=0
  71.     SBUF = dat;         //读入数据
  72.     while(!TI);         //等待发送完毕
  73.     TI = 0;                 //清零
  74. }

  75. bit Command(unsigned char *p,unsigned char MaxTime) //命令解析,给模块发送一个命令
  76. {
  77.           unsigned char count=0,tmpdat=0,temp=0,i=0,package=0,flag=0,checksum=0;
  78.                   
  79.         bit result=0, start=0,stop=0;

  80.           TxdByte(0xef);//数据包包头识别码
  81.            TxdByte(0x01);//数据包包头识别码
  82.           i=*p;         //数组的第“0”个元素、里面存放了本数组的长度,把这个长度给变量i,方便进行操作
  83.           p++;
  84.         p++;
  85.           for (count=i-1; count!=1;count--)  //Sent command String
  86.                   {
  87.                          temp=*p++;//取第个“1”个元素的内容,然后发送
  88.                     TxdByte(temp);//将数据发送出去
  89.                 }  
  90.           result=TURE;//发送完成,结果为真 (真为1)           
  91.           FifoNumber=0;
  92.           for (count=MAX_NUMBER+1; count!=0; count--)//清空所有FIFO[]数组里面的内容,写入0X00
  93.                  FIFO[count-1]=0x00;  
  94.         
  95.           if (result)   
  96.            {               
  97.              result=FALSE;
  98.               start =FALSE;
  99.                    stop  =FALSE;
  100.                count=0;
  101.                clk0=0;        //清零CL0计数
  102.                
  103.                do /////////////////////////////do的内容////////////////////////////////
  104.                 {        
  105.                         restart0:                                
  106.                          if (RI==1)//如果接收到数据
  107.                               {                                 
  108.                                           tmpdat=SBUF;//先把接收到的数据放到tmpdat中
  109.                             RI=0;
  110.                             if ((tmpdat==0xef)&&(start==FALSE))//这个数据为第一个传回来的数据,也就是“指令应答”的第一个字节
  111.                                     {
  112.                                                         count=0;
  113.                                                     FIFO[0]=tmpdat;//读入第一个应答字节(0XEF),存在第“0”个元素中   
  114.                                                         flag=1;        
  115.                                                         goto
  116.                                                                 restart0;//可以用中断方式进行                        
  117.                                                 
  118.                                      }
  119.                                         if(flag==1)//第一个字节已经回来,所以flag==1成立
  120.                                                 {  
  121.                                                         if(tmpdat!=0x01)  //接收数据错误,将重新从缓冲区接收数据
  122.                                                         {         
  123.                                                                 flag=0;//接收应答失败
  124.                                                                 result=FALSE;
  125.                                                               start =FALSE;
  126.                                                                    stop=FALSE;
  127.                                                                count=0;
  128.                                                                 goto
  129.                                                                         restart0;                                       
  130.                                                         
  131.                                                         }
  132.                                                         //如果成功接收到0xef01,可以开始接收数据
  133.                                                         flag=2;//flag=2;表示应答成功,可以开始接收数据了
  134.                                                         count++;//现在count=1;
  135.                                                         FIFO[count]=tmpdat;//读入第二个应答字节(0X01),存在第“1”个元素中   
  136.                                                         start=TURE;        //应答成功可以开始接收数据
  137.                                                             goto
  138.                                                                         restart0;        
  139.                                                 }                  
  140.                              if((flag==2)&&(start==TURE))//flag=2;表示应答成功,可以开始接收数据了
  141.                                      {                                                              
  142.                                                            count++; //数据元素下标++
  143.                                             FIFO[count]=tmpdat;//存入数据
  144.                                                         if(count>=6)
  145.                                                                 {
  146.                                                                         checksum=FIFO[count]+checksum; //计算校验和
  147.                                                                 }
  148.                                                         if(count==8)
  149.                                                                 {
  150.                                                                         package=FIFO[7]*0X100+FIFO[8];        //计算包长度                                                        
  151.                                                                         stop= TURE;
  152.                                                                 }
  153.                                                         if(stop)
  154.                                                                 {                                                
  155.                                                                         if(count==package+8)
  156.                                                                         {
  157.                                                                                 checksum=checksum-FIFO[count-1];
  158.                                                                                 if(checksum!=FIFO[count]&0xff)
  159.                                                                                         result=FALSE; //校验失败,置结果标志为0                                                        
  160.                                                                                 else
  161.                                                                                         result=TURE;
  162.                                                                                 flag=0;
  163.                                                                                 break;
  164.                                                                         }
  165.                                                                 }
  166.                                      }
  167.                                 }
  168.           }/////////////////////////////do的内容----------------结束////////////////////////////////
  169.                   
  170.                 while ((clk0<=MaxTime)&&(count<=MAX_NUMBER)); //由定时器以及最大接收数据来控制,保证不会在此一直循环
  171.                   
  172.         FifoNumber=count;        //保存接收到的数据个数
  173.          }
  174.    return (result);
  175. }

  176. bit VefPSW(void)//验证设备握手口令,成功返回1     
  177. {
  178.          unsigned char  count=0;
  179.         while (1)
  180.            {
  181.              if(Command(VPWD,20) && (FifoNumber==11) && (FIFO[9]==0x00))  
  182.                         return(1);
  183.              count++;
  184.                      if (count>=2)//如果不成功,再验证一次,如果两次不成功,返回失败
  185.                 {  
  186.                     return(0);   
  187.                       }
  188.         }
  189. }


  190. void Clear_All(void) //清空所有指纹信息,清空前首先验证和指纹模块通讯是否正常  
  191. {
  192.         if(VefPSW()!=1)//验证失败
  193.         {                        
  194.                  
  195.         }
  196.         else                  //验证成功
  197.         {                        
  198.                    Command(DELE_all,50); //清空指纹库
  199.                    Command(DENT,20);           //清空用户记事本
  200.         }
  201. }

  202. unsigned char ImgProcess(unsigned char BUFID)  //发获取图像并生成特征文件,存入BUFID中//输入参数为缓冲区号  
  203. {        
  204.     if(Command(GIMG,89) && (FifoNumber==11) && (FIFO[9]==0x00))  
  205.     {
  206.                 if(BUFID==1)      
  207.                 {                                   
  208.                       if(Command(GENT1,60) && (FifoNumber==11) && (FIFO[9]==0x00))  
  209.                 {                                                        
  210.                                 return 1;
  211.                         }  
  212.                 else
  213.                 {
  214.                                   return 0;
  215.                  }
  216.                  }
  217.                 else if(BUFID==2)
  218.                 {
  219.                           if(Command(GENT2,60) && (FifoNumber==11) && (FIFO[9]==0x00))  
  220.                 {                                
  221.                                 return 1;
  222.                         }  
  223.                 else
  224.                 {
  225.                                   return 0;
  226.                  }                        
  227.                 }
  228.     }
  229.     else
  230.     {
  231.       return 0;      
  232.     }
  233.         return 0;            
  234. }

  235. bit Searchfinger(void)//搜索指纹(发送搜索命令、以及根据返回值确定是否存在)        
  236. {               
  237.            if(Command(SEAT,60) && (FifoNumber==15) && (FIFO[9]==0x00) )  
  238.     {
  239.                 SearchNumber=FIFO[10]*0x100+FIFO[11];//搜索到的页码
  240.                 //MatchScore=FIFO[12]*0x100+FIFO[13]   可以在此计算得分,从而进行安全级别设定,本程序忽略
  241.                    return 1;
  242.         }     
  243.     else            
  244.     {
  245.                return 0;
  246.     }  
  247. }


  248. unsigned char search(void)//搜索用户
  249. {
  250.          unsigned char SearchBuf=0,i=0;
  251.           while (i<20)
  252.     {
  253.            if(modeflag==0)//在识别模式下
  254.            {
  255.                     if (ImgProcess(1)==1)//首先读入一次指纹  
  256.                 {
  257.                        SearchBuf=Searchfinger();//进行指纹比对,如果搜索到,返回搜索到的指纹序号
  258.                        if(SearchBuf==1)         //成功搜索到指纹号
  259.                        {
  260.                                return SearchNumber;
  261.                        }
  262.                        else  //如果没有搜索到指纹号
  263.                         {
  264.                            return 255; //由于只能录入162个指纹,这时候如果返回255表示识别错误,或者这个指纹没有登记
  265.                         }     
  266.                }
  267.           }
  268.           else
  269.            {
  270.                    break;
  271.                    return 0;
  272.                 }
  273.                 i++;        
  274.     }
  275.    return 255;//这里其实是超时了,由于只能录入162个指纹,这时候如果返回255表示识别错误,或者这个指纹没有登记
  276. }

  277. bit savefingure(unsigned char ID)//保存指纹
  278. {
  279.          unsigned char i=0;
  280.          //现在开始进行存储指纹模板的操作
  281.      for (i=0;i<16;i++)        //保存指纹信息
  282.                  {
  283.                         FIFO[i]=STOR[i];
  284.                 }  
  285.      FIFO[12]=ID; //把指纹模板存放的PAGE_ID也就是FLASH的位置
  286.      FIFO[14]=FIFO[14]+ID;        //校验和
  287.      if (Command(FIFO,70)==1)//不成功返回0        //此处进行存放指纹模板的命令
  288.                  {return(1);}  //成功返回1
  289.      else
  290.                  {return(0);} //不成功返回0
  291. }



  292. unsigned char enroll(void) //采集两次指纹,生成1个 指纹模板
  293. {
  294.          unsigned char temp=0,count=0;
  295.           while(1)
  296.           {
  297.                    temp=ImgProcess(1); //生成特征1   
  298.                    if (temp==1)//生成特征文件成功            
  299.               {                        
  300.                 break;
  301.                }
  302.              else                  
  303.                {
  304.                 if (temp==0)//采集指纹没有成功
  305.                   {
  306.                                 count++;
  307.                     if (count>=40)//如果采集了40次,还不成功,直接采集失败,直接退出enroll函数----返回0  
  308.                                         return(0);
  309.             }
  310.         }
  311.            }
  312.         //采集第一个特征成功

  313.         //开始采集第二个特征
  314.          while(1)
  315.           {
  316.                    temp=ImgProcess(2); //生成特征2   
  317.                    if (temp==1)//生成特征文件2成功
  318.               {
  319.                 if (  (Command(MERG,40)&& (FifoNumber==11) && (FIFO[9]==0x00))==0  ) //合并不成功返回0,成功返回1
  320.                         {
  321.                                 return 0;        //合并不成功
  322.                         }        
  323.                         else//特征文件合并生成模板,结果存于ModelBuffer
  324.                         {
  325.                                  buzzer=0;
  326.                      delay1ms(100);
  327.                      buzzer=1;
  328.                                  delay1ms(100);
  329.                                  buzzer=0;
  330.                      delay1ms(100);        //响两声,表示生成一个模板成功
  331.                      buzzer=1;
  332.                      return 1;        //合并成功
  333.                         }        
  334.         }
  335.               else      
  336.                {        
  337.                                 count++;                                 
  338.                                 if (count>=25)         return(0);                       
  339.              }
  340.            }
  341. }


  342. void shownum(unsigned char dat)        //显示指纹编号
  343. {
  344.         unsigned char numdat=0;
  345.         numdat=dat;
  346.         GotoXY(0,1);//x,y
  347.         Print("Finger:");
  348.         GotoXY(0,8);//x,y
  349.         str[0]=0x30+numdat/10;
  350.         str[1]=0x30+numdat%10;
  351.         Print(str);
  352. }

  353. void changeState(void)//显示状态转换
  354. {
  355.         if(changeflag==1)        //模式识别键按下
  356.         {
  357.            GotoXY(0,0);//x,y
  358.            if(modeflag==0)
  359.            {
  360.                            Print(Identify);//显示 识别模式  Indentity
  361.            }
  362.            else
  363.            {
  364.                            Print(Input);   //显示  输入模式 Input
  365.            }
  366.            changeflag=0;               
  367.         }
  368.                
  369. }




  370. void main(void)//主函数
  371. {
  372.         unsigned char i=0;

  373.         ET0=1;     //定时器0开中断
  374.         TL0=0x97;  //17ms的初值
  375.         TH0=0xBD;
  376.    delay1ms(5);
  377.    LcdRw=0;      //只对液晶进行写操作,不进行读操作
  378.    delay1ms(5);
  379.    LCD_Initial();//液晶初始化
  380.    delay1ms(5);
  381.    GotoXY(0,0);//x,y
  382.    Print(Identify);

  383.     //串口初始化
  384.     SCON=0x50;   //UART方式1:8位UART;   REN=1:允许接收
  385.     PCON=0x00;   //SMOD=0:波特率不加倍
  386.     TMOD=0x21;   //T1方式2,用于UART波特率
  387.     TH1=0xFD;
  388.     TL1=0xFD;   //UART波特率设置:9600
  389.     TR1=1;
  390.     TR0=1;// 开定时器0
  391.         
  392.         IT0=0;//中断0低电平中断
  393.         IT1=1;//中断1低电平中断
  394.         EX0=1;//开中断0
  395.         EX1=1;//开中断1
  396.     EA=1;

  397.         

  398.         for(i=0;i<6;i++)//开始握手6次,如果没有一次成功,表示模块通信不正常。只要成功就跳出此循环
  399.         {
  400.                 if(VefPSW())//与模块握手通过,绿灯亮起。进入识别模式
  401.                   {
  402.                                 green=0; //读一次
  403.                                 buzzer=0;
  404.                                 shownum(0);
  405.                         delay1ms(300);
  406.                         buzzer=1;
  407.                                 break;        
  408.                   }
  409.             else   //通信握手失败
  410.                  {
  411.                         red=1;           //红灯亮
  412.                         green=1;
  413.                         break;

  414.                  }
  415.         }

  416.     while(1)
  417.         {
  418.                
  419.                 if(k2==0)//录入一个指纹
  420.                         {
  421.                                   delay1ms(10);        
  422.                                   if(k2==0)//如果仍为低电平,表示按键有效
  423.                                           {
  424.                                             while(k2==0);//等待松手
  425.                                           if(VefPSW()==1&&modeflag==1&&SaveNumber<10)//与模块握手通过
  426.                                                   {
  427.                                                             if(enroll()==1)//采集两次,生成1个指纹模板成功
  428.                                                                   {
  429.                                                                    if(savefingure(SaveNumber+1)==1)//保存也成功
  430.                                                                            {
  431.                                                                                 SaveNumber++;//加一次
  432.                                                                                 shownum(SaveNumber);
  433.                                                                         }
  434.                         
  435.                                                                 }
  436.                                                   }
  437.                                                 else
  438.                                                 {
  439.                                                                 buzzer=0;
  440.                                                                 for(i=0;i<8;i++)
  441.                                                                 {
  442.                                                                   delay1ms(100);
  443.                                                                   red=~red;
  444.                                                                 }
  445.                                                                 red=0;
  446.                                                                 buzzer=1;               
  447.                                                 }         
  448.                                           }
  449.                                        
  450.                         }        //k2==0

  451.                 if(modeflag==0)//为识别模式
  452.                 {
  453.                           searchnum=search();
  454.                           if(searchnum>=1&&searchnum<=162)//最多只能存储162个指纹
  455.                           {
  456.                                       //显示搜索到的指纹号
  457.                                         shownum(searchnum);
  458.                                        
  459.                                //蜂鸣器响一声
  460.                                buzzer=0;
  461.                                delay1ms(100);
  462.                                buzzer=1;
  463.                           }
  464.                           if(searchnum==255)//表示放上了没有登记的指纹或者识别错误
  465.                           {
  466.                             ;//用户可以自己处理,比如显示 指纹是非法的 或者蜂鸣器响三声,表示错误等
  467.                           }
  468.                 }

  469.                 if(clearallflag==1)
  470.                 {
  471.                         clearallflag=0;
  472.                         Clear_All();
  473.                     red=0;             //红色灯亮
  474.                     green=1;
  475.                         //蜂鸣器长响一次,表示清除所有指纹结束
  476.                         modeflag=1;//进入录入指纹模式
  477.                         GotoXY(0,0);//x,y
  478.                         Print(Input);
  479.                         shownum(0);
  480.                         buzzer=0;
  481.                         delay1ms(800);
  482.                         buzzer=1;
  483.                         SaveNumber=0;
  484.                 }

  485.         }////////////////////////////while(1)结束////////////////////////////////
  486.         
  487. }


  488. void int0(void) interrupt 0//中断0,清除所有指纹
  489. {
  490.         if(k3==0) //清除所有指纹
  491.                 {
  492.                         delay1ms(10);
  493.                         if(k3==0)//如果仍为低电平,表示按键有效
  494.                         {
  495.                                 while(k3==0);//等待松手        
  496.                                 clearallflag=1;
  497.                         }        
  498.                 }        
  499. }

  500. void  Timer0(void) interrupt 1//定时器0中断函数
  501. {
  502.          TL0=0x97;
  503.         TH0=0xBD;
  504.         clk0++;   //延时17ms
  505. }


  506. void int1(void) interrupt 2//外部中断1,模式转换键按下
  507. {
  508.         if(k1==0)//模式转换 其中用modeflag 来标志,默认从第1个指纹开始录入
  509.         {
  510.           delay1ms(10);        
  511.           if(k1==0)//如果仍为低电平,表示按键有效
  512.                   {
  513.                         while(k1==0);//等待松手
  514.                         changeflag=1;//表示模式有变化,即K1键按下
  515.                          modeflag=~modeflag;//0表示录入指纹  1表示识别指纹
  516.                          changeState();//看一下是不是要显示状态转换
  517.                          if(modeflag==0)//根据模式改变当前模式指示灯的颜色
  518.                          {
  519.                                 green=0;                red=1;         //录入指纹绿灯亮
  520.                          }
  521.                          else
  522.                          {
  523.                                 red=0;                 green=1;  //指纹识别红灯灯亮
  524.                          }
  525.                 }        
  526.         }        
  527. }
复制代码

1602 指纹识别 程序.rar

54.92 KB, 下载次数: 92, 下载积分: 黑币 -5

回复

使用道具 举报

ID:58591 发表于 2019-11-15 15:58 | 显示全部楼层
感谢分享
回复

使用道具 举报

ID:243394 发表于 2019-11-16 12:48 | 显示全部楼层
看起来很NB
回复

使用道具 举报

ID:661575 发表于 2019-12-22 09:41 | 显示全部楼层
感谢分享
回复

使用道具 举报

ID:666748 发表于 2019-12-22 16:00 | 显示全部楼层
请问里面有原理图吗?
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表