物聯(lián)傳媒 旗下網(wǎng)站
登錄 注冊
RFID世界網(wǎng) >  技術文章  >  其他  >  正文

結(jié)合51代碼解析rfid讀卡器的編程思想

作者:佚名
來源:RFID世界網(wǎng)
日期:2019-02-13 14:55:47
摘要:最近這兩天在忙這個rfid的模塊,首先我承認,本人是菜鳥,平臺是基于初學者入門的51單片機,但是我還是總結(jié)一下最近這兩天看代碼的收獲。
關鍵詞:51代碼rfid讀卡器

  最近這兩天在忙這個rfid的模塊,首先我承認,本人是菜鳥,平臺是基于初學者入門的51單片機,但是我還是總結(jié)一下最近這兩天看代碼的收獲。

  讀卡器的軟件設計:看Pdf文檔好像已經(jīng)給出來了了,但是初學者的話,理解也好很久,這里我曬出datasheet中的設計流程,在結(jié)合代碼分析,達到可以識別出卡片的效果。

結(jié)合51代碼代碼解析rfid讀卡器的編程思想

  首先好像是復位應答,根據(jù)datasheet的說明,講的是MIFARE射頻卡的通信協(xié)議和通信的波特率是定義好的,當有卡進入讀卡器的操作范圍時,讀卡器就會以特定的協(xié)議與他進行通信,判斷進入的卡片是否是MIFARE射頻卡

  其實datasheet上這段話好像看起來很吊,但實際上卻給人一種摸不到頭腦的感覺,其實,看了代碼才知道,軟件上的劉晨叫初始化,要對讀卡器進行一次軟件復位,并設定讀卡器的工作方式

  這里的代碼是:

  PcdReset();//rc522初始化

  PcdAntennaOff(); //關閉天線

  PcdAntennaOn(); //打開天線

  M500PcdConfigISOType( 'A' );//設定工作模式

  第二步驟是防沖突,datasheet里面講:當有多張卡進入讀卡器的感應范圍的時候,防沖突機制就會啟動,自動從多張卡中進行操作,之后好像有一大堆的話,講怎么樣防沖突

  其實個人感覺還是代碼重要,因為不讀卡,哪里來的防沖突,其實防沖突講了這么多實現(xiàn)起來也就是一個函數(shù)

  status = PcdRequest(PICC_REQALL, g_ucTempbuf);

  //PICC_REQALL他是個宏定義 意思是尋找天線內(nèi)的所有卡片,

  // g_ucTempbuf 是個數(shù)組,在這里函數(shù)讀取卡內(nèi)的前兩位放在數(shù)組中

  status = PcdAnticoll(g_ucTempbuf);//防沖突

  這里有點意思的是,有的時候要實行判斷卡的種類,這里有個片段,應該可以用到程序上去

  判斷卡的種類,判斷讀卡的時候返回的第一位數(shù)據(jù),其他的函數(shù)用的是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,選中卡片,對卡片進行操作,根據(jù)datasheet,被選中的卡的序列碼,并同時返回卡的容量:

  代碼實現(xiàn):

  status = PcdSelect(g_ucTempbuf);

  流程4,也就是對選中的卡片進行操作,首先進行密碼的核實,這里包括讀寫操作

  代碼的實現(xiàn),也是兩句話:

  status = PcdAuthState(PICC_AUTHENT1A, 5, DefaultKey, g_ucTempbuf);

  if (status != MI_OK)

  { continue; }

  //寫數(shù)據(jù)到塊

  status = PcdWrite(5, data1);

  if (status != MI_OK)

  { continue; }

  //讀一塊數(shù)據(jù)

  流程5:將卡片處于休眠狀態(tài):

  PcdHalt();

  這5步驟可以實現(xiàn)對卡片的具體操作的流程,現(xiàn)在我們通過刷卡控制步進電機,其實可以省略步驟4

  如果做一個不記名的刷卡,也就是沒有綁定特定的卡號的開門,現(xiàn)在就可以實現(xiàn)了

  示例代碼:

  #include

  #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');//設定工作模式

  //防沖突,這里需要一個循環(huán),讓讀卡器不斷去讀卡

  while(1)

  {

  status = PcdRequest(PICC_REQALL, g_ucTempbuf);

  //PICC_REQALL他是個宏定義 意思是尋找天線內(nèi)的所有卡片,

  // g_ucTempbuf 是個數(shù)組,在這里函數(shù)讀取卡內(nèi)的前兩位放在數(shù)組中

  if(status != MI_OK)//沒有找到卡,繼續(xù)執(zhí)行PcdRequest()

  {

  continue;

  }

  status = PcdAnticoll(g_ucTempbuf);//防沖突

  //卡片序列號,4字節(jié),這里的status可以判PcdAnticoll的執(zhí)行情況

  //如果執(zhí)行成功,表示g_ucTempbuf上面已經(jīng)記在唯一的卡號了

  //在這里g_ucTempbuf已經(jīng)用了2+4個

  if(status != MI_OK)//沒有找到卡,繼續(xù)執(zhí)行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