13-CubeMx+Keil+Proteus模擬STM32 - Flash ROM

来源:https://www.cnblogs.com/sheepeach/archive/2022/05/19/STM32F103_FlashROM.html
-Advertisement-
Play Games

本文例子參考《STM32單片機開發實例——基於Proteus虛擬模擬與HAL/LL庫》 源代碼:https://github.com/LanLinnet/STM33F103R6 項目要求 單片機將由串口收到的1位元組數據存入Flash ROM的指定地址;按下按鈕BTN,單片機將存儲在Flash ROM ...


本文例子參考《STM32單片機開發實例——基於Proteus虛擬模擬與HAL/LL庫》
源代碼:https://github.com/LanLinnet/STM33F103R6

項目要求

單片機將由串口收到的1位元組數據存入Flash ROM的指定地址;按下按鈕BTN,單片機將存儲在Flash ROM指定地址的位元組數據通過串口發送。串口通信參數:波特率為19200bit/s,無校驗。

硬體設計

  1. 第一節的基礎上,在Proteus中添加電路如下圖所示。其中我們添加了:串口組件COMPIM,用於連接電腦虛擬串口;

    調試過程也可以添加一個虛擬儀器VIRTUAL TERMINAL,用來查看單片機收到的串口數據,具體參考第11節
    由於要實現串口通信,我們要將其波特率、字長、校驗方式、停止位等都設置一下,具體參數如下圖所示
    COMPIM設置

  2. Flash ROM簡介:STM32單片機Flash ROM(程式存儲器)的作用是存放用戶編寫的單片機程式(機器碼),但是其除了用來存放單片機的程式外,也可以用來存儲一些既可以修改又能斷電保存的數據,如設備或模塊的設定參數。但是在實際中,由於STM32單片機的Flash ROM擦除次數有限,因此不建議在Flash ROM擦寫,可以通過外擴\(E^2 PROM\)、FRAM、存儲卡等方式實現保護產品設定參數的目的。不過為了熟悉Flash ROM操作,本節我們使用Flash ROM來存儲數據。
    1)STM32F103R6單片機具有32KB的FlashROM,地址為0x0800 0000 ~ 0x0800 7FFF,每KB為一頁,共32頁。
    2)Flash ROM數據寫入步驟:Flash ROM解鎖 → 擦除扇區 → 向指定地址寫入數據 → Flash ROM鎖定。
    3)Flash ROM數據讀取沒有繁瑣的步驟,直接讀取即可。

  3. 打開CubeMX,建立工程。
    首先,設置PA5為GPIO_Input
    然後,點擊“Connectivity”列表中的“USART”進行串口配置。將Mode設置為Asynchronous(非同步),波特率設為19200Bits/s,字長設為8Bits,校驗設為None,停止位設為1,數據傳送設為Receive and Transmit(接收與發送)。設置完成後,會看到右側的PA9和PA10引腳被自動設置為USART1_TXUSART1_RX,即USART1的發送端和接收端。

    隨後,再點擊“NVIC Settings”,選中USART global interrupt,使能Enabled串口1的中斷功能。

  4. 點擊“Generator Code”生成Keil工程。

軟體編寫

  1. 本次我們需要實現串口助手發送單位元組數據,單片機收到數據後存入Flash ROM,按鍵按下後將存儲的數據通過串口發回串口助手,需要用到Flash ROM相關函數其API文檔如下:
    HAL_FLASH_Unlock 解鎖Flash ROM函數

    HAL_FLASH_Lock 鎖定Flash ROM函數

    HAL_FLASHEx_Erase 擦除Flash ROM指定部分函數

    HAL_FLASH_Program 將數據寫入Flash ROM函數

    其中,TypeErase形參有以下2個巨集定義

    TypeProgram有以下3個巨集定義

  2. 點擊“Open Project”在Keil中打開工程,雙擊“main.c”文件。

  3. 首先我們需要在main.c文件中的最前面設置全局變數、聲明自定義函數。

    /* Private macro -------------------------------------------------------------*/
    /* USER CODE BEGIN PM */
    #define _FLASH_ADD 0x08006400  //寫入Flash ROM首地址(Page 25)
    /* USER CODE END PM */
    
    /* Private variables ---------------------------------------------------------*/
    
    /* USER CODE BEGIN PV */
    uint8_t rf = 0;		//自定義串口接收完畢標誌
    uint8_t RcvBuf[1];		//接收緩衝
    uint8_t SndBuf[1];		//發送緩衝
    /* USER CODE END PV */
    
    /* Private function prototypes -----------------------------------------------*/
    void SystemClock_Config(void);
    /* USER CODE BEGIN PFP */
    void FlashErase(uint32_t Add);		//聲明自定義Flash ROM擦除函數
    void FlashWrite(uint32_t Add, uint16_t Dat);		//聲明自定義Flash ROM寫函數
    uint16_t FlashRead(uint32_t Add);		//聲明自定義Flash ROM讀函數
    /* USER CODE END PFP */
    

    然後,在main函數中中插入代碼如下,定義中間變數,打開串口1接收中斷

    /* USER CODE BEGIN 1 */
    uint16_t flash_wdat;  //寫入Flash數據存儲變數
    /* USER CODE END 1 */
    
    /* USER CODE BEGIN 2 */
    //打開串口1接收中斷,接收數據存入RcvBuf數組,數組長度為1
    HAL_UART_Receive_IT(&huart1,RcvBuf,1);
    /* USER CODE END 2 */
    

    隨後,在/* USER CODE BEGIN 4 *//* USER CODE END 4 */中插入接收完畢回調函數、自定義的Flash頁擦除函數、Flash寫函數、Flash讀函數代碼如下

    /* USER CODE BEGIN 4 */
    //串口接收完畢回調函數
    void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart)
    {
      if(huart==&huart1)		//如果串口1接收完畢
      {
        rf = 1;		//標誌位置1
      }
    }
    
    /*Flash頁擦除
    *-Add表示待擦除頁的首地址
    *-Flash必須整頁擦除,也就是整頁的每個地址單元內容都為FFH才能寫入新數據
    */
    void FlashErase(uint32_t Add)
    {
      uint32_t page_error = 0;  	//錯誤指針
      FLASH_EraseInitTypeDef erase_initstruct = 
      {
        .TypeErase = FLASH_TYPEERASE_PAGES,		//擦除方式為頁擦除
        .NbPages = 1,		//頁數量為1頁
        .PageAddress = Add		//擦除頁起始地址
      };
      HAL_FLASH_Unlock();		//解鎖Flash ROM
      HAL_FLASHEx_Erase(&erase_initstruct, &page_error);		//擦除
      HAL_FLASH_Lock();		//鎖定Flash ROM
    }
    
    /*Flash寫函數
    *-寫入一個Half Word(16位)型數據
    *-Add表示Flash ROM地址
    *-Dat表示寫入數據(16位)
    *-註意:寫入時,高位元組在高地址
    */
    void FlashWrite(uint32_t Add, uint16_t Dat)
    {
      HAL_FLASH_Unlock();
      HAL_FLASH_Program(FLASH_TYPEPROGRAM_HALFWORD, Add, Dat);		//將數據寫入Flash
      HAL_FLASH_Lock();
    }
    
    /*Flash讀函數
    *-返回一個Half Word(16位)型數據
    *-Add表示Flash ROM地址
    */
    uint16_t FlashRead(uint32_t Add)
    {
      uint16_t dat;
      dat = *(uint16_t *)Add;
      return dat;
    }
    /* USER CODE END 4 */
    

    最後,在while(1)中插入代碼如下,進行Flash和串口相關操作

    /* USER CODE BEGIN WHILE */
    while (1)
    {
      if(rf == 1)		//串口接收完畢
      {
        rf = 0;		//標誌位清0
        flash_wdat = RcvBuf[0];		//將接收到的數據存入寫Flash變數中
        FlashErase(_FLASH_ADD);		//擦除指定部分
        FlashWrite(_FLASH_ADD, flash_wdat);		//寫入Flash
        HAL_UART_Receive_IT(&huart1, RcvBuf, 1);		//每次接收前都需要調用一次
      }
      if(HAL_GPIO_ReadPin(GPIOA, GPIO_PIN_5) == GPIO_PIN_RESET)		
      {
        HAL_Delay(25);		//消抖
        if(HAL_GPIO_ReadPin(GPIOA, GPIO_PIN_5) == GPIO_PIN_RESET)		//如果按鍵按下
        {
          SndBuf[0] = (uint8_t)FlashRead(_FLASH_ADD);		//讀Flash中值並存入發送緩衝
          HAL_UART_Transmit(&huart1, SndBuf, 1, 10);		//由串口1發送緩衝中的值
          while(HAL_GPIO_ReadPin(GPIOA, GPIO_PIN_5) == GPIO_PIN_RESET);		//等待按鍵鬆開
        }
      }
    /* USER CODE END WHILE */
    
    /* USER CODE BEGIN 3 */
    }
    /* USER CODE END 3 */
    

聯合調試

  1. 點擊運行,生成HEX文件。

  2. 在Proteus中載入相應HEX文件,點擊運行。

  3. 打開串口調試助手“XCOM”,選擇COM4,設置相應的波特率、停止位、數據位、奇偶校驗等,勾選“16進位顯示”和“16進位發送”,點擊“打開串口”。在發送框輸入“00”,點擊“發送”。在Proteus中我們可以看到“VIRTUAL TERMINAL”接收到數據“00”。按下按鍵,同時再觀察串口調試助手“XCOM”,可以看到接收視窗收到數據“00”。同理,發送“AA”和“BB”也能得到相應的結果。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 系列文章 p2p-tunnel 打洞內網穿透系列(一)客戶端配置及打洞 p2p-tunnel 打洞內網穿透系列(二)TCP轉發訪問遠程共用文件夾 p2p-tunnel 打洞內網穿透系列(三)TCP轉發訪問內網web服務,其它服務同理 p2p-tunnel 打洞內網穿透系列(四)socks5代理和ht ...
  • 系列文章 p2p-tunnel 打洞內網穿透系列(一)客戶端配置及打洞 p2p-tunnel 打洞內網穿透系列(二)TCP轉發訪問遠程共用文件夾 p2p-tunnel 打洞內網穿透系列(三)TCP轉發訪問內網web服務,其它服務同理 p2p-tunnel 打洞內網穿透系列(四)socks5代理和ht ...
  • 一、YUM安裝Apache服務的搭建與配置 1、關閉selinux ①修改selinux的配置文件 [root@localhost ~]# vim /etc/selinux/config SELINUX=disabled ②關閉selinux [root@localhost ~]# setenfor ...
  • Linux下的可執行程式在運行時經常需要傳一些參數,而這些參數是有規範的。包括我們自己寫的在Linux系統下運行的Shell腳本、Python腳本等可執行程式,最好也遵循相關規範。我們下麵以Linux命令為例來講解參數規範。 中括弧[]並不存在於實際的命令中,表示該參數是可選的,而加入選項設置時,通... ...
  • 認識並安裝WSL(基於Windows的Linux子系統) 什麼是WSL WSL(Windows Subsystem for Linux),這是在windows平臺運行的linux子系統。也就是說可是不用安裝虛擬機的情況下獲得相對完整的linux系統體驗。 WSL相比於虛擬機(eg:VMware、Vi ...
  • sed sed命令 sed全稱是:Stream EDitor(流編輯器。 Linux sed 命令是利用腳本來處理文本文件,sed 可依照腳本的指令來處理、編輯文本文件。Sed 主要用來自動編輯一個或多個文件、簡化對文件的反覆操作、編寫轉換程式等。 當處理數據時,sed 從輸入源一次讀入一行,並將它 ...
  • 常用命令 sudo -i然後輸入密碼登錄root賬戶(群暉預設只能使用admin賬號登陸) vim xxx編輯(編輯是進去之後按i,退出並保存是按esc,然後:wq!再回車) mkdir xx創建文件夾 準備 1. 群暉一般預設安裝docker 我們不需要進行安裝,如果沒有安裝,則去套件中心進行安裝 ...
  • 鏡像下載、功能變數名稱解析、時間同步請點擊 阿裡雲開源鏡像站 你可以在虛擬機環境里運行任何操作系統,不論是測試還是為了某種需要。 對於 Linux 而言,它在虛擬環境下的性能會優於其他操作系統。即便你可能會猶豫是否在物理機(裸金屬)上安裝 Linux 系統,你仍然可以在虛擬機中安裝一個性能幾乎和物理機一樣好 ...
一周排行
    -Advertisement-
    Play Games
  • 基於.NET Framework 4.8 開發的深度學習模型部署測試平臺,提供了YOLO框架的主流系列模型,包括YOLOv8~v9,以及其系列下的Det、Seg、Pose、Obb、Cls等應用場景,同時支持圖像與視頻檢測。模型部署引擎使用的是OpenVINO™、TensorRT、ONNX runti... ...
  • 十年沉澱,重啟開發之路 十年前,我沉浸在開發的海洋中,每日與代碼為伍,與演算法共舞。那時的我,滿懷激情,對技術的追求近乎狂熱。然而,隨著歲月的流逝,生活的忙碌逐漸占據了我的大部分時間,讓我無暇顧及技術的沉澱與積累。 十年間,我經歷了職業生涯的起伏和變遷。從初出茅廬的菜鳥到逐漸嶄露頭角的開發者,我見證了 ...
  • C# 是一種簡單、現代、面向對象和類型安全的編程語言。.NET 是由 Microsoft 創建的開發平臺,平臺包含了語言規範、工具、運行,支持開發各種應用,如Web、移動、桌面等。.NET框架有多個實現,如.NET Framework、.NET Core(及後續的.NET 5+版本),以及社區版本M... ...
  • 前言 本文介紹瞭如何使用三菱提供的MX Component插件實現對三菱PLC軟元件數據的讀寫,記錄了使用電腦模擬,模擬PLC,直至完成測試的詳細流程,並重點介紹了在這個過程中的易錯點,供參考。 用到的軟體: 1. PLC開發編程環境GX Works2,GX Works2下載鏈接 https:// ...
  • 前言 整理這個官方翻譯的系列,原因是網上大部分的 tomcat 版本比較舊,此版本為 v11 最新的版本。 開源項目 從零手寫實現 tomcat minicat 別稱【嗅虎】心有猛虎,輕嗅薔薇。 系列文章 web server apache tomcat11-01-官方文檔入門介紹 web serv ...
  • 1、jQuery介紹 jQuery是什麼 jQuery是一個快速、簡潔的JavaScript框架,是繼Prototype之後又一個優秀的JavaScript代碼庫(或JavaScript框架)。jQuery設計的宗旨是“write Less,Do More”,即倡導寫更少的代碼,做更多的事情。它封裝 ...
  • 前言 之前的文章把js引擎(aardio封裝庫) 微軟開源的js引擎(ChakraCore))寫好了,這篇文章整點js代碼來測一下bug。測試網站:https://fanyi.youdao.com/index.html#/ 逆向思路 逆向思路可以看有道翻譯js逆向(MD5加密,AES加密)附完整源碼 ...
  • 引言 現代的操作系統(Windows,Linux,Mac OS)等都可以同時打開多個軟體(任務),這些軟體在我們的感知上是同時運行的,例如我們可以一邊瀏覽網頁,一邊聽音樂。而CPU執行代碼同一時間只能執行一條,但即使我們的電腦是單核CPU也可以同時運行多個任務,如下圖所示,這是因為我們的 CPU 的 ...
  • 掌握使用Python進行文本英文統計的基本方法,並瞭解如何進一步優化和擴展這些方法,以應對更複雜的文本分析任務。 ...
  • 背景 Redis多數據源常見的場景: 分區數據處理:當數據量增長時,單個Redis實例可能無法處理所有的數據。通過使用多個Redis數據源,可以將數據分區存儲在不同的實例中,使得數據處理更加高效。 多租戶應用程式:對於多租戶應用程式,每個租戶可以擁有自己的Redis數據源,以確保數據隔離和安全性。 ...