在單片機芯片上,如果不考慮出廠固化的ROM空間的話,通常開發者能接觸到的存儲空間主要分兩種:掉電可保存數據的片內FLASH和掉電不可保存數據的片內RAM。
片內RAM(通常理解為內存)的訪問速度比較快,可以按照變量地址隨機訪問,但斷電后數據丟失。片內FLASH(通常理解為硬盤)所保存的內容比較固定,主要用來保存程序本身的數據內容,保存的內容斷電不丟失。
對于單片機的片內RAM內存,主要有堆和棧之分,本章的內存管理,主要是基于堆內存管理進行開展的,在RT-Thread中,有兩種堆內存管理方式:動態內存堆管理和靜態內存池管理。
本文嘗試從以下幾個方面總結一下RT-Thread內存管理的學習過程
內存管理相關介紹
在運行操作系統的單片機上面,代碼和變量會占用一部分固定的內存開銷,操作系統在初始化的時候,會去除掉這部分已經占用的內存,把剩下的閑置內存納入到系統堆里面進行統一管理,不管是動態堆內存,還是靜態內存池,都是使用這部分閑置空間的。
由于在實時操作系統里面對時間的要求十分嚴格,為了保證內存分配的時候不影響系統的實時性,就需要確保分配內存的時間是確定并可控的;并且在內存分配達到一定次數后,就不可避免地產生內存碎片;與此同時,嵌入式設備的內存資源相對有限,有些系統只有幾十KB內存,而有些系統則有幾十MB。
所以,為了解決以上內存分配可能出現的問題,需要使用一些內存管理算法來進行這些內存分配管理,RT-Thread提供了兩種內存管理方式,分別是:動態內存堆管理和靜態內存池管理。
動態內存堆管理
內存堆管理分配主要用于系統動態分配內存的場合,比如,我們使用動態方式創建某些內核對象(如消息隊列,郵箱,信號量,等等)的時候,所使用到的內存空間就是動態內存堆。動態內存堆的意思是,要用多少,系統就分配多少給你,不用的時候,就要進行釋放,還給系統再進行統一管理。
關于動態內存堆的管理,主要有三種算法:小內存分配算法,slab算法,memheap算法。關于這三種管理算法的實現原理介紹,RT-Thread官方已經給出了比較詳細的解釋,這里不再重復論述。
需要注意的是,這三種內存管理算法,我們只能通過menuconfig來配置系統內核,選擇其中一種內存管理方法,對于用戶的應用程序接口而言,這三種算法是透明的,也就是說提供給用戶的內存管理接口是相同的,只是算法的實現原理不同。
關于動態堆內存管理,操作系統提供了以下API接口函數,如下圖所示。
靜態內存池管理
在使用動態內存堆管理系統內存的時候,這種方式非常靈活和方便,想用內存的時候就向系統申請分配,不用的時候就釋放還給系統,但這種方式也存在一定的弊端。
主要是向系統申請內存的時候,都要遍歷一次空閑內存的鏈表,查找可用的內存塊,然后再分配給用戶,而且這種方式不可避免地會產生內存碎片,所以這種內存管理方式的效率不是很高。這是一種“用時間換空間”的內存管理方式。
為了提高內存的分配效率,RT-Thread提供了靜態內存池管理的方式。靜態內存池就是系統把自身管理的內存預先劃分為多個固定大小的內存塊,當用戶需要申請內存的時候,就從這些固定大小的內存塊里面申請。
靜態內存池管理的方式,還支持線程掛起操作,當系統沒有內存塊可用時,線程就會掛起等待,直到能申請到可用的內存塊,這種特性可以用做線程間同步。
關于靜態內存池的工作機制,如下圖所示。
RT-Thread提供了以下API函數接口,用于靜態內存池管理。
內存堆和內存池的應用示例
內存管理相關的應用示例,主要是為了驗證動態內存堆管理和靜態內存池管理相關的API函數接口,這里包含兩個示例,分別是內存堆管理示例和內存池管理示例。
示例源碼下載鏈接:
https://github.com/embediot/rtthread_study_notes
https://gitee.com/embediot/rtthread_study_notes
內存堆管理示例會創建一個動態的線程,這個線程會動態申請內存并釋放,每次申請更大的內存,當申請不到的時候就結束。例程中分配內存成功并打印信息;當試圖申請 65536 byte 即 64KB 內存時,由于開發板的單片機 RAM 總大小只有 64K,而可用 RAM 小于 64K,所以分配失敗。
內存池管理示例會創建一個靜態的內存池對象,2 個動態線程。一個線程會試圖從內存池中獲得內存塊,另一個線程釋放內存塊內存塊。總共初始化了 4096 /(80+4) = 48 個內存塊。
1、線程 1 申請了 48 個內存塊之后,此時內存塊已經被用完,需要其他地方釋放才能再次申請;但此時,線程 1 以一直等待的方式又申請了 1 個,由于無法分配,所以線程 1 掛起;
2、線程 2 開始執行釋放內存的操作;當線程 2 釋放一個內存塊的時候,就有一個內存塊空閑出來,喚醒線程 1 申請內存,申請成功后再申請,線程 1 又掛起,再循環一次步驟2;
3、線程 2 繼續釋放剩余的內存塊,釋放完畢。
在memory_test.h頭文件里面,通過打開相應的宏定義開關,重新編譯工程源碼,下載到開發板即可驗證實驗現象,如下圖所示。
內存管理相關注意事項
在使用RT-Thread內存管理相關接口的時候,為了確保系統穩定性,有以下注意事項:
1、由于系統為了保證內存在多線程的狀態下能安全分配,引入了互斥操作,因此不能在中斷服務程序里面分配或釋放內存塊,否則會引起當前線程被掛起。
2、在使用內存堆管理的時候,產生的內存碎片會在系統空閑線程運行的時候進行回收。
3、用戶應用程序在申請內存分配的時候,建議判斷是否申請成功,并對申請成功的內存空間進行初始化后再使用。
4、動態內存堆管理是一種“用時間換空間”的內存管理方式,這種方式可以節省一定的內存空間,但會損失一點效率。
5、靜態內存池管理是一種“用空間換時間”的內存管理方式,這種方式相對來說比較高效,但會造成一定的空間浪費。
6、對于以KB為單位的單片機片內RAM內存,一般采用動態內存堆里面的小內存管理算法即可。
原文標題:【RT-Thread學習筆記】RT-Thread內存管理學習總結
文章出處:【微信公眾號:RTThread物聯網操作系統】歡迎添加關注!文章轉載請注明出處。
責任編輯:haq
-
內存
+關注
關注
8文章
3040瀏覽量
74176 -
RT-Thread
+關注
關注
31文章
1300瀏覽量
40264 -
RTThread
+關注
關注
8文章
132瀏覽量
40924
原文標題:【RT-Thread學習筆記】RT-Thread內存管理學習總結
文章出處:【微信號:RTThread,微信公眾號:RTThread物聯網操作系統】歡迎添加關注!文章轉載請注明出處。
發布評論請先 登錄
相關推薦
評論