STM32學習筆記之存儲器映射(原理篇)

📢:如果你也對機器人、人工智能感興趣,看來我們志同道合?
📢:不妨瀏覽一下我的博客主頁【https://blog.csdn.net/weixin_51244852】
📢:文章若有幸對你有幫助,可點贊 👍 收藏 ?不迷路🙉
📢:內容若有錯誤,敬請留言 📝指正!原創文,轉載注明出處

文章目錄

  • 什么是存儲器映射呢?
  • 為什么叫存儲器映射,而不是寄存器映射呢?
  • 存儲器映射實現
  • 什么是寄存器映射?
  • 寄存器映射實現
  • 為什么要設置存儲器映射?
  • stm32單片機地址總線訪問大小是4G,是不是意味著內存大小也是4G?


什么是存儲器映射呢?

存儲器本身不具備地址,所以把芯片內核所預先設定好的地址分配給寄存器,就是存儲器映射。因為stm32的地址線是32位,也就是2的32次方,正好是對應4G的虛擬存儲空間。把內核廠商(也就是ARM公司)定義的這個虛擬空間與芯片廠商(這里是ST)芯片內部外設進行對應,也就是給存儲器分配地址,即存儲器映射。4個G的地址這么大,用不完沒關系,可以保留。

為什么叫存儲器映射,而不是寄存器映射呢?

在 STM32 單片機中,“存儲器映射” 是指將不同的物理存儲區域(如內部閃存、SRAM、外設寄存器等)和一些邏輯存儲區域(如系統保留區域等)按照一定的規則,映射到一個統一的地址空間中,使得 CPU 可以通過訪問這個統一地址空間中的不同地址來訪問不同的物理或邏輯存儲區域。因此存儲器映射包含寄存器映射。

存儲器映射實現

請查看存儲器映射關系在代碼中是如何體現的。下圖是我項目中的 stm32f4xx.h 文件的部分代碼。

#define FLASH_BASE            ((uint32_t)0x08000000) /*!< FLASH(up to 1 MB) base address in the alias region                         */
#define CCMDATARAM_BASE       ((uint32_t)0x10000000) /*!< CCM(core coupled memory) data RAM(64 KB) base address in the alias region  */
#define SRAM1_BASE            ((uint32_t)0x20000000) /*!< SRAM1(112 KB) base address in the alias region                             */
#define SRAM2_BASE            ((uint32_t)0x2001C000) /*!< SRAM2(16 KB) base address in the alias region                              */
#define SRAM3_BASE            ((uint32_t)0x20020000) /*!< SRAM3(64 KB) base address in the alias region                              */
#define PERIPH_BASE           ((uint32_t)0x40000000) /*!< Peripheral base address in the alias region                                */
#define BKPSRAM_BASE          ((uint32_t)0x40024000) /*!< Backup SRAM(4 KB) base address in the alias region                         */#if defined (STM32F40_41xxx)
#define FSMC_R_BASE           ((uint32_t)0xA0000000) /*!< FSMC registers base address                                                */
#endif /* STM32F40_41xxx */#if defined (STM32F427_437xx) || defined (STM32F429_439xx)
#define FMC_R_BASE            ((uint32_t)0xA0000000) /*!< FMC registers base address                                                 */
#endif /* STM32F427_437xx ||  STM32F429_439xx */#define CCMDATARAM_BB_BASE    ((uint32_t)0x12000000) /*!< CCM(core coupled memory) data RAM(64 KB) base address in the bit-band region  */
#define SRAM1_BB_BASE         ((uint32_t)0x22000000) /*!< SRAM1(112 KB) base address in the bit-band region                             */
#define SRAM2_BB_BASE         ((uint32_t)0x2201C000) /*!< SRAM2(16 KB) base address in the bit-band region                              */
#define SRAM3_BB_BASE         ((uint32_t)0x22400000) /*!< SRAM3(64 KB) base address in the bit-band region                              */
#define PERIPH_BB_BASE        ((uint32_t)0x42000000) /*!< Peripheral base address in the bit-band region                                */
#define BKPSRAM_BB_BASE       ((uint32_t)0x42024000) /*!< Backup SRAM(4 KB) base address in the bit-band region
/*!< Peripheral memory map */
#define APB1PERIPH_BASE       PERIPH_BASE
#define APB2PERIPH_BASE       (PERIPH_BASE + 0x00010000)
#define AHB1PERIPH_BASE       (PERIPH_BASE + 0x00020000)
#define AHB2PERIPH_BASE       (PERIPH_BASE + 0x10000000)/*!< APB1 peripherals */
#define TIM2_BASE             (APB1PERIPH_BASE + 0x0000)
#define TIM3_BASE             (APB1PERIPH_BASE + 0x0400)
#define TIM4_BASE             (APB1PERIPH_BASE + 0x0800)
#define TIM5_BASE             (APB1PERIPH_BASE + 0x0C00)
#define TIM6_BASE             (APB1PERIPH_BASE + 0x1000)
#define TIM7_BASE             (APB1PERIPH_BASE + 0x1400)
#define TIM12_BASE            (APB1PERIPH_BASE + 0x1800)
#define TIM13_BASE            (APB1PERIPH_BASE + 0x1C00)
#define TIM14_BASE            (APB1PERIPH_BASE + 0x2000)
#define RTC_BASE              (APB1PERIPH_BASE + 0x2800)
#define WWDG_BASE             (APB1PERIPH_BASE + 0x2C00)
#define IWDG_BASE             (APB1PERIPH_BASE + 0x3000)
#define I2S2ext_BASE          (APB1PERIPH_BASE + 0x3400)
#define SPI2_BASE             (APB1PERIPH_BASE + 0x3800)
#define SPI3_BASE             (APB1PERIPH_BASE + 0x3C00)
#define I2S3ext_BASE          (APB1PERIPH_BASE + 0x4000)

什么是寄存器映射?

在存儲器Block2這塊區域,設計的是片上外設,它們以4個字節為一個單元,共32bit,那么每一個單元對應不同的功能,當我們控制這些單元時就可以驅動外設工作。我們可以找到每個單元的起始地址,然后通過C語言指針的操作方式來訪問這些單元,如果每次都是通過這種地址的方式來訪問,不僅不好記憶還容易出錯,這是我們可以根據每個單元的功能不同,以功能為名給這個存儲單元取一個別名,這個別名就是我們經常說的寄存器,這個給已經分配好地址的有特定功能的內存單元取別名的過程就叫寄存器映射。

寄存器映射實現

第一步:宏定義GPIO 口的基地址,AHB1PERIPH_BASE 依次累加 0x400的地址偏移量,就得到GPIOA~GPIOK的基地址。

#define GPIOA_BASE            (AHB1PERIPH_BASE + 0x0000)
#define GPIOB_BASE            (AHB1PERIPH_BASE + 0x0400)
#define GPIOC_BASE            (AHB1PERIPH_BASE + 0x0800)
#define GPIOD_BASE            (AHB1PERIPH_BASE + 0x0C00)
#define GPIOE_BASE            (AHB1PERIPH_BASE + 0x1000)
#define GPIOF_BASE            (AHB1PERIPH_BASE + 0x1400)
#define GPIOG_BASE            (AHB1PERIPH_BASE + 0x1800)
#define GPIOH_BASE            (AHB1PERIPH_BASE + 0x1C00)
#define GPIOI_BASE            (AHB1PERIPH_BASE + 0x2000)
#define GPIOJ_BASE            (AHB1PERIPH_BASE + 0x2400)
#define GPIOK_BASE            (AHB1PERIPH_BASE + 0x2800)

第二步:把這個GPIO的基地址通過加上(GPIO_TypeDef *)這步騷操作,來把地址強轉成具有GPIO_TypeDef 性質的指針變量,并且用#define進行宏定義,實現取了個別名的效果。這就是寄存器的映射。

#define GPIOA               ((GPIO_TypeDef *) GPIOA_BASE)
#define GPIOB               ((GPIO_TypeDef *) GPIOB_BASE)
#define GPIOC               ((GPIO_TypeDef *) GPIOC_BASE)
#define GPIOD               ((GPIO_TypeDef *) GPIOD_BASE)
#define GPIOE               ((GPIO_TypeDef *) GPIOE_BASE)
#define GPIOF               ((GPIO_TypeDef *) GPIOF_BASE)
#define GPIOG               ((GPIO_TypeDef *) GPIOG_BASE)
#define GPIOH               ((GPIO_TypeDef *) GPIOH_BASE)
#define GPIOI               ((GPIO_TypeDef *) GPIOI_BASE)
#define GPIOJ               ((GPIO_TypeDef *) GPIOJ_BASE)
#define GPIOK               ((GPIO_TypeDef *) GPIOK_BASE)

GPIO_TypeDef結構體的聲明:

typedef struct
{__IO uint32_t MODER;    /*!< GPIO port mode register,               Address offset: 0x00      */__IO uint32_t OTYPER;   /*!< GPIO port output type register,        Address offset: 0x04      */__IO uint32_t OSPEEDR;  /*!< GPIO port output speed register,       Address offset: 0x08      */__IO uint32_t PUPDR;    /*!< GPIO port pull-up/pull-down register,  Address offset: 0x0C      */__IO uint32_t IDR;      /*!< GPIO port input data register,         Address offset: 0x10      */__IO uint32_t ODR;      /*!< GPIO port output data register,        Address offset: 0x14      */__IO uint16_t BSRRL;    /*!< GPIO port bit set/reset low register,  Address offset: 0x18      */__IO uint16_t BSRRH;    /*!< GPIO port bit set/reset high register, Address offset: 0x1A      */__IO uint32_t LCKR;     /*!< GPIO port configuration lock register, Address offset: 0x1C      */__IO uint32_t AFR[2];   /*!< GPIO alternate function registers,     Address offset: 0x20-0x24 */
} GPIO_TypeDef;

為什么要設置存儲器映射?

之所以這樣做,有以下幾個原因:

  • 統一訪問接口:CPU通過地址總線訪問內存和外設等不同部件時,采用統一的存儲器映射方式,就可以使用相同的指令和操作來進行數據的讀寫。例如,無論是訪問內部SRAM中的數據,還是向外設寄存器寫入控制命令,都可以通過對相應地址的操作來完成,無需為不同的部件設計不同的訪問指令和接口,簡化了硬件設計和軟件開發的復雜度。
  • 靈活的資源分配:存儲器映射可以根據不同的應用需求,靈活地將地址空間分配給各種不同的存儲介質和外設。例如,在設計一個具體的產品時,可以根據實際需要,將一部分地址空間分配給外部擴展的大容量Flash存儲器用于存儲程序和數據,將另一部分地址空間分配給特定的外設,如網絡控制器、SD卡控制器等,使系統能夠高效地利用各種資源,滿足不同的功能需求。
  • 方便系統擴展:當需要對系統進行擴展時,無論是增加新的存儲設備還是添加新的外設,都可以通過合理地分配存儲器映射地址來實現。新的設備可以很容易地集成到現有的系統中,只需將其映射到未使用的地址空間,并編寫相應的驅動程序來訪問這些地址即可,而無需對整個系統的架構進行大規模的修改。

stm32單片機地址總線訪問大小是4G,是不是意味著內存大小也是4G?

STM32單片機地址總線訪問大小是4G,但這并不意味著其內存大小就是4G。

地址總線的寬度決定了CPU可以訪問的地址空間范圍。STM32單片機基于ARM Cortex - M內核,其具有32位地址總線,所以理論上可訪問的地址空間為(2^{32})= 4G字節。然而,這4G的地址空間是包括了多個部分的,不僅僅是內存(RAM和ROM),還包括以下部分:

  • 片上外設寄存器空間:用于訪問各種片上外設,如GPIO、USART、SPI等外設的控制寄存器。每個外設都有其特定的地址范圍,通過地址總線來訪問這些寄存器以實現對外設的控制和數據傳輸。
  • 外部設備擴展空間:如果單片機擴展了外部的Flash、RAM、FPGA等設備,這些設備也會占用一定的地址空間,與片上資源共同構成整個可尋址的4G空間。
  • 系統保留區域:一些地址范圍可能被保留用于特定的系統功能或未來擴展,并不對應實際的物理存儲或外設。

實際上,STM32單片機內部的內存(如SRAM和Flash)容量通常遠小于4G。不同型號的STM32其內部SRAM一般在幾十KB到幾百KB之間,內部Flash存儲器一般在幾十KB到幾MB之間。
在這里插入圖片描述

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/web/73470.shtml
繁體地址,請注明出處:http://hk.pswp.cn/web/73470.shtml
英文地址,請注明出處:http://en.pswp.cn/web/73470.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

mapbox V3 新特性,添加三維球鷹眼圖控件

????? 主頁: gis分享者 ????? 感謝各位大佬 點贊?? 收藏? 留言?? 加關注?! ????? 收錄于專欄:mapbox 從入門到精通 文章目錄 一、??前言1.1 ??mapboxgl.Map 地圖對象1.2 ??mapboxgl.Map style屬性1.3 ??mapbox-gl-globe-minimap 三維球體鷹眼…

MySQL-調優策略-SQL語句

引言 架構調優&#xff0c;在系統設計時首先需要充分考慮業務的實際情況&#xff0c;是否可以把不適合數據庫做的事情放到數據倉庫、搜索引擎或者緩存中去做&#xff1b;然后考慮寫的并發量有多大&#xff0c;是否需要采用分布式&#xff1b;最后考慮讀的壓力是否很大&#xf…

6502電氣集中聯鎖道岔控制電路的工作過程

6502電氣集中聯鎖道岔控制電路的工作過程主要包括選擇進路、轉換道岔、鎖閉進路、開放信號和解鎖進路等環節&#xff0c;以下是其具體工作過程模擬&#xff1a; 選擇進路&#xff1a; 按壓按鈕&#xff1a;操作人員在控制臺上按壓進路兩端的按鈕&#xff0c;如始端按鈕和終端按…

DS足球監控【比分直播】監控,釘釘實現自動提醒

文章目錄 目標網站分析詳細分析提醒工具代碼截圖成功提示對爬蟲、逆向感興趣的同學可以查看文章,一對一小班教學:https://blog.csdn.net/weixin_35770067/article/details/142514698 目標網站分析 https://live.dszuqiu.com/監控目標:實現固定時間內對比分監控,實現自動下單…

基于ssm的醫院預約掛號系統

一、系統架構 前端&#xff1a;jsp | bootstrap | jquery | css | ajax 后端&#xff1a;spring | springmvc | mybatis 環境&#xff1a;jdk1.8 | mysql | maven | tomcat 二、代碼及數據 三、功能介紹 01. 注冊 02. 登錄 03. 首頁 04. 醫院掛號 05. …

華為OD機試A卷 - 快遞業務站 計算快遞主站點(C++ Java JavaScript Python )

最新華為OD機試 真題目錄:點擊查看目錄 華為OD面試真題精選:點擊立即查看 題目描述 快遞業務范圍有 N 個站點,A 站點與 B 站點可以中轉快遞,則認為 A-B 站可達, 如果 A-B 可達,B-C 可達,則 A-C 可達。 現在給 N 個站點編號 0、1、…n-1,用 s[i][j]表示 i-j 是否可…

三維動態規劃-LeetCode3418. 機器人可以獲得的最大金幣數

太爽了&#xff01;做完這道題&#xff0c;讓我感覺就像是斬殺了一條大龍&#xff01;歷時72天&#xff0c;分3次花掉30小時。終獲突破&#xff01; 零、題目 3418. 機器人可以獲得的最大金幣數 給你一個 m x n 的網格。一個機器人從網格的左上角 (0, 0) 出發&#xff0c;目…

相生、相克、乘侮、復雜病機及對應的臟腑功能聯系

一、五行相生關系&#xff08;母子關系&#xff09; 五行生序臟腑關系生理表現舉例木生火肝&#xff08;木&#xff09;滋養心&#xff08;火&#xff09;肝血充足則心血旺盛火生土心&#xff08;火&#xff09;溫煦脾&#xff08;土&#xff09;心陽充足則脾胃運化功能正常土…

Ubuntu22.04搭建freeradius操作說明

Ubuntu22.04搭建freeradius操作說明 更新依賴庫 sudo apt update sudo apt install build-essential sudo apt install libtalloc-dev sudo apt install libssl-dev 按照freeradius sudo apt install freeradius 修改freeradius配置 文件路徑如下 /etc/freeradius/3.…

es中安裝ik分詞器

在線安裝ik插件&#xff08;較慢&#xff09; docker exec -it es /bin/bash ./bin/es-plugin install https://github.com/medcl/elasticsearch-analysis-ik/releases/download/v7.12.1/elasticsearch-analysis-ik-7.12.1.zip 看到報錯了&#xff0c;我訪問一下。就是沒有了…

最大字段和問題 C++(窮舉、分治法、動態規劃)

問題描述 給定由n個整數&#xff08;包含負整數&#xff09;組成的序列a1,a2,…,an&#xff0c;求該序列子段和的最大值。規定當所有整數均為負值時定義其最大子段和為0 窮舉法 最簡單的方法就是窮舉法&#xff0c;用一個變量指示求和的開始位置&#xff0c;一個變量指示結束…

如何理解三極管截至區、放大區、飽和區

一、 三極管符號&#xff1a; NPN : PNP: 二、Vce、與Ic曲線圖 1、截至區&#xff1a;ib很小的時候就是截至區。因為Ib很小的時候等價于Ub很小&#xff0c;Ub如果不足以達到0.7V PN結就不會導通&#xff0c;所以三極管就…

電腦上我的windows目錄下,什么是可以刪除的

在Windows系統目錄&#xff08;通常是C:\Windows&#xff09;中&#xff0c;大部分文件和文件夾都是系統運行所必需的&#xff0c;隨意刪除可能導致系統崩潰或程序無法運行。不過&#xff0c;部分文件可以安全清理。以下是詳細指南&#xff1a; 可安全清理的內容 臨時文件&…

工作中遇到的spark SQL小問題:包含某個或某些字符的條件

今天又來總結工作中遇到的問題了&#xff0c;今天是SQL&#xff0c;spark引擎 需求描述&#xff0c;篩選渠道包含”線上化“的數據 也就是討論where里面的這個篩選條件怎么寫 一般起手都是 where QD like %線上化%‘ 學習了其他的寫法: 1.INSTR函數 where INSTR(QD,&quo…

Git 命令操作完全指南

Git 是現代軟件開發中不可或缺的分布式版本控制系統。它不僅能追蹤代碼變更&#xff0c;還能協調多人協作、管理項目歷史。本文從核心概念入手&#xff0c;逐步深入講解 Git 的基礎與高級命令&#xff0c;結合實用場景&#xff0c;幫助您從入門到精通。 一、Git 核心概念 理解…

深入剖析帶頭循環雙向鏈表的實現與應用

引言 場景描述 想象一個 環形地鐵線路&#xff08;如深圳地鐵11號線&#xff09;&#xff0c;這條線路首尾相連&#xff0c;列車可以順時針或逆時針循環行駛。為了方便管理&#xff0c;地鐵系統設置了一個 “虛擬調度中心”&#xff08;頭節點&#xff09;&#xff0c;它不承…

DeepSeek Smallpond 在火山引擎 AI 數據湖的探索實踐

資料來源&#xff1a;火山引擎-開發者社區 DeepSeek Smallpond 介紹 Smallpond 是一套由 DeepSeek 推出的 、針對 AI 領域&#xff0c;基于 Ray 和 DuckDB 實現的輕量級數據處理引擎&#xff0c;具有以下優點&#xff1a; 1.輕量級 2.高性能 3.支持規模大 4.無需運維 5.P…

Linux進程間的通信

進程間通信 1.進程間通信介紹2.匿名命名管道原理操作 1.進程間通信介紹 1.1 進程間通信目的&#xff1a;一個進程需要將他的數據發送給另一個進程&#xff0c;大家應該都多少接觸過linux中的管道符"|"&#xff0c;這個符號就是用來多個命令執行&#xff0c;在Linux中…

直播預告 | TDgpt 智能體發布 時序數據庫 TDengine 3.3.6 發布會即將開啟

從海量監控數據&#xff0c;到工業、能源、交通等場景中實時更新的各類傳感器數據&#xff0c;時序數據正在以指數級速度增長。而面對如此龐雜的數據&#xff0c;如何快速分析、自動發現問題、精準預測未來&#xff0c;成為企業數字化轉型過程中的關鍵挑戰。 TDengine 的答案是…

手撕FIO工具指南:從壓測翻車到避坑實戰

文章目錄 手撕FIO工具指南&#xff1a;從壓測翻車到避坑實戰一、背景&#xff1a;一次FIO壓測引發的驚魂夜二、FIO vs 其他IO工具&#xff1a;為何讓人又愛又怕&#xff1f;三、安裝指南&#xff1a;避開依賴地獄四、參數詳解五、避坑指南&#xff1a;血淚經驗總結六、安全壓測…