国产毛片a精品毛-国产毛片黄片-国产毛片久久国产-国产毛片久久精品-青娱乐极品在线-青娱乐精品

從0開始設計_基于STM32F1的RC522讀寫卡

發布時間:2022-4-20 15:27    發布者:LOTO2018
從0開始設計_基于STM32F1的RC522讀寫卡

1.介紹
看網上很多RC522的教程都是基于讀卡ID的,這個對于很多應用來說其實沒有什么用,最近剛好有個項目需要讀寫卡,而RC522又是非常常用的且不容易缺貨的芯片,所以準備用RC522來進行讀寫卡。
2.設備準備
首先準備一個開發板和一個RC522模塊,開發板這里我選擇正點原子的精英板(STM32F103ZET6),具體如下板子如下圖1所示。

接下來就是接線,我選擇的是SPI2,對應的接線如下:
RST   -->  PC4
MISO -->  PB14
MOSI -->  PB15
SCK   -->  PB13
SDA   -->  PB0
上面是硬件名稱的相應接口,對于SPI來說SDA就是SPI的CS(片選)線,記得RC522模塊的供電采用3.3V,可別接成5V了。
3.工程配置
首先打開外部時鐘,配置如下圖2所示。

根據外部晶振配置對應的外部晶振頻率,設置為最大的72MB。
配置SPI2,首先配置位數,頻率,以及模式,片選采用軟件方式。

接下來配置引腳,由于片選已經采用軟件的方式,所以只需要配置MISO,MOSI和SCK了。

RST和CS直接采用GPIO的配置。
最后配置一下UART即可,選擇115200波特率,引腳默認。

設置完成之后,所有引腳如圖8所示。

4.程序編寫
首先需要導入RC522的庫,只有兩個文件分別是【RC522.c】和【RC522.h】。
接下來修改RC522.c中的硬件接口,將SPI讀寫修改成如下代碼。
#include "RC522.h"

//三目運算符true取前面那個
#define RS522_RST(N) HAL_GPIO_WritePin(RC522_RST_GPIO_Port, RC522_RST_Pin, N==1?GPIO_PIN_SET:GPIO_PIN_RESET)
#define RS522_NSS(N) HAL_GPIO_WritePin(RC522_CS_GPIO_Port, RC522_CS_Pin, N==1?GPIO_PIN_SET:GPIO_PIN_RESET)
#define osDelay HAL_Delay
/**************************************************************************************
* 函數名稱:MFRC_Init
* 功能描述:MFRC初始化
* 入口參數:無
* 出口參數:無
* 返 回 值:無
* 說    明:MFRC的SPI接口速率為0~10Mbps
***************************************************************************************/
void MFRC_Init(void)
{
    RS522_NSS(1);
    RS522_RST(1);
}

/**************************************************************************************
* 函數名稱: SPI_RW_Byte
* 功能描述: 模擬SPI讀寫一個字節
* 入口參數: -byte:要發送的數據
* 出口參數: -byte:接收到的數據
***************************************************************************************/
static uint8_t ret;       //些函數是HAL與標準庫不同和地方,【讀寫函數】
uint8_t SPI2_RW_Byte(uint8_t byte)
{
    HAL_SPI_TransmitReceive(&hspi2, &byte, &ret, 1, 10);//把byte寫入,并讀出一個值 存入ret
    return   ret;                 //入口是byte的地址,讀取時用的也是ret的地址;1:一次只寫入一個值 10:timeout     
}   

/**************************************************************************************
* 函數名稱:MFRC_WriteReg
* 功能描述:寫一個寄存器
* 入口參數:-addr:待寫的寄存器地址
*           -data:待寫的寄存器數據
* 出口參數:無
* 返 回 值:無
* 說    明:無
***************************************************************************************/
void MFRC_WriteReg(uint8_t addr, uint8_t data)
{
    uint8_t AddrByte;
    AddrByte = (addr << 1 ) & 0x7E; //求出地址字節
    RS522_NSS(0);                   //NSS拉低
    SPI2_RW_Byte(AddrByte);         //寫地址字節
    SPI2_RW_Byte(data);             //寫數據
    RS522_NSS(1);                   //NSS拉高
}


/**************************************************************************************
* 函數名稱:MFRC_ReadReg
* 功能描述:讀一個寄存器
* 入口參數:-addr:待讀的寄存器地址
* 出口參數:無
* 返 回 值:-data:讀到寄存器的數據
* 說    明:無
***************************************************************************************/
uint8_t MFRC_ReadReg(uint8_t addr)
{
    uint8_t AddrByte, data;
    AddrByte = ((addr << 1 ) & 0x7E ) | 0x80;   //求出地址字節
    RS522_NSS(0);                               //NSS拉低
    SPI2_RW_Byte(AddrByte);                     //寫地址字節
    data = SPI2_RW_Byte(0x00);                  //讀數據
    RS522_NSS(1);                               //NSS拉高
    return data;
}
其他接口保持不變,我們來看一下RC522提供的接口和指令有哪些。
#ifndef _RC522_H
#define _RC522_H

//頭文件
//************************************************
#include "gpio.h"http://要一些引腳上的宏定義
#include "spi.h"http://硬件SPI的定義
#include "printf.h"
#include "main.h"http://Laber User上的宏定義
//************************************************

//MFRC522驅動程序
//************************************************

/*MFRC522寄存器定義*/
//PAGE0
#define MFRC_RFU00                      0x00   
#define MFRC_CommandReg                 0x01   
#define MFRC_ComIEnReg                     0x02   
#define MFRC_DivlEnReg                     0x03   
#define MFRC_ComIrqReg                     0x04   
#define MFRC_DivIrqReg                     0x05
#define MFRC_ErrorReg                      0x06   
#define MFRC_Status1Reg                    0x07   
#define MFRC_Status2Reg                    0x08   
#define MFRC_FIFODataReg                   0x09
#define MFRC_FIFOLevelReg                  0x0A
#define MFRC_WaterLevelReg                 0x0B
#define MFRC_ControlReg                    0x0C
#define MFRC_BitFramingReg                 0x0D
#define MFRC_CollReg                       0x0E
#define MFRC_RFU0F                         0x0F
//PAGE1     
#define MFRC_RFU10                         0x10
#define MFRC_ModeReg                       0x11
#define MFRC_TxModeReg                     0x12
#define MFRC_RxModeReg                     0x13
#define MFRC_TxControlReg                  0x14
#define MFRC_TxAutoReg                     0x15 //中文手冊有誤
#define MFRC_TxSelReg                      0x16
#define MFRC_RxSelReg                      0x17
#define MFRC_RxThresholdReg                0x18
#define MFRC_DemodReg                      0x19
#define MFRC_RFU1A                         0x1A
#define MFRC_RFU1B                         0x1B
#define MFRC_MifareReg                     0x1C
#define MFRC_RFU1D                         0x1D
#define MFRC_RFU1E                         0x1E
#define MFRC_SerialSpeedReg                0x1F
//PAGE2   
#define MFRC_RFU20                         0x20  
#define MFRC_CRCResultRegM                 0x21
#define MFRC_CRCResultRegL                 0x22
#define MFRC_RFU23                         0x23
#define MFRC_ModWidthReg                   0x24
#define MFRC_RFU25                         0x25
#define MFRC_RFCfgReg                      0x26
#define MFRC_GsNReg                        0x27
#define MFRC_CWGsCfgReg                    0x28
#define MFRC_ModGsCfgReg                   0x29
#define MFRC_TModeReg                      0x2A
#define MFRC_TPrescalerReg                 0x2B
#define MFRC_TReloadRegH                   0x2C
#define MFRC_TReloadRegL                   0x2D
#define MFRC_TCounterValueRegH             0x2E
#define MFRC_TCounterValueRegL             0x2F
//PAGE3      
#define MFRC_RFU30                         0x30
#define MFRC_TestSel1Reg                   0x31
#define MFRC_TestSel2Reg                   0x32
#define MFRC_TestPinEnReg                  0x33
#define MFRC_TestPinValueReg               0x34
#define MFRC_TestBusReg                    0x35
#define MFRC_AutoTestReg                   0x36
#define MFRC_VersionReg                    0x37
#define MFRC_AnalogTestReg                 0x38
#define MFRC_TestDAC1Reg                   0x39  
#define MFRC_TestDAC2Reg                   0x3A   
#define MFRC_TestADCReg                    0x3B   
#define MFRC_RFU3C                         0x3C   
#define MFRC_RFU3D                         0x3D   
#define MFRC_RFU3E                         0x3E   
#define MFRC_RFU3F                         0x3F

/*MFRC522的FIFO長度定義*/
#define MFRC_FIFO_LENGTH                       64

/*MFRC522傳輸的幀長定義*/
#define MFRC_MAXRLEN                18               

/*MFRC522命令集,中文手冊P59*/
#define MFRC_IDLE                              0x00        //取消當前命令的執行
#define MFRC_CALCCRC                           0x03    //激活CRC計算
#define MFRC_TRANSMIT                          0x04    //發送FIFO緩沖區內容
#define MFRC_NOCMDCHANGE            0x07        //無命令改變
#define MFRC_RECEIVE                           0x08    //激活接收器接收數據
#define MFRC_TRANSCEIVE                        0x0C    //發送并接收數據
#define MFRC_AUTHENT                           0x0E    //執行Mifare認證(驗證密鑰)
#define MFRC_RESETPHASE                        0x0F    //復位MFRC522

/*MFRC522通訊時返回的錯誤代碼*/
#define MFRC_OK                         (char)(0)
#define MFRC_NOTAGERR                    (char)(-1)
#define MFRC_ERR                        (char)(-2)

/*MFRC522函數聲明*/
void MFRC_Init(void);
void MFRC_WriteReg(uint8_t addr, uint8_t data);
uint8_t MFRC_ReadReg(uint8_t addr);
void MFRC_SetBitMask(uint8_t addr, uint8_t mask);
void MFRC_ClrBitMask(uint8_t addr, uint8_t mask);
void MFRC_CalulateCRC(uint8_t *pInData, uint8_t len, uint8_t *pOutData);
char MFRC_CmdFrame(uint8_t cmd, uint8_t *pInData, uint8_t InLenByte, uint8_t *pOutData, uint16_t *pOutLenBit);
//********************************************************************

//MFRC552與MF1卡通訊接口程序
//*********************************************************************
/*Mifare1卡片命令字*/
#define PICC_REQIDL                   0x26                       //尋天線區內未進入休眠狀態的卡
#define PICC_REQALL                   0x52                       //尋天線區內全部卡
#define PICC_ANTICOLL1                0x93                       //防沖撞
#define PICC_ANTICOLL2                0x95                       //防沖撞
#define PICC_AUTHENT1A                0x60                       //驗證A密鑰
#define PICC_AUTHENT1B                0x61                       //驗證B密鑰
#define PICC_READ                     0x30                       //讀塊
#define PICC_WRITE                    0xA0                       //寫塊
#define PICC_DECREMENT                0xC0                       //減值(扣除)
#define PICC_INCREMENT                0xC1                       //增值(充值)
#define PICC_TRANSFER                 0xB0                       //轉存(傳送)
#define PICC_RESTORE                  0xC2                       //恢復(重儲)
#define PICC_HALT                     0x50                       //休眠

/*PCD通訊時返回的錯誤代碼*/
#define PCD_OK                         (char)0                                //成功
#define PCD_NOTAGERR            (char)(-1)                        //無卡
#define PCD_ERR                        (char)(-2)                        //出錯

/*PCD函數聲明*/
void PCD_Init(void);//讀寫器初始化
void PCD_Reset(void);
void PCD_AntennaOn(void);
void PCD_AntennaOff(void);
char PCD_Request(uint8_t RequestMode, uint8_t *pCardType);  //尋卡,并返回卡的類型
char PCD_Anticoll(uint8_t *pSnr);                           //防沖突,返回卡號
char PCD_Select(uint8_t *pSnr);                             //選卡
char PCD_AuthState(uint8_t AuthMode, uint8_t BlockAddr, uint8_t *pKey, uint8_t *pSnr); //驗證密碼(密碼A和密碼B)   
char PCD_WriteBlock(uint8_t BlockAddr, uint8_t *pData);   //寫數據
char PCD_ReadBlock(uint8_t BlockAddr, uint8_t *pData);    //讀數據
char PCD_Value(uint8_t mode, uint8_t BlockAddr, uint8_t *pValue);   
char PCD_BakValue(uint8_t sourceBlockAddr, uint8_t goalBlockAddr);                                 
char PCD_Halt(void);
//******************************************************************************************

#endif
不過接下來我們需要測試一下,SPI是否正常,接上LOTO示波器OSCA02,最近出門在外,不方便帶示波器,所以帶了一個LOTO的便攜示波器,不過他剛好有邏輯分析儀的功能,剛好測試一下它的性能,接線圖如下,需要將ChA口接到時鐘線上,這樣才能執行觸發功能。

然后將代碼進入調試,進入讀寄存器函數,在進入前打個斷點,然后開啟LOTO示波器的觸發功能,然后運行到讀取結束,可以看到讀取到了【0x83】這個值。
再來看看邏輯分析儀讀取到的值,可以看到也是【0x83】,說明這個邏輯分析儀性能還行。

SPI功能測試完了,接下來就要進行讀寫卡了。首先科普一下讀寫卡的整個過程【尋卡-》放沖撞-》選卡-》解密卡-》讀/寫卡】。
按照上面的流程,調用相關的函數,整體代碼如下。
/* USER CODE BEGIN Header */
/**
  ******************************************************************************
  * @file           : main.c
  * @brief          : Main program body
  ******************************************************************************
  * @attention
  *
  *

© Copyright (c) 2021 STMicroelectronics.
  * All rights reserved.


  *
  * This software component is licensed by ST under BSD 3-Clause license,
  * the "License"; You may not use this file except in compliance with the
  * License. You may obtain a copy of the License at:
  *                        opensource.org/licenses/BSD-3-Clause
  *
  ******************************************************************************
  */
/* USER CODE END Header */
/* Includes ------------------------------------------------------------------*/
#include "main.h"
#include "spi.h"
#include "usart.h"
#include "gpio.h"

/* Private includes ----------------------------------------------------------*/
/* USER CODE BEGIN Includes */
#include "delay.h"
#include "printf.h"
#include "rc522.h"
#include "string.h"
/* USER CODE END Includes */

/* Private typedef -----------------------------------------------------------*/
/* USER CODE BEGIN PTD */

/* USER CODE END PTD */

/* Private define ------------------------------------------------------------*/
/* USER CODE BEGIN PD */
/* USER CODE END PD */

/* Private macro -------------------------------------------------------------*/
/* USER CODE BEGIN PM */

/* USER CODE END PM */

/* Private variables ---------------------------------------------------------*/

/* USER CODE BEGIN PV */

/* USER CODE END PV */

/* Private function prototypes -----------------------------------------------*/
void SystemClock_Config(void);
/* USER CODE BEGIN PFP */

/* USER CODE END PFP */

/* Private user code ---------------------------------------------------------*/
/* USER CODE BEGIN 0 */
uint8_t key_A[6] = {0xFF,0xFF,0xFF,0xFF,0xFF,0xFF};
char *WriteData = {"1234567890ABCDEF"};
char ReadData[16] = {0};
/* USER CODE END 0 */

/**
  * @brief  The application entry point.
  * @retval int
  */
int main(void)
{
  /* USER CODE BEGIN 1 */
  char pcd_err = 0;
  /* USER CODE END 1 */

  /* MCU Configuration--------------------------------------------------------*/

  /* Reset of all peripherals, Initializes the Flash interface and the Systick. */
  HAL_Init();

  /* USER CODE BEGIN Init */

  /* USER CODE END Init */

  /* Configure the system clock */
  SystemClock_Config();

  /* USER CODE BEGIN SysInit */

  /* USER CODE END SysInit */

  /* Initialize all configured peripherals */
  MX_GPIO_Init();
  MX_USART1_UART_Init();
  MX_SPI2_Init();
  /* USER CODE BEGIN 2 */
  //初始化
  //*************************
  PCD_Init();//RC522初始化
  //*************************
  //全局變量
  //*************************
  uint8_t RxBuffer[4];
  char Card_ID[8];
  //*************************
  /* USER CODE END 2 */

  /* Infinite loop */
  /* USER CODE BEGIN WHILE */
  while (1)
  {
    /* USER CODE END WHILE */

    /* USER CODE BEGIN 3 */
    pcd_err = PCD_Request(PICC_REQIDL, RxBuffer);//返回值為0,代表尋卡成功;并把卡類型存入RxBuffer中
    if(pcd_err == PCD_OK)
    {
        uint16_t cardType = (RxBuffer[0] << 8) | RxBuffer[1];
        
        printf("卡類型:0x%04X\r\n", cardType);
        
        pcd_err = PCD_Anticoll(RxBuffer);   //防沖撞,完成這部就可以簡單地 讀取卡號,本次不涉及更高層次應用
        if(pcd_err == PCD_OK)
        {
            sprintf(Card_ID,"%x%x%x%x",RxBuffer[0],RxBuffer[1],RxBuffer[2],RxBuffer[3]);
            printf("ID=%s\r\n",Card_ID);
        }
        
        pcd_err = PCD_Select((uint8_t *)RxBuffer);       //選卡
        if(pcd_err == PCD_OK)
        {
            printf("Select Card OK\r\n");
        }
        else
        {
            printf("Select Card Error\r\n");
        }
        
        pcd_err = PCD_AuthState(PICC_AUTHENT1A, 5, key_A, RxBuffer);    //解密
        if(pcd_err == PCD_OK)
        {
            printf("Auth Card OK\r\n");
        }
        else
        {
            printf("Auth Card Error\r\n");
        }
        
        pcd_err = PCD_WriteBlock(6, (uint8_t *)WriteData);  //寫卡
        if(pcd_err == PCD_OK)
        {
            printf("寫卡成功\r\n");
        }
        else
        {
            printf("寫卡失敗:%d\r\n",pcd_err);
        }
        
        HAL_Delay(1);
        pcd_err = PCD_ReadBlock(6, (uint8_t *)ReadData);    //讀卡
        if(pcd_err == PCD_OK)
        {
            printf("讀卡成功:%s\r\n", ReadData);
        }
        else
        {
            printf("讀卡失敗:%d\r\n",pcd_err);
        }
        
        PCD_Halt();
        
        memset(RxBuffer, 0, sizeof(RxBuffer));//清空字符串,這里要清除RxBuffer才行
        HAL_Delay(1000);
    }
    HAL_Delay(100);
  }
  /* USER CODE END 3 */
}

/**
  * @brief System Clock Configuration
  * @retval None
  */
void SystemClock_Config(void)
{
  RCC_OscInitTypeDef RCC_OscInitStruct = {0};
  RCC_ClkInitTypeDef RCC_ClkInitStruct = {0};

  /** Initializes the RCC Oscillators according to the specified parameters
  * in the RCC_OscInitTypeDef structure.
  */
  RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSE;
  RCC_OscInitStruct.HSEState = RCC_HSE_ON;
  RCC_OscInitStruct.HSEPredivValue = RCC_HSE_PREDIV_DIV1;
  RCC_OscInitStruct.HSIState = RCC_HSI_ON;
  RCC_OscInitStruct.PLL.PLLState = RCC_PLL_ON;
  RCC_OscInitStruct.PLL.PLLSource = RCC_PLLSOURCE_HSE;
  RCC_OscInitStruct.PLL.PLLMUL = RCC_PLL_MUL9;
  if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK)
  {
    Error_Handler();
  }
  /** Initializes the CPU, AHB and APB buses clocks
  */
  RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_HCLK|RCC_CLOCKTYPE_SYSCLK
                              |RCC_CLOCKTYPE_PCLK1|RCC_CLOCKTYPE_PCLK2;
  RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_PLLCLK;
  RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1;
  RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV2;
  RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1;

  if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_2) != HAL_OK)
  {
    Error_Handler();
  }
}

/* USER CODE BEGIN 4 */

/* USER CODE END 4 */

/**
  * @brief  This function is executed in case of error occurrence.
  * @retval None
  */
void Error_Handler(void)
{
  /* USER CODE BEGIN Error_Handler_Debug */
  /* User can add his own implementation to report the HAL error return state */
  __disable_irq();
  printf("error\r\n");
  while (1)
  {
  }
  /* USER CODE END Error_Handler_Debug */
}

#ifdef  USE_FULL_ASSERT
/**
  * @brief  Reports the name of the source file and the source line number
  *         where the assert_param error has occurred.
  * @param  file: pointer to the source file name
  * @param  line: assert_param error line source number
  * @retval None
  */
void assert_failed(uint8_t *file, uint32_t line)
{
  /* USER CODE BEGIN 6 */
  /* User can add his own implementation to report the file name and line number,
     ex: printf("Wrong parameters value: file %s on line %d\r\n", file, line) */
  /* USER CODE END 6 */
}
#endif /* USE_FULL_ASSERT */

/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/
代碼是先進**,然后進**,再進**。關于解密,卡默認的密碼是【FFFFFFFFFFFF】,一共是6個【0xFF】。
最終的輸出結果如下圖12所示,寫入【“1234567890ABCDEF”】內容,讀出的也是【“1234567890ABCDEF”】內容。

5.總結
SPI配置下來還是比較簡單的,這個工程最主要的還是得讀懂RC522的工作流程,如果能對IC卡進行讀寫,項目的基本功能就實現了,后續只要調用相關的接口接可以了,這次的分享就到這里了!

本文地址:http://m.qingdxww.cn/thread-788078-1-1.html     【打印本頁】

本站部分文章為轉載或網友發布,目的在于傳遞和分享信息,并不代表本網贊同其觀點和對其真實性負責;文章版權歸原作者及原出處所有,如涉及作品內容、版權和其它問題,我們將根據著作權人的要求,第一時間更正或刪除。
您需要登錄后才可以發表評論 登錄 | 立即注冊

廠商推薦

  • Microchip視頻專區
  • 實時控制解決方案的正確選擇——數字信號控制器(DSC)或通用MCU
  • dsPIC® DSC集成電機驅動器:非常適合在緊湊空間內進行實時控制
  • PIC32CK SG單片機——輕松滿足新型網絡安全要求
  • PIC32CM LS00 Curiosity Pro評估工具包
  • 貿澤電子(Mouser)專區

相關視頻

關于我們  -  服務條款  -  使用指南  -  站點地圖  -  友情鏈接  -  聯系我們
電子工程網 © 版權所有   京ICP備16069177號 | 京公網安備11010502021702
快速回復 返回頂部 返回列表
主站蜘蛛池模板: 思思久久96热在精品国产免费| 青青精品| 色网综合| 日韩av线上| 一本到高清| 日韩另类视频| 一级二级三级黄色片| 四虎免费观看| 小说区 亚洲 自拍 另类| 亚洲日本国产| 青青青在线视频播放免费| 天天艹综合| 污片在线看| 亚洲欧洲国产精品久久| 亚洲欧美日韩精品一区| 色黄啪啪网18以下免费进| 亚洲成在人色婷婷| 亚洲欧美中文日韩二区一区 | 午夜国产福利在线观看| 色综合欧美综合天天综合| 日韩精品视频免费观看| 四虎黄色网| 日本九九精品一区二区| 青青操原| 亚洲午夜小视频| 性的小视频在线观看免费| 日韩3区| 日韩黄色在线| 四虎最新永久免费网址| 亚洲欧美综合国产精品一区| 日本欧美国产| 日韩精品你懂的在线播放| 一级毛片特级毛片免费的| 日本高新1区2区3区| 婷婷涩五月| 四虎影院在线观看免费| 亚洲国产一区二区在线| 四虎影视永久在线精品免费| 欧美色图88| 色综合久久一本首久久| 亚洲免费国产|