當(dāng)前位置:首頁 > 單片機 > 單片機
[導(dǎo)讀]原文地址:http://blog.csdn.net/slj_win/article/details/16906141文章排版不是很好,但是寫的還是很有道理的。關(guān)于堆和棧已經(jīng)是程序員的一個月經(jīng)話題,大部分有是基于os層來聊的。那么,在赤裸裸的單片機下的堆和棧

原文地址:http://blog.csdn.net/slj_win/article/details/16906141

文章排版不是很好,但是寫的還是很有道理的。


關(guān)于堆和棧已經(jīng)是程序員的一個月經(jīng)話題,大部分有是基于os層來聊的。


那么,在赤裸裸的單片機下的堆和棧是什么樣的分布呢?以下是網(wǎng)摘:


intmain()

{

while(1);

}

BUILD://Program Size: Code=340 RO-data=252 RW-data=0 ZI-data=1632


編譯后,就會發(fā)現(xiàn)這么個程序已用了1600多的RAM,要是在51單片機上,會心疼死了,這1600多的RAM跑哪兒去了,


分析map,你會發(fā)現(xiàn)是堆和棧占用的,在startup_stm32f10x_md.s文件(這個是stm32的啟動文件)中,它的前面幾行就有以上定義,


這下該明白了吧。


Stack_Size EQU 0x00000400


Heap_Size EQU 0x00000200


以下引用網(wǎng)上資料 理解堆和棧的區(qū)別


(1)棧區(qū)(stack):由編譯器自動分配和釋放,存放函數(shù)的參數(shù)值、局部變量的值等,其操作方式類似于數(shù)據(jù)結(jié)構(gòu)中的棧。


(2)堆區(qū)(heap):一般由程序員分配和釋放,若程序員不釋放,程序結(jié)束時可能由操作系統(tǒng)回收。分配方式類似于數(shù)據(jù)結(jié)構(gòu)中的鏈表。


(3)全局區(qū)(靜態(tài)區(qū))(static):全局變量和靜態(tài)變量的存儲是放在一塊的,初始化的全局變量和靜態(tài)變量在一塊區(qū)域,未初始化的全局變量和未初始化的靜態(tài)變量在相鄰的另一塊區(qū)域。程序結(jié)束后由系統(tǒng)自動釋放。


(4)文字常量區(qū):常量字符串就是存放在這里的。


(5)程序代碼區(qū):存放函數(shù)體的二進制代碼。



例如:

inta=0;//全局初始化區(qū)

char*p1;//全局未初始化區(qū)

main()

{

intb;//棧

chars[]="abc";//棧

char*p3="1234567";//在文字常量區(qū)Flash

staticintc=0;//靜態(tài)初始化區(qū)

p1=(char*)malloc(10);//堆區(qū)

strcpy(p1,"123456");//"123456"放在常量區(qū)

}


所以堆和棧的區(qū)別:


stack的空間由操作系統(tǒng)自動分配/釋放,heap上的空間手動分配/釋放。


stack的空間有限,heap是很大的自由存儲區(qū)(heap雖然有很大的存儲區(qū),但是這個存儲區(qū)并不是無限大的,在stm32中,heap區(qū)的最大值由SRAM區(qū)決定,而SRAM區(qū)的大小可以參考具體的數(shù)據(jù)手冊)。


程序在編譯期和函數(shù)分配內(nèi)存都是在棧上進行,且程序運行中函數(shù)調(diào)用時參數(shù)的傳遞也是在棧上進行。


------------------------------------------------------------------------------------------------------


1.堆和棧大小


定義大小在startup_stm32f2xx.s (這個地方應(yīng)該是有錯,定義的大小是在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個字節(jié)(它根據(jù)堆棧大小,由編譯器自動生成)


顯然堆和棧是相鄰的。




3.堆和??臻g分配


棧:向低地址擴展


堆:向高地址擴展


顯然如果依次定義變量,先定義的棧變量的內(nèi)存地址比后定義的棧變量的內(nèi)存地址要大先定義的堆變量的內(nèi)存地址比后定義的堆變量的內(nèi)存地址要小


4.堆和棧變量


棧:臨時變量,退出該作用域就會自動釋放


堆:malloc變量,通過free函數(shù)釋放


另外:堆棧溢出,編譯不會提示,需要注意





------------------------------------------------------------------------------------------------------





如果使用了HEAP,則必須設(shè)置HEAP大小。
如果是STACK,可以設(shè)置為0,不影響程序運行。
IAR STM8定義STACK,是預(yù)先在RAM尾端分配一個字節(jié)的區(qū)域作為堆棧預(yù)留區(qū)域。
當(dāng)程序靜態(tài)變量,全局變量,或者堆與預(yù)留堆棧區(qū)域有沖突,編譯器連接的時候就會報錯。
你可以吧STACK設(shè)置為0,并不影響運行。(會影響調(diào)試,調(diào)試會報堆棧溢出警告)。
其實沒必要這么做。
一般程序,(在允許范圍內(nèi))設(shè)置多少STACK,并不影響程序真實使用的RAM大小,
(可以試驗,把STACK設(shè)置多少,編譯出來的HEX文件都是一樣),
程序還是按照它原本的狀態(tài)使用RAM,把STACK設(shè)置為0,并不是真實地減少RAM使用。
僅僅是欺騙一下編譯器,讓程序表面上看起來少用了RAM。
而設(shè)置一定size的STACK,也并不是真的就多使用了RAM,只是讓編譯器幫你
檢查一下,是否能夠保證有size大小的RAM沒有被占用,可以用來作為堆棧。
以上僅針對IAR STM8.





------------------------------------------------------------------------------------------------------





從以上網(wǎng)摘來看單片機的堆和棧是分配在RAM里的,有可能是內(nèi)部也有可能是外部,可以讀寫;(堆在stm32是分配在SRAM中的)





棧:存函數(shù)的臨時變量,即局部變量,函數(shù)返回時隨時有可能被其他函數(shù)棧用。所以棧是一種分時輪流使用的存儲區(qū),


編譯器里定義的Stack_Size,是為了限定函數(shù)的局部數(shù)據(jù)活動的范圍,操過這么范圍有可以跑飛,也就是棧溢出;


Stack_Size不影響Hex,更不影響Hex怎么運行的,只是在Debug調(diào)試時會提示錯。棧溢出也有是超過了國界進行


活動,只要老外沒有意見,你可以接著玩,有老外不讓你玩,你就的得死,或是大家都死(互相撕殺),有的人寫


單片機代碼在函數(shù)里定義一個大數(shù)組 int buf[8192],棧要是小于8192是會死的很慘。





堆:存的是全局變量,這變量理論上是所有函數(shù)都可以訪問的,全局變量有的有初始值,但這個值不是存在RAM里的,是


存在Hex里,下載到Flash里,上電由代碼(編譯器生成的匯編代碼)搬過去的。有的人很“霸道”,上電就霸占已一塊很


大的RAM(Heap_Size),作為己有(malloc_init),別人用只能通過他們管家借(malloc),用完還得換(free)。所以


一旦有“霸道”的人出現(xiàn)是編譯器里必須定義Heap_Size,否則和他管家借也沒有用。





總之:堆和棧有存在RAM里,他兩各分多少看函數(shù)需求,但是他兩的總值不能超過單片機硬件的實際RAM尺寸,否則只能


到海里玩(淹死了)或是自己打造船接著玩(外擴RAM)。


參考資料:

1 百度文庫《stm32堆棧分析》

http://wenku.baidu.com/link?url=Fk8ht47IULQFtxV43tc36LMSqDaGihhImEiVWmch-3nqRwRv2Arg9HUxU-JtQy8E64Kfiuhn0QghFpFQBFV9jdTRqpmMoaowX2DiMTGHC7u


本站聲明: 本文章由作者或相關(guān)機構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點,本站亦不保證或承諾內(nèi)容真實性等。需要轉(zhuǎn)載請聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請及時聯(lián)系本站刪除。
換一批
延伸閱讀

9月2日消息,不造車的華為或?qū)⒋呱龈蟮莫毥谦F公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關(guān)鍵字: 阿維塔 塞力斯 華為

加利福尼亞州圣克拉拉縣2024年8月30日 /美通社/ -- 數(shù)字化轉(zhuǎn)型技術(shù)解決方案公司Trianz今天宣布,該公司與Amazon Web Services (AWS)簽訂了...

關(guān)鍵字: AWS AN BSP 數(shù)字化

倫敦2024年8月29日 /美通社/ -- 英國汽車技術(shù)公司SODA.Auto推出其旗艦產(chǎn)品SODA V,這是全球首款涵蓋汽車工程師從創(chuàng)意到認(rèn)證的所有需求的工具,可用于創(chuàng)建軟件定義汽車。 SODA V工具的開發(fā)耗時1.5...

關(guān)鍵字: 汽車 人工智能 智能驅(qū)動 BSP

北京2024年8月28日 /美通社/ -- 越來越多用戶希望企業(yè)業(yè)務(wù)能7×24不間斷運行,同時企業(yè)卻面臨越來越多業(yè)務(wù)中斷的風(fēng)險,如企業(yè)系統(tǒng)復(fù)雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務(wù)連續(xù)性,提升韌性,成...

關(guān)鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據(jù)媒體報道,騰訊和網(wǎng)易近期正在縮減他們對日本游戲市場的投資。

關(guān)鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會開幕式在貴陽舉行,華為董事、質(zhì)量流程IT總裁陶景文發(fā)表了演講。

關(guān)鍵字: 華為 12nm EDA 半導(dǎo)體

8月28日消息,在2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會上,華為常務(wù)董事、華為云CEO張平安發(fā)表演講稱,數(shù)字世界的話語權(quán)最終是由生態(tài)的繁榮決定的。

關(guān)鍵字: 華為 12nm 手機 衛(wèi)星通信

要點: 有效應(yīng)對環(huán)境變化,經(jīng)營業(yè)績穩(wěn)中有升 落實提質(zhì)增效舉措,毛利潤率延續(xù)升勢 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務(wù)引領(lǐng)增長 以科技創(chuàng)新為引領(lǐng),提升企業(yè)核心競爭力 堅持高質(zhì)量發(fā)展策略,塑強核心競爭優(yōu)勢...

關(guān)鍵字: 通信 BSP 電信運營商 數(shù)字經(jīng)濟

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺與中國電影電視技術(shù)學(xué)會聯(lián)合牽頭組建的NVI技術(shù)創(chuàng)新聯(lián)盟在BIRTV2024超高清全產(chǎn)業(yè)鏈發(fā)展研討會上宣布正式成立。 活動現(xiàn)場 NVI技術(shù)創(chuàng)新聯(lián)...

關(guān)鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會上,軟通動力信息技術(shù)(集團)股份有限公司(以下簡稱"軟通動力")與長三角投資(上海)有限...

關(guān)鍵字: BSP 信息技術(shù)
關(guān)閉
關(guān)閉