[活动专区] 【AT-START-F425测评】6、硬件SPI1驱动RC522

[复制链接]
 楼主| freeelectron 发表于 2022-3-8 15:42 | 显示全部楼层 |阅读模式
本帖最后由 freeelectron 于 2022-3-8 15:49 编辑

1、关于AT32F425的SPI接口
36821622704450ef4f.png
可以看出包含3路SPI,分别是SPI1、SPI2、SPI3,本文使用SPI1。

2、开发板上SPI的引脚

98070622704d68b670.png

3、spi1引脚复用

445356227052c609ca.png


4、开发板与RC522模块连接

587766227089d72dd6.png

开发板                             rc522

PA4 <----------------------------->SDA(NSS)

PA5 <----------------------------->SCK

PA6 <----------------------------->MISO

PA7 <----------------------------->MOSI

PA3   <-----------------------------> RST

3.3V  <----------------------------->3V3

GND <----------------------------->GND


5、与SPI相关操作代码

  1. #define SET_SPI_CS      gpio_bits_set(GPIOA, GPIO_PINS_4)
  2. #define CLR_SPI_CS      gpio_bits_reset(GPIOA, GPIO_PINS_4)

  3. #define SET_RC522RST    gpio_bits_set(GPIOA, GPIO_PINS_3)
  4. #define CLR_RC522RST    gpio_bits_reset(GPIOA, GPIO_PINS_3)

  5. #define SPIReadByte()        SPIWriteByte(0)

  6. void SpiIoInit(void)
  7. {
  8.         /*SPI1 :  
  9.                 PA4/CS  
  10.                 PA5/SCK   
  11.                 PA6/MISO   
  12.                 PA7/MOSI   
  13.         */
  14.         crm_periph_clock_enable(CRM_GPIOA_PERIPH_CLOCK, TRUE);

  15.         gpio_init_type gpio_initstructure;
  16.        
  17.         gpio_initstructure.gpio_out_type       = GPIO_OUTPUT_PUSH_PULL;
  18.         gpio_initstructure.gpio_pull           = GPIO_PULL_UP;
  19.         gpio_initstructure.gpio_mode           = GPIO_MODE_OUTPUT;
  20.         gpio_initstructure.gpio_drive_strength = GPIO_DRIVE_STRENGTH_STRONGER;
  21.         gpio_initstructure.gpio_pins = GPIO_PINS_3;
  22.         gpio_init(GPIOA, &gpio_initstructure);
  23.        
  24.         gpio_initstructure.gpio_out_type       = GPIO_OUTPUT_PUSH_PULL;
  25.         gpio_initstructure.gpio_pull           = GPIO_PULL_UP;
  26.         gpio_initstructure.gpio_mode           = GPIO_MODE_OUTPUT;
  27.         gpio_initstructure.gpio_drive_strength = GPIO_DRIVE_STRENGTH_STRONGER;
  28.         gpio_initstructure.gpio_pins = GPIO_PINS_4;
  29.         gpio_init(GPIOA, &gpio_initstructure);
  30.   
  31.         gpio_pin_mux_config(GPIOA, GPIO_PINS_SOURCE5, GPIO_MUX_0);
  32.         gpio_pin_mux_config(GPIOA, GPIO_PINS_SOURCE6, GPIO_MUX_0);
  33.         gpio_pin_mux_config(GPIOA, GPIO_PINS_SOURCE7, GPIO_MUX_0);

  34.         gpio_default_para_init(&gpio_initstructure);

  35.         gpio_initstructure.gpio_out_type       = GPIO_OUTPUT_PUSH_PULL;
  36.         gpio_initstructure.gpio_pull           = GPIO_PULL_DOWN;
  37.         gpio_initstructure.gpio_mode           = GPIO_MODE_MUX;
  38.         gpio_initstructure.gpio_drive_strength = GPIO_DRIVE_STRENGTH_STRONGER;
  39.         gpio_initstructure.gpio_pins = GPIO_PINS_5;
  40.         gpio_init(GPIOA, &gpio_initstructure);

  41.         gpio_initstructure.gpio_out_type       = GPIO_OUTPUT_PUSH_PULL;
  42.         gpio_initstructure.gpio_pull           = GPIO_PULL_UP;
  43.         gpio_initstructure.gpio_mode           = GPIO_MODE_MUX;
  44.         gpio_initstructure.gpio_drive_strength = GPIO_DRIVE_STRENGTH_STRONGER;
  45.         gpio_initstructure.gpio_pins = GPIO_PINS_6;
  46.         gpio_init(GPIOA, &gpio_initstructure);

  47.         gpio_initstructure.gpio_out_type       = GPIO_OUTPUT_PUSH_PULL;
  48.         gpio_initstructure.gpio_pull           = GPIO_PULL_UP;
  49.         gpio_initstructure.gpio_mode           = GPIO_MODE_MUX;
  50.         gpio_initstructure.gpio_drive_strength = GPIO_DRIVE_STRENGTH_STRONGER;
  51.         gpio_initstructure.gpio_pins = GPIO_PINS_7;
  52.         gpio_init(GPIOA, &gpio_initstructure);
  53. }


  54. void SpiInit(void)
  55. {
  56.   spi_init_type spi_init_struct;

  57.   crm_periph_clock_enable(CRM_SPI1_PERIPH_CLOCK, TRUE);
  58.   
  59.   spi_default_para_init(&spi_init_struct);
  60.   spi_init_struct.transmission_mode = SPI_TRANSMIT_FULL_DUPLEX;
  61.   spi_init_struct.master_slave_mode = SPI_MODE_MASTER;
  62.   spi_init_struct.mclk_freq_division = SPI_MCLK_DIV_8;
  63.   spi_init_struct.first_bit_transmission = SPI_FIRST_BIT_MSB;
  64.   spi_init_struct.frame_bit_num = SPI_FRAME_8BIT;
  65.   spi_init_struct.clock_polarity = SPI_CLOCK_POLARITY_LOW;
  66.   spi_init_struct.clock_phase = SPI_CLOCK_PHASE_1EDGE;
  67.   spi_init_struct.cs_mode_selection = SPI_CS_SOFTWARE_MODE;
  68.   spi_init(SPI1, &spi_init_struct);


  69.   spi_enable(SPI1, TRUE);
  70.   
  71. }



  72. uint8_t  SPIWriteByte(uint8_t Byte)
  73. {
  74.         uint16_t ret=0;

  75.         while(spi_i2s_flag_get(SPI1, SPI_I2S_TDBE_FLAG) == RESET) {};
  76.     spi_i2s_data_transmit(SPI1, Byte);
  77.                
  78.     while(spi_i2s_flag_get(SPI1, SPI_I2S_RDBF_FLAG) == RESET) {};
  79.     ret=spi_i2s_data_receive(SPI1);
  80.        
  81.         return (ret&0x0ff);
  82. }


6、与RC522操作相关代码

  1. void RC522Init(void)
  2. {
  3.         uint8_t ver=0;
  4.        
  5.         SpiIoInit();
  6.         SpiInit();
  7.        
  8.         PcdReset();
  9.        
  10.         ver=ReadRawRC(VersionReg);
  11.        
  12.         if(ver==0x91||ver==0x92)
  13.         {
  14.                 printf("rc522 exist\r\n");
  15.         }
  16.         else
  17.         {
  18.                 printf("rc522 not exist\r\n");
  19.         }
  20.        
  21.         PcdAntennaOff();
  22.         delay_ms(2);  
  23.         PcdAntennaOn();
  24.         M500PcdConfigISOType( 'A' );

  25. }


  26. unsigned char CT[2];//卡类型
  27. unsigned char SN[4]; //卡号


  28. char cStr [ 30 ];
  29. uint8_t  ucArray_ID [ 4 ];


  30. void RC522Polling(void)
  31. {
  32.         unsigned char status;       
  33.         uint8_t j;
  34.        
  35.         status = PcdRequest(PICC_REQALL,CT);/*尋卡*/
  36.         if(status==MI_OK)//尋卡成功
  37.         {
  38.                 status=MI_ERR;
  39.                 status = PcdAnticoll(SN);/*防冲撞*/
  40.         }

  41.         if(status==MI_OK)//防衝撞成功
  42.         {
  43.                 status=MI_ERR;               
  44.                
  45.                 printf("\r\nThe Card ID is:");
  46.                 for(j=0;j<4;j++)
  47.                 {
  48.                         printf("%02X",SN[j]);
  49.                 }
  50.                 printf("\r\n");
  51.                 status =PcdSelect(SN);
  52.         }                              
  53. }
  54.                         
  55. /////////////////////////////////////////////////////////////////////
  56. //功    能:寻卡
  57. //参数说明: req_code[IN]:寻卡方式
  58. //                0x52 = 寻感应区内所有符合14443A标准的卡
  59. //                0x26 = 寻未进入休眠状态的卡
  60. //          pTagType[OUT]:卡片类型代码
  61. //                0x4400 = Mifare_UltraLight
  62. //                0x0400 = Mifare_One(S50)
  63. //                0x0200 = Mifare_One(S70)
  64. //                0x0800 = Mifare_Pro(X)
  65. //                0x4403 = Mifare_DESFire
  66. //返    回: 成功返回MI_OK
  67. /////////////////////////////////////////////////////////////////////
  68. char PcdRequest(uint8_t   req_code,uint8_t *pTagType)
  69. {
  70.         char   status;  
  71.         uint8_t   unLen;
  72.         uint8_t   ucComMF522Buf[MAXRLEN];

  73.         ClearBitMask(Status2Reg,0x08);
  74.         WriteRawRC(BitFramingReg,0x07);
  75.         SetBitMask(TxControlReg,0x03);

  76.         ucComMF522Buf[0] = req_code;

  77.         status = PcdComMF522(PCD_TRANSCEIVE,ucComMF522Buf,1,ucComMF522Buf,&unLen);

  78.         if ((status == MI_OK) && (unLen == 0x10))
  79.         {   
  80.                 *pTagType     = ucComMF522Buf[0];
  81.                 *(pTagType+1) = ucComMF522Buf[1];
  82.         }
  83.         else
  84.         {   status = MI_ERR;   }
  85.    
  86.         return status;
  87. }

  88. /////////////////////////////////////////////////////////////////////
  89. //功    能:防冲撞
  90. //参数说明: pSnr[OUT]:卡片序列号,4字节
  91. //返    回: 成功返回MI_OK
  92. /////////////////////////////////////////////////////////////////////  
  93. char PcdAnticoll(uint8_t *pSnr)
  94. {
  95.     char   status;
  96.     uint8_t   i,snr_check=0;
  97.     uint8_t   unLen;
  98.     uint8_t   ucComMF522Buf[MAXRLEN];
  99.    

  100.     ClearBitMask(Status2Reg,0x08);
  101.     WriteRawRC(BitFramingReg,0x00);
  102.     ClearBitMask(CollReg,0x80);

  103.     ucComMF522Buf[0] = PICC_ANTICOLL1;
  104.     ucComMF522Buf[1] = 0x20;

  105.     status = PcdComMF522(PCD_TRANSCEIVE,ucComMF522Buf,2,ucComMF522Buf,&unLen);

  106.     if (status == MI_OK)
  107.     {
  108.              for (i=0; i<4; i++)
  109.          {   
  110.              *(pSnr+i)  = ucComMF522Buf[i];
  111.              snr_check ^= ucComMF522Buf[i];
  112.          }
  113.          if (snr_check != ucComMF522Buf[i])
  114.          {   status = MI_ERR;    }
  115.     }
  116.    
  117.     SetBitMask(CollReg,0x80);
  118.     return status;
  119. }

  120. /////////////////////////////////////////////////////////////////////
  121. //功    能:选定卡片
  122. //参数说明: pSnr[IN]:卡片序列号,4字节
  123. //返    回: 成功返回MI_OK
  124. /////////////////////////////////////////////////////////////////////
  125. char PcdSelect(uint8_t *pSnr)
  126. {
  127.     char   status;
  128.     uint8_t   i;
  129.     uint8_t   unLen;
  130.     uint8_t   ucComMF522Buf[MAXRLEN];
  131.    
  132.     ucComMF522Buf[0] = PICC_ANTICOLL1;
  133.     ucComMF522Buf[1] = 0x70;
  134.     ucComMF522Buf[6] = 0;
  135.     for (i=0; i<4; i++)
  136.     {
  137.             ucComMF522Buf[i+2] = *(pSnr+i);
  138.             ucComMF522Buf[6]  ^= *(pSnr+i);
  139.     }
  140.     CalulateCRC(ucComMF522Buf,7,&ucComMF522Buf[7]);
  141.   
  142.     ClearBitMask(Status2Reg,0x08);

  143.     status = PcdComMF522(PCD_TRANSCEIVE,ucComMF522Buf,9,ucComMF522Buf,&unLen);
  144.    
  145.     if ((status == MI_OK) && (unLen == 0x18))
  146.     {   status = MI_OK;  }
  147.     else
  148.     {   status = MI_ERR;    }

  149.     return status;
  150. }

  151. /////////////////////////////////////////////////////////////////////
  152. //功    能:验证卡片密码
  153. //参数说明: auth_mode[IN]: 密码验证模式
  154. //                 0x60 = 验证A密钥
  155. //                 0x61 = 验证B密钥
  156. //          addr[IN]:块地址
  157. //          pKey[IN]:密码
  158. //          pSnr[IN]:卡片序列号,4字节
  159. //返    回: 成功返回MI_OK
  160. /////////////////////////////////////////////////////////////////////               
  161. char PcdAuthState(uint8_t   auth_mode,uint8_t   addr,uint8_t *pKey,uint8_t *pSnr)
  162. {
  163.     char   status;
  164.     uint8_t   unLen;
  165.     uint8_t   i,ucComMF522Buf[MAXRLEN];

  166.     ucComMF522Buf[0] = auth_mode;
  167.     ucComMF522Buf[1] = addr;
  168. //    for (i=0; i<6; i++)
  169. //    {    ucComMF522Buf[i+2] = *(pKey+i);   }
  170. //    for (i=0; i<6; i++)
  171. //    {    ucComMF522Buf[i+8] = *(pSnr+i);   }
  172.     memcpy(&ucComMF522Buf[2], pKey, 6);
  173.     memcpy(&ucComMF522Buf[8], pSnr, 4);
  174.    
  175.     status = PcdComMF522(PCD_AUTHENT,ucComMF522Buf,12,ucComMF522Buf,&unLen);
  176.     if ((status != MI_OK) || (!(ReadRawRC(Status2Reg) & 0x08)))
  177.     {   status = MI_ERR;   }
  178.    
  179.     return status;
  180. }

  181. /////////////////////////////////////////////////////////////////////
  182. //功    能:读取M1卡一块数据
  183. //参数说明: addr[IN]:块地址
  184. //          p [OUT]:读出的数据,16字节
  185. //返    回: 成功返回MI_OK
  186. /////////////////////////////////////////////////////////////////////
  187. char PcdRead(uint8_t   addr,uint8_t *p )
  188. {
  189.     char   status;
  190.     uint8_t   unLen;
  191.     uint8_t   i,ucComMF522Buf[MAXRLEN];

  192.     ucComMF522Buf[0] = PICC_READ;
  193.     ucComMF522Buf[1] = addr;
  194.     CalulateCRC(ucComMF522Buf,2,&ucComMF522Buf[2]);
  195.    
  196.     status = PcdComMF522(PCD_TRANSCEIVE,ucComMF522Buf,4,ucComMF522Buf,&unLen);
  197.     if ((status == MI_OK) && (unLen == 0x90))
  198.     {
  199.         for (i=0; i<16; i++)
  200.         {   
  201.                         *(p +i) = ucComMF522Buf[i];   
  202.                 }
  203.     }
  204.     else
  205.     {   
  206.                 status = MI_ERR;   
  207.         }
  208.    
  209.     return status;
  210. }

  211. /////////////////////////////////////////////////////////////////////
  212. //功    能:写数据到M1卡一块
  213. //参数说明: addr[IN]:块地址
  214. //          p [IN]:写入的数据,16字节
  215. //返    回: 成功返回MI_OK
  216. /////////////////////////////////////////////////////////////////////                  
  217. char PcdWrite(uint8_t   addr,uint8_t *p )
  218. {
  219.     char   status;
  220.     uint8_t   unLen;
  221.     uint8_t   i,ucComMF522Buf[MAXRLEN];
  222.    
  223.     ucComMF522Buf[0] = PICC_WRITE;
  224.     ucComMF522Buf[1] = addr;
  225.     CalulateCRC(ucComMF522Buf,2,&ucComMF522Buf[2]);

  226.     status = PcdComMF522(PCD_TRANSCEIVE,ucComMF522Buf,4,ucComMF522Buf,&unLen);

  227.     if ((status != MI_OK) || (unLen != 4) || ((ucComMF522Buf[0] & 0x0F) != 0x0A))
  228.     {   status = MI_ERR;   }
  229.         
  230.     if (status == MI_OK)
  231.     {
  232.         //memcpy(ucComMF522Buf, p , 16);
  233.         for (i=0; i<16; i++)
  234.         {   
  235.                 ucComMF522Buf[i] = *(p +i);   
  236.         }
  237.         CalulateCRC(ucComMF522Buf,16,&ucComMF522Buf[16]);

  238.         status = PcdComMF522(PCD_TRANSCEIVE,ucComMF522Buf,18,ucComMF522Buf,&unLen);
  239.         if ((status != MI_OK) || (unLen != 4) || ((ucComMF522Buf[0] & 0x0F) != 0x0A))
  240.         {   status = MI_ERR;   }
  241.     }
  242.    
  243.     return status;
  244. }

  245. /////////////////////////////////////////////////////////////////////
  246. //功    能:命令卡片进入休眠状态
  247. //返    回: 成功返回MI_OK
  248. /////////////////////////////////////////////////////////////////////
  249. char PcdHalt(void)
  250. {
  251.     uint8_t   status;
  252.     uint8_t   unLen;
  253.     uint8_t   ucComMF522Buf[MAXRLEN];

  254.     ucComMF522Buf[0] = PICC_HALT;
  255.     ucComMF522Buf[1] = 0;
  256.     CalulateCRC(ucComMF522Buf,2,&ucComMF522Buf[2]);

  257.     status = PcdComMF522(PCD_TRANSCEIVE,ucComMF522Buf,4,ucComMF522Buf,&unLen);

  258.     return MI_OK;
  259. }

  260. /////////////////////////////////////////////////////////////////////
  261. //用MF522计算CRC16函数
  262. /////////////////////////////////////////////////////////////////////
  263. void CalulateCRC(uint8_t *pIn ,uint8_t   len,uint8_t *pOut )
  264. {
  265.     uint8_t   i,n;
  266.     ClearBitMask(DivIrqReg,0x04);
  267.     WriteRawRC(CommandReg,PCD_IDLE);
  268.     SetBitMask(FIFOLevelReg,0x80);
  269.     for (i=0; i<len; i++)
  270.     {   WriteRawRC(FIFODataReg, *(pIn +i));   }
  271.     WriteRawRC(CommandReg, PCD_CALCCRC);
  272.     i = 0xFF;
  273.     do
  274.     {
  275.         n = ReadRawRC(DivIrqReg);
  276.         i--;
  277.     }
  278.     while ((i!=0) && !(n&0x04));
  279.     pOut [0] = ReadRawRC(CRCResultRegL);
  280.     pOut [1] = ReadRawRC(CRCResultRegM);
  281. }

  282. /////////////////////////////////////////////////////////////////////
  283. //功    能:复位RC522
  284. //返    回: 成功返回MI_OK
  285. /////////////////////////////////////////////////////////////////////
  286. char PcdReset(void)
  287. {
  288.         SET_RC522RST;
  289.         delay_ms(1);
  290.         CLR_RC522RST;
  291.         delay_ms(1);
  292.         SET_RC522RST;
  293.         delay_ms(1);

  294.     WriteRawRC(CommandReg,PCD_RESETPHASE);
  295.         WriteRawRC(CommandReg,PCD_RESETPHASE);
  296.         delay_ms(1);

  297.     WriteRawRC(ModeReg,0x3D);            //和Mifare卡通讯,CRC初始值0x6363
  298.     WriteRawRC(TReloadRegL,30);           
  299.     WriteRawRC(TReloadRegH,0);
  300.     WriteRawRC(TModeReg,0x8D);
  301.     WriteRawRC(TPrescalerReg,0x3E);
  302.        
  303.         WriteRawRC(TxAutoReg,0x40);//必须要
  304.    
  305.     return MI_OK;
  306. }
  307. //////////////////////////////////////////////////////////////////////
  308. //设置RC632的工作方式
  309. //////////////////////////////////////////////////////////////////////
  310. char M500PcdConfigISOType(uint8_t   type)
  311. {
  312.    if (type == 'A')                     //ISO14443_A
  313.    {
  314.        ClearBitMask(Status2Reg,0x08);
  315.        WriteRawRC(ModeReg,0x3D);//3F
  316.        WriteRawRC(RxSelReg,0x86);//84
  317.        WriteRawRC(RFCfgReg,0x7F);   //4F
  318.               WriteRawRC(TReloadRegL,30);//tmoLength);// TReloadVal = 'h6a =tmoLength(dec)
  319.            WriteRawRC(TReloadRegH,0);
  320.        WriteRawRC(TModeReg,0x8D);
  321.            WriteRawRC(TPrescalerReg,0x3E);
  322.            delay_ms(2);
  323.        PcdAntennaOn();
  324.    }
  325.    else{ return 1; }
  326.    
  327.    return MI_OK;
  328. }
  329. /////////////////////////////////////////////////////////////////////
  330. //功    能:读RC632寄存器
  331. //参数说明:Address[IN]:寄存器地址
  332. //返    回:读出的值
  333. /////////////////////////////////////////////////////////////////////
  334. uint8_t ReadRawRC(uint8_t   Address)
  335. {
  336.     uint8_t   ucAddr;
  337.     uint8_t   ucResult=0;
  338.         CLR_SPI_CS;
  339.     ucAddr = ((Address<<1)&0x7E)|0x80;
  340.        
  341.         SPIWriteByte(ucAddr);
  342.         ucResult=SPIReadByte();
  343.         SET_SPI_CS;
  344.    return ucResult;
  345. }

  346. /////////////////////////////////////////////////////////////////////
  347. //功    能:写RC632寄存器
  348. //参数说明:Address[IN]:寄存器地址
  349. //          value[IN]:写入的值
  350. /////////////////////////////////////////////////////////////////////
  351. void WriteRawRC(uint8_t   Address, uint8_t   value)
  352. {  
  353.     uint8_t   ucAddr;
  354.         uint8_t tmp;

  355.         CLR_SPI_CS;
  356.     ucAddr = ((Address<<1)&0x7E);

  357.         SPIWriteByte(ucAddr);
  358.         SPIWriteByte(value);
  359.         SET_SPI_CS;
  360. }
  361. /////////////////////////////////////////////////////////////////////
  362. //功    能:置RC522寄存器位
  363. //参数说明:reg[IN]:寄存器地址
  364. //          mask[IN]:置位值
  365. /////////////////////////////////////////////////////////////////////
  366. void SetBitMask(uint8_t   reg,uint8_t   mask)  
  367. {
  368.     char   tmp = 0x0;
  369.     tmp = ReadRawRC(reg);
  370.     WriteRawRC(reg,tmp | mask);  // set bit mask
  371. }

  372. /////////////////////////////////////////////////////////////////////
  373. //功    能:清RC522寄存器位
  374. //参数说明:reg[IN]:寄存器地址
  375. //          mask[IN]:清位值
  376. /////////////////////////////////////////////////////////////////////
  377. void ClearBitMask(uint8_t   reg,uint8_t   mask)  
  378. {
  379.     char   tmp = 0x0;
  380.     tmp = ReadRawRC(reg);
  381.     WriteRawRC(reg, tmp & ~mask);  // clear bit mask
  382. }

  383. /////////////////////////////////////////////////////////////////////
  384. //功    能:通过RC522和ISO14443卡通讯
  385. //参数说明:Command[IN]:RC522命令字
  386. //          pIn [IN]:通过RC522发送到卡片的数据
  387. //          InLenByte[IN]:发送数据的字节长度
  388. //          pOut [OUT]:接收到的卡片返回数据
  389. //          *pOutLenBit[OUT]:返回数据的位长度
  390. /////////////////////////////////////////////////////////////////////
  391. char PcdComMF522(uint8_t   Command,
  392.                  uint8_t *pIn ,
  393.                  uint8_t   InLenByte,
  394.                  uint8_t *pOut ,
  395.                  uint8_t *pOutLenBit)
  396. {
  397.     char   status = MI_ERR;
  398.     uint8_t   irqEn   = 0x00;
  399.     uint8_t   waitFor = 0x00;
  400.     uint8_t   lastBits;
  401.     uint8_t   n;
  402.     uint16_t   i;
  403.     switch (Command)
  404.     {
  405.         case PCD_AUTHENT:
  406.                         irqEn   = 0x12;
  407.                         waitFor = 0x10;
  408.                         break;
  409.                 case PCD_TRANSCEIVE:
  410.                         irqEn   = 0x77;
  411.                         waitFor = 0x30;
  412.                         break;
  413.                 default:
  414.                         break;
  415.     }
  416.    
  417.     WriteRawRC(ComIEnReg,irqEn|0x80);
  418.     ClearBitMask(ComIrqReg,0x80);        //清所有中断位
  419.     WriteRawRC(CommandReg,PCD_IDLE);
  420.     SetBitMask(FIFOLevelReg,0x80);                 //清FIFO缓存
  421.    
  422.     for (i=0; i<InLenByte; i++)
  423.     {   WriteRawRC(FIFODataReg, pIn [i]);    }
  424.     WriteRawRC(CommandReg, Command);          
  425. //            n = ReadRawRC(CommandReg);
  426.    
  427.     if (Command == PCD_TRANSCEIVE)
  428.     {    SetBitMask(BitFramingReg,0x80);  }         //开始传送
  429.                                                                                      
  430.     //i = 600;//根据时钟频率调整,操作M1卡最大等待时间25ms
  431.         i = 2000;
  432.     do
  433.     {
  434.         n = ReadRawRC(ComIrqReg);
  435.         i--;
  436.     }
  437.     while ((i!=0) && !(n&0x01) && !(n&waitFor));
  438.     ClearBitMask(BitFramingReg,0x80);

  439.     if (i!=0)
  440.     {   
  441.         if(!(ReadRawRC(ErrorReg)&0x1B))
  442.         {
  443.             status = MI_OK;
  444.             if (n & irqEn & 0x01)
  445.             {   status = MI_NOTAGERR;   }
  446.             if (Command == PCD_TRANSCEIVE)
  447.             {
  448.                        n = ReadRawRC(FIFOLevelReg);
  449.                       lastBits = ReadRawRC(ControlReg) & 0x07;
  450.                 if (lastBits)
  451.                 {   *pOutLenBit = (n-1)*8 + lastBits;   }
  452.                 else
  453.                 {   *pOutLenBit = n*8;   }
  454.                 if (n == 0)
  455.                 {   n = 1;    }
  456.                 if (n > MAXRLEN)
  457.                 {   n = MAXRLEN;   }
  458.                 for (i=0; i<n; i++)
  459.                 {   pOut [i] = ReadRawRC(FIFODataReg);    }
  460.             }
  461.         }
  462.         else
  463.         {   status = MI_ERR;   }
  464.         
  465.     }
  466.    

  467.     SetBitMask(ControlReg,0x80);           // stop timer now
  468.     WriteRawRC(CommandReg,PCD_IDLE);
  469.     return status;
  470. }

  471. /////////////////////////////////////////////////////////////////////
  472. //开启天线  
  473. //每次启动或关闭天险发射之间应至少有1ms的间隔
  474. /////////////////////////////////////////////////////////////////////
  475. void PcdAntennaOn(void)
  476. {
  477.     uint8_t   i;
  478.     i = ReadRawRC(TxControlReg);
  479.     if (!(i & 0x03))
  480.     {
  481.         SetBitMask(TxControlReg, 0x03);
  482.     }
  483. }


  484. /////////////////////////////////////////////////////////////////////
  485. //关闭天线
  486. /////////////////////////////////////////////////////////////////////
  487. void PcdAntennaOff(void)
  488. {
  489.         ClearBitMask(TxControlReg, 0x03);
  490. }


  491. void Reset_RC522(void)
  492. {
  493.   PcdReset();
  494.   PcdAntennaOff();
  495.   delay_ms(2);  
  496.   PcdAntennaOn();
  497. }


7、实验现象

单片机复位后,循环读卡,测试中使用了两张卡

610076227081e06420.gif





huquanz711 发表于 2022-3-8 19:28 来自手机 | 显示全部楼层
可以加解密嘛?
 楼主| freeelectron 发表于 2022-3-8 19:29 | 显示全部楼层

本帖子,只移植了基础驱动
您需要登录后才可以回帖 登录 | 注册

本版积分规则

个人签名:stm32/LoRa物联网:304350312

65

主题

785

帖子

11

粉丝
快速回复 在线客服 返回列表 返回顶部