zoukankan      html  css  js  c++  java
  • 结合51代码代码解析rfid读卡器的编程思想

    最近这两天在忙这个rfid的模块,首先我承认,本人是菜鸟,平台是基于初学者入门的51单片机,但是我还是总结一下最近这两天看代码的收获

    读卡器的软件设计:看Pdf文档好像已经给出来了了,但是初学者的话,理解也好很久,这里我晒出datasheet中的设计流程,在结合代码分析,达到可以识别出卡片的效果

    首先好像是复位应答,根据datasheet的说明,讲的是MIFARE射频卡的通信协议和通信的波特率是定义好的,当有卡进入读卡器的操作范围时,读卡器就会以特定的协议与他进行通信,判断进入的卡片是否是MIFARE射频卡

    其实datasheet上这段话好像看起来很吊,但实际上却给人一种摸不到头脑的感觉,其实,看了代码才知道,软件上的刘晨叫初始化,要对读卡器进行一次软件复位,并设定读卡器的工作方式

    这里的代码是:

     PcdReset();//rc522初始化
     PcdAntennaOff(); //关闭天线
     PcdAntennaOn();  //打开天线
     M500PcdConfigISOType( 'A' );//设定工作模式

     第二步骤是防冲突,datasheet里面讲:当有多张卡进入读卡器的感应范围的时候,防冲突机制就会启动,自动从多张卡中进行操作,之后好像有一大堆的话,讲怎么样防冲突

    其实个人感觉还是代码重要,因为不读卡,哪里来的防冲突,其实防冲突讲了这么多实现起来也就是一个函数

     status = PcdRequest(PICC_REQALL, g_ucTempbuf);
            //PICC_REQALL他是个宏定义 意思是寻找天线内的所有卡片,
            // g_ucTempbuf 是个数组,在这里函数读取卡内的前两位放在数组中
       status = PcdAnticoll(g_ucTempbuf);//防冲突

    这里有点意思的是,有的时候要实行判断卡的种类,这里有个片段,应该可以用到程序上去

    判断卡的种类,判断读卡的时候返回的第一位数据,其他的函数用的是12864的,这里大家不必去深究

                     //有卡则判断是什么卡,然后显示在液晶上
                        //                0x4400 = Mifare_UltraLight
                        //                0x0400 = Mifare_One(S50)
                        //                0x0200 = Mifare_One(S70)
                        //                0x0800 = Mifare_Pro(X)
                        //                0x4403 = Mifare_DESFire
                     switch(g_ucTempbuf[0])
                     {
                         case 0x44:
                                 ck12864_com(0x93);
                                for(i=0;i<10;i++)
                                {
                                        ck12864_data(leixing1[i]);
                                }
                             break;
                         case 0x02:
                              ck12864_com(0x93);
                                 for(i=0;i<8;i++)
                                {
                                        ck12864_data(leixing2[i]);
                                }
                             break;
                         case 0x04:
                              ck12864_com(0x93);
                                for(i=0;i<8;i++)
                                {
                                        ck12864_data(leixing3[i]);
                                }
                             break;
                         case 0x08:
                              ck12864_com(0x93);
                                 for(i=0;i<6;i++)
                                {
                                        ck12864_data(leixing4[i]);
                                }
                             break;
                     }
                     

    接下来就是流程3,选中卡片,对卡片进行操作,根据datasheet,被选中的卡的序列码,并同时返回卡的容量:

    代码实现:

     status = PcdSelect(g_ucTempbuf);

    流程4,也就是对选中的卡片进行操作,首先进行密码的核实,这里包括读写操作

    代码的实现,也是两句话:

      status = PcdAuthState(PICC_AUTHENT1A, 5, DefaultKey, g_ucTempbuf);
             if (status != MI_OK)
             {    continue;    }
             //写数据到块
             status = PcdWrite(5, data1);
             if (status != MI_OK)
             {    continue;    }
             //读一块数据
             status = PcdRead(5, g_ucTempbuf);

    流程5:将卡片处于休眠状态:

         PcdHalt();

     这5步骤可以实现对卡片的具体操作的流程,现在我们通过刷卡控制步进电机,其实可以省略步骤4

    如果做一个不记名的刷卡,也就是没有绑定特定的卡号的开门,现在就可以实现了

    示例代码:

    #include<reg52.h>
    #include"mian.h"
    #include"rc522.h"
    
    typedef unsigned int uint;
    typedef unsigned char uchar;
    uchar  status;
    uchar g_ucTempbuf[20];
    
    void main()
    {
        uint i;
        //初始化:
        PcdReset();//rc522初始化
        PcdAntennaOff(); //关闭和打开天线
        PcdAntennaOn();
        M500PcdConfigISOType('A');//设定工作模式
    
        //防冲突,这里需要一个循环,让读卡器不断去读卡
        while(1)
        {
            status = PcdRequest(PICC_REQALL, g_ucTempbuf);
            //PICC_REQALL他是个宏定义 意思是寻找天线内的所有卡片,
            // g_ucTempbuf 是个数组,在这里函数读取卡内的前两位放在数组中
            if(status != MI_OK)//没有找到卡,继续执行PcdRequest()
            {
                  continue;
            }
            
            status = PcdAnticoll(g_ucTempbuf);//防冲突
            //卡片序列号,4字节,这里的status可以判PcdAnticoll的执行情况
            //如果执行成功,表示g_ucTempbuf上面已经记在唯一的卡号了
            //在这里g_ucTempbuf已经用了2+4个
            if(status != MI_OK)//没有找到卡,继续执行PcdRequest()
            {
                  continue;
            }
            
            PcdHalt();
            if(status == MI_OK)
            {
              LED_GREEN =0;
              for(i=0;i<125;i++)
                 {
                    step();
                  }
              LED_GREEN = 1;
               }
        
           }
    }
    
    void DelayMs(unsigned int _MS)
    {
        TH1 = (unsigned char)(RCAP2_1ms>>8);
        TL1 = (unsigned char)(RCAP2_1ms);
    
        ET1     = 0;                                        // Disable timer2 interrupt
        TR1     = 1;
        while (_MS--)
        {  
            while (!TF1);
            TF1 = 0;
            TH1 = (unsigned char)(RCAP2_1ms>>8);
            TL1 = (unsigned char)(RCAP2_1ms);
        }
        TR1 = 0;
    }

    解析一下代码:这个代码只是包含一个寻卡,防冲突,确定有卡就可以开门,这个是最原始的开门例程,就像点一个led灯一样,接下来的工作只需要对卡片进行具体的操作就可以了

    版权所有,转载请注明链接地址:http://www.cnblogs.com/fengdashen/

  • 相关阅读:
    Hibernate中使用Spring Data JPA
    Spring Boot入门——全局异常处理
    Spring Boot入门——Redis
    Spring Boot入门——集成Mybatis
    Spring Boot入门——JDBCTemplate使用及其相关问题解决
    Spring Boot连接Mysql数据库问题解决
    Spring Boot入门——JPA
    Spring Boot入门——tomcat配置
    Spring Boot 配置文件
    启动图案配置
  • 原文地址:https://www.cnblogs.com/fengdashen/p/3468741.html
Copyright © 2011-2022 走看看