關于堆和棧已經是程序員的一個月經話題,大部分有是基于os層來聊的。
那么,在赤裸裸的單片機下的堆和棧是什么樣的分布呢?以下是網摘:
剛接手STM32時,你只編寫一個
int main()
{
while(1);
}
BUILD://Program Size: Code=340 RO-data=252 RW-data=0 ZI-data=1632
編譯后,就會發現這么個程序已用了1600多的RAM,要是在51單片機上,會心疼死了,這1600多的RAM跑哪兒去了,
分析map,你會發現是堆和棧占用的,在startup_stm32f10x_md.s文件(這個是stm32的啟動文件)中,它的前面幾行就有以上定義,
這下該明白了吧。
Stack_Size EQU 0x00000400
Heap_Size EQU 0x00000200
以下引用網上資料 理解堆和棧的區別
(1)棧區(stack):由編譯器自動分配和釋放,存放函數的參數值、局部變量的值等,其操作方式類似于數據結構中的棧。
(2)堆區(heap):一般由程序員分配和釋放,若程序員不釋放,程序結束時可能由操作系統回收。分配方式類似于數據結構中的鏈表。
(3)全局區(靜態區)(static):全局變量和靜態變量的存儲是放在一塊的,初始化的全局變量和靜態變量在一塊區域,未初始化的全局變量和未初始化的靜態變量在相鄰的另一塊區域。程序結束后由系統自動釋放。
(4)文字常量區:常量字符串就是存放在這里的。
(5)程序代碼區:存放函數體的二進制代碼。
例如:
int a=0; //全局初始化區
char *p1; //全局未初始化區
main()
{
int b; //棧
char s[]="abc"; //棧
char *p3= "1234567"; //在文字常量區Flash
static int c =0 ; //靜態初始化區
p1= (char *)malloc(10); //堆區
strcpy(p1,"123456"); //"123456"放在常量區
}
所以堆和棧的區別:
stack的空間由操作系統自動分配/釋放,heap上的空間手動分配/釋放。
stack的空間有限,heap是很大的自由存儲區(heap雖然有很大的存儲區,但是這個存儲區并不是無限大的,在stm32中,heap區的最大值由SRAM區決定,而SRAM區的大小可以參考具體的數據手冊)。
程序在編譯期和函數分配內存都是在棧上進行,且程序運行中函數調用時參數的傳遞也是在棧上進行。
------------------------------------------------------------------------------------------------------
1.堆和棧大小
定義大小在startup_stm32f2xx.s (這個地方應該是有錯,定義的大小是在startup_stm32f10x_hd.s,也就是啟動文件中)
Stack_Size EQU 0x00000400
AREA STACK, NOINIT, READWRITE, ALIGN=3
Stack_Mem SPACE Stack_Size
__initial_sp
; Heap Configuration
; Heap Size (in Bytes) <0x0-0xFFFFFFFF:8>
;
Heap_Size EQU 0x00000200 //這里就是分配的堆空間大小
AREA HEAP, NOINIT, READWRITE, ALIGN=3
__heap_base
2.堆和棧位置
通過MAP文件可知
HEAP 0x200106f8 Section 512 startup_stm32f2xx.o(HEAP)
STACK 0x200108f8 Section 1024 startup_stm32f2xx.o(STACK)
__heap_base 0x200106f8 Data 0 startup_stm32f2xx.o(HEAP)
__heap_limit 0x200108f8 Data 0 startup_stm32f2xx.o(HEAP)
__initial_sp 0x20010cf8 Data 0 startup_stm32f2xx.o(STACK)
顯然 Cortex-m3資料可知:__initial_sp是堆棧指針,它就是FLASH的0x8000000地址前面4個字節(它根據堆棧大小,由編譯器自動生成)
顯然堆和棧是相鄰的。
3.堆和棧空間分配
棧:向低地址擴展
堆:向高地址擴展
顯然如果依次定義變量,先定義的棧變量的內存地址比后定義的棧變量的內存地址要大先定義的堆變量的內存地址比后定義的堆變量的內存地址要小
4.堆和棧變量
棧:臨時變量,退出該作用域就會自動釋放
堆:malloc變量,通過free函數釋放
另外:堆棧溢出,編譯不會提示,需要注意
------------------------------------------------------------------------------------------------------
如果使用了HEAP,則必須設置HEAP大小。
如果是STACK,可以設置為0,不影響程序運行。
IAR STM8定義STACK,是預先在RAM尾端分配一個字節的區域作為堆棧預留區域。
當程序靜態變量,全局變量,或者堆與預留堆棧區域有沖突,編譯器連接的時候就會報錯。
你可以吧STACK設置為0,并不影響運行。(會影響調試,調試會報堆棧溢出警告)。
其實沒必要這么做。
一般程序,(在允許范圍內)設置多少STACK,并不影響程序真實使用的RAM大小,(可以試驗,把STACK設置多少,編譯出來的HEX文件都是一樣), 程序還是按照它原本的狀態使用RAM,把STACK設置為0,并不是真實地減少RAM使用。
僅僅是欺騙一下編譯器,讓程序表面上看起來少用了RAM。
而設置一定size的STACK,也并不是真的就多使用了RAM,只是讓編譯器幫你檢查一下,是否能夠保證有size大小的RAM沒有被占用,可以用來作為堆棧。
以上僅針對IAR STM8.
------------------------------------------------------------------------------------------------------
從以上網摘來看單片機的堆和棧是分配在RAM里的,有可能是內部也有可能是外部,可以讀寫;(堆在stm32是分配在SRAM中的)
棧:存函數的臨時變量,即局部變量,函數返回時隨時有可能被其他函數棧用。所以棧是一種分時輪流使用的存儲區,編譯器里定義的Stack_Size,是為了限定函數的局部數據活動的范圍,操過這么范圍有可以跑飛,也就是棧溢出;
Stack_Size不影響Hex,更不影響Hex怎么運行的,只是在Debug調試時會提示錯。棧溢出也有是超過了國界進行活動,只要老外沒有意見,你可以接著玩,有老外不讓你玩,你就的得死,或是大家都死(互相撕殺),有的人寫單片機代碼在函數里定義一個大數組 int buf[8192],棧要是小于8192是會死的很慘。
堆:存的是全局變量,這變量理論上是所有函數都可以訪問的,全局變量有的有初始值,但這個值不是存在RAM里的,是存在Hex里,下載到Flash里,上電由代碼(編譯器生成的匯編代碼)搬過去的。有的人很“霸道”,上電就霸占已一塊很大的RAM(Heap_Size),作為己有(malloc_init),別人用只能通過他們管家借(malloc),用完還得換(free)。所以
一旦有“霸道”的人出現是編譯器里必須定義Heap_Size,否則和他管家借也沒有用。
總之:堆和棧有存在RAM里,他兩各分多少看函數需求,但是他兩的總值不能超過單片機硬件的實際RAM尺寸,否則只能到海里玩(淹死了)或是自己打造船接著玩(外擴RAM)。
參考資料:
1 百度文庫《stm32堆棧分析》
http://wenku.baidu.com/link?url=Fk8ht47IULQFtxV43tc36LMSqDaGihhImEiVWmch-3nqRwRv2Arg9HUxU-JtQy8E64Kfiuhn0QghFpFQBFV9jdTRqpmMoaowX2DiMTGHC7u
上一篇:STM32堆和棧(Heap & Stack)及SRAM存儲使用
下一篇:淺談STM32堆棧與uCOS堆棧,其實是兩碼事
推薦閱讀
史海拾趣
奧松電子擁有一支近200名工程師組成的專職研發團隊,并配備了超過7000㎡的研發實驗室。實驗室中配置了步進式投影光刻機、雙面光刻機等先進設備,為公司的產品研發提供了有力的支持。這些設備不僅滿足了產品研發、小試以及中試各個階段的試驗條件,也為公司的技術創新提供了堅實的基礎。
奧松電子在傳感器領域取得了顯著的突破。公司憑借其強大的研發實力和技術創新能力,成功開發出一系列具有自主知識產權的傳感器產品。這些傳感器產品不僅性能穩定、測量精度高,而且具有廣泛的應用領域。在醫療、軌道交通、智能家電等多個領域,奧松電子的傳感器產品都得到了廣泛的應用和認可。這些產品的成功應用,不僅提升了公司的市場競爭力,也為公司的未來發展提供了更多的可能性。
這五個故事展示了廣州奧松電子股份有限公司在電子行業中發展起來的歷程和成就。通過不斷的技術創新和市場拓展,奧松電子已經成功成為了國內電子行業中的佼佼者。未來,隨著技術的不斷進步和市場需求的不斷變化,奧松電子將繼續保持其領先地位,為電子行業的發展做出更大的貢獻。
作為一家具有高度責任感的企業,DiCon長期以來一直致力于為國防工業提供高質量的光纖通信產品。公司的首批產品之一FDDI交換機在許多海軍艦艇上部署了30多年,為國家的安全穩定做出了重要貢獻。這一里程碑事件不僅彰顯了DiCon在光纖通信技術領域的實力,也進一步提升了公司的品牌形象和市場地位。
歌普電子不僅注重產品的品質和技術創新,還非常重視客戶服務。公司始終秉承“用心服務”的理念,為客戶提供全方位的售前、售中和售后服務。無論是產品咨詢、技術支持還是售后保障,歌普都能迅速響應并滿足客戶的需求。這種高效、專業的服務贏得了客戶的高度評價和信賴,也為公司的持續發展奠定了堅實的基礎。
Bridgelux在LED照明技術方面取得了重要突破。公司研發出了一種新型的LED芯片結構,提高了光效和穩定性,降低了成本。這一創新使得Bridgelux的產品在市場上更具競爭力,也為其贏得了眾多客戶的青睞。同時,公司還注重知識產權保護,積極申請專利,確保自身技術的領先地位。
在1948年,當美國民眾開始大量購買電視機時,電視機的維修需求也隨之增加。芝加哥的企業家卡爾·科恩(Carl Korn)和菲利普·班(Philip Ban)注意到了這一市場機會。他們發現,當時缺乏能夠輕松測試電視組件的設備,于是決定自己動手制作。兩位創業者從車庫起步,開始制造自己的測試設備,并成立了中央電視服務公司。他們的業務迅速擴展,開始向其他電視維修店銷售CRT再生器和真空管測試儀,為公司的后續發展奠定了堅實的基礎。
本帖最后由 paulhyde 于 2014-9-15 09:48 編輯 活動介紹 恩智浦半導體(NXP Semiconductors)是一家新近獨立的半導體公司,由飛利浦公司創立,已擁有五十年的悠久歷史,主要提供工程師與設計人員各種半導體產品與軟件,為消費電子、安全應用、 ...… 查看全部問答∨ |
|
不知各位遇到過TI C2000 Piccolo Microcontrollers不能燒寫FLASH的問題沒? 小弟遇到的問題如下: 軟件配置:CCS3.3.82.13+F2802x_CSP_RTM.exe+FlashAPIInterface2802xV1_00a.out+F28027.CMD 硬件配置:TDS510仿真器+F28027最小系統板 問題 ...… 查看全部問答∨ |
|
6集成電路運算放大器 6 .1.1集成電路運算放大器的特點 6.1.2集成運算放大電路中的電流源 6.1.3基本型差動放大器 6.1.4雙電源長尾式差放 6.1.5差動電路的幾種接法 6.1.6恒流源式差放… 查看全部問答∨ |
本帖最后由 paulhyde 于 2014-9-15 09:18 編輯 衰減器的功能是能處理大功率信號且不會影響信號性能,所以在對大功率信號處理和測試時需要用到高頻衰減器。尤其是一些特殊的信號,例如在商業廣播和軍用雷達系統中常見的超過1kW的信號來講,在任何測 ...… 查看全部問答∨ |
中嵌學院的嵌入式培訓怎么樣?保證就業是真的嗎?就業怎么樣?有在中嵌培訓過的大哥大姐能回答下嗎?學費太貴,農村的我想把來之不易的錢用到實處, 有知道的說下… 查看全部問答∨ |
該平臺可以管理車輛控制系統、導航儀和音響等娛樂設備,并能監控車輛運行及環境信息。通過合理使用STR750的各種通信接口,滿足車載控制系統的功能要求。要求按照實際需要設計出一個功能豐富的通用控制平臺,進行軟硬件系統設計,實現相應的功能。… 查看全部問答∨ |
程序在窗口輸入的情況下,一切正常。 但當把數據編入軟件,不用輸入,直接使用時,會提示錯誤。 下面這段程序,問題在那里? CByteArray m_1; m_1.SetSize(8); m_1[0]=0x01; m_1[1]=0x03; m_1[2]=0x00; m_1[3]=0x08; m_1[4]=0x00; m_1[5] ...… 查看全部問答∨ |
大家都知道,LED燈采用半導體原理,其光源的壽命是很長的,可達50000h以上,然而最近和節能燈行業一位前輩交流,他卻稱,這種說法是不實之言,他是這樣給我分析的: LED發光原理普通節能一樣,都是利用稀土三基色熒光粉發光,只是激發方式不同。 ...… 查看全部問答∨ |