在學(xué)校或者各種編程類書本上,基本上都會(huì)看到一句話:"函數(shù)是程序的基本組成單位",可以說理解函數(shù)對(duì)編程是非常重要的,與函數(shù)調(diào)用緊密結(jié)合的機(jī)制就是函數(shù)調(diào)用棧了,而棧有一個(gè)特別的屬性就是棧的增長方向問題了,也發(fā)現(xiàn)一些多年編程經(jīng)驗(yàn)的朋友對(duì)這一塊都有點(diǎn)迷迷糊糊的。在閱讀RTOS源碼的時(shí)候也會(huì)經(jīng)常看到棧的增長方向配置項(xiàng)目,那么今天就帶大家了解一下棧的增長方向到底是咋回事。
1、棧的增長方向
首先我們要明確的是棧同樣也是分布在我們的內(nèi)存之中,而內(nèi)存是通過地址來進(jìn)行編排訪問的,如下是堆棧的示意圖:
對(duì)于堆棧而言原本并沒有方向一說,只有入棧和出棧一說,程序中執(zhí)行push指令則棧頂向上移動(dòng),執(zhí)行pop指令則棧頂向下移動(dòng),其僅僅只是一種先進(jìn)后出的數(shù)據(jù)結(jié)構(gòu),增長方向都是從棧底向棧頂方向移動(dòng),即分配數(shù)據(jù)的過程。
而我們平時(shí)所說的棧的增長方向又是怎么回事呢?
為了在內(nèi)存中分配一段內(nèi)存給堆棧,我們必須要區(qū)分堆棧相對(duì)于內(nèi)存的地址而言的方向性,通常棧頂增長的方向是從內(nèi)存的低地址向高地址變化,我們則稱為向上增長;反之則向下增長。
所謂"水往高處流,即向上增長",這樣應(yīng)該就很好記憶了。
2、有什么用?
當(dāng)了解處理器中棧指針的增長方向以后,我們?cè)赿ebug程序的時(shí)候才能真正的把控程序的運(yùn)行過程。
在移植RTOS的過程中我們都需要對(duì)每個(gè)任務(wù)的堆棧分配一個(gè)合適的連續(xù)內(nèi)存區(qū)域來使用,此時(shí)初始狀態(tài)堆棧指針指向什么位置就跟堆棧的增長方向密切相關(guān),有過RTOS移植經(jīng)驗(yàn)的朋友應(yīng)該都有在RTOS配置項(xiàng)中關(guān)注過這塊的選擇。
RTOS在任務(wù)初始化的時(shí)候,其堆棧指針應(yīng)該指向其棧底位置,那么對(duì)于堆棧向上增長,任務(wù)初始化的時(shí)候我們需要把堆棧指針設(shè)置在所分配內(nèi)存的低地址內(nèi)存處,反之則設(shè)置到高地址處。
設(shè)置好以后,其在堆棧分配的過程中才會(huì)朝著所分配的內(nèi)存區(qū)域中,否則就會(huì)堆棧反向自爆,導(dǎo)致程序異常;如果你的堆棧分配不合理,同樣了解堆棧變化方向后也變得有跡可循。
同樣在裸機(jī)程序中也需要了解一下處理器的堆棧變化方向,從而用來排查一些堆棧溢出所導(dǎo)致的程序異常問題。
3、用C語言如何判斷?
要了解一個(gè)CPU的堆棧的變換方向,一方面就是查詢相應(yīng)的芯片參考手冊(cè),另外一方面就是實(shí)際測(cè)試了。
畢竟堆棧也就是內(nèi)存,自然就可以通過堆棧的分配過程取出所分配的內(nèi)存地址來比較判斷,而C語言可以方便的訪問內(nèi)存,也就比較容易判斷當(dāng)前處理器中堆棧指針的增長方向了。
那還不簡單,直接在函數(shù)內(nèi)部先后定義兩個(gè)局部變量,直接比較兩個(gè)變量的地址大小不就搞定了嗎?其實(shí)這種方式是依賴于編譯器實(shí)現(xiàn)的,畢竟哪個(gè)變量先進(jìn)行內(nèi)存申請(qǐng),并沒有太大的影響。
那么是否有一種方法不依賴于編譯器實(shí)現(xiàn)呢?
必須有的,那就是函數(shù)調(diào)用棧了,因?yàn)橄日{(diào)用的函數(shù)必然首先入棧。
基于這樣的思想,這里bug菌寫一個(gè)判斷堆棧增長方向的demo供大家參考:
可以拿去試一試,看看你的芯片堆棧咋變化的~
審核編輯:劉清
-
處理器
+關(guān)注
關(guān)注
68文章
19920瀏覽量
235659 -
芯片
+關(guān)注
關(guān)注
460文章
52566瀏覽量
441834 -
C語言
+關(guān)注
關(guān)注
180文章
7633瀏覽量
141972 -
RTOS
+關(guān)注
關(guān)注
24文章
851瀏覽量
121245
發(fā)布評(píng)論請(qǐng)先 登錄
深入理解C語言:C語言循環(huán)控制

用MATLAB或者C語言開發(fā)FPGA有什么問題嗎
EE-33:用C語言對(duì)ADSP-21xx定時(shí)器進(jìn)行編程

AWTK-WEB 快速入門(1) - C 語言應(yīng)用程序

在ads1261的通用c語言例程中的390行的if是用來區(qū)分什么的呢?
使用C語言實(shí)現(xiàn)函數(shù)模板
技術(shù)干貨驛站 ▏深入理解C語言:掌握C語言條件判斷,從if到switch的應(yīng)用

評(píng)論