找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

搜索
查看: 7403|回復(fù): 2
收起左側(cè)

關(guān)于單片機(jī)編譯器中對函數(shù)中局部變量的處理

[復(fù)制鏈接]
ID:153041 發(fā)表于 2017-2-20 12:48 | 顯示全部樓層 |閱讀模式

原作者: chinseeker

一個平凡的探索者


通常我們都是學(xué)了標(biāo)準(zhǔn)c語言教程后從事單片機(jī)c語言的編寫的, 那就先要明白一點, 標(biāo)準(zhǔn)c語言實際上是起源于pc平臺上的一種語言, 標(biāo)準(zhǔn)c語言肯定是不會照顧到單片機(jī)的特殊性的. 因此單片機(jī)c編譯器中的c語言是一種基于標(biāo)準(zhǔn)c,但是又有相應(yīng)修改擴(kuò)充的擴(kuò)展c語言.  所以在單片機(jī)c編譯器里寫程序時一定要了解單片機(jī)編譯器擴(kuò)展c語言的不同之處, 絕不能死板地照搬標(biāo)準(zhǔn)c。


在標(biāo)準(zhǔn)c里, 局部變量是函數(shù)在調(diào)用的時候才臨時分配存儲空間的,全局變量是程序整個生命周期都一直存在的.  不過要知道,臨時分配存儲空間是需要操作系統(tǒng)內(nèi)存管理程序支持的, 單片機(jī)中通常都沒有操作系統(tǒng),也就不能實現(xiàn)像pc平臺中那樣的局部變量的空間分配.  這里就需要深入了解一下單片機(jī)的c編譯器究竟是如何處理局部變量的,如果對此沒有概念,碰到調(diào)試過程中的一些奇異現(xiàn)象恐怕只能覺匪夷所思了。


另外需要知道的一點是, 不同的編譯器對于局部變量的處理方法也不一樣, 不能學(xué)了一個就到處照搬. 這里拿KEIL C ,IARAVR, ICCAVR這個三個編譯器做分析比較。


首先說 Keil C51 , 它的局部變量并不是在堆棧中, C51 為了提高代碼的效率, 根據(jù) 51 處理器的特性. 編譯器對函數(shù)局部變量的安排進(jìn)行了處理.局部變量如果不能分配到 寄存器里, 就放在 RAM 中了.編譯器通過覆蓋分析, 可以共享局部變量的地址空間.。 最終的DATA使用量取決于調(diào)用鏈中那個使用DATA最多的鏈。所以,在程序中增加一個局部變量,如果不是位于那個使用DATA最多的鏈中,需要的DATA數(shù)量是有可能不會增加的。


      如:main()->f11()->f12()->f13().... // 鏈1

              |----->f21()->f22()->f23().... // 鏈2


因為f11(),f21()不在同一個調(diào)用鏈上,顯然,f11()中使用的局部變量,可以和f21()中的局部變量,使用同一個存儲單元。因為它們中的任何一個處在生命期內(nèi)的話,另一個必然已經(jīng)離開它的生命周期,同時它的局部變量也離開了它的生命周期,這些局部變量所占用的存儲單元當(dāng)然可以另做它用了。


假設(shè)鏈1目前的局部變量需要50個存儲單元,鏈2需要40個存儲單元。那么你在鏈2中加入不多于10個單元的局部變量的話,程序最終需要的存儲單元數(shù)量是不會增加的。



再說ICCAVR , 它把局部變量存放在軟件堆?臻g中.  ICCAVR使用兩個堆棧:一個用于子程序調(diào)用和中斷操作的硬件堆棧,一個用于傳遞參數(shù)、臨時變量和局部變量的軟件堆棧。硬件堆棧是從數(shù)據(jù)內(nèi)存的頂部開始分配的,在硬件堆棧下面再分配一定數(shù)量的字節(jié)作為軟件堆棧。


IARAVR對于局部變量的處理方法與ICCAVR一樣. 它也有兩個堆棧,一個是data stack ,一個是return address stack.  分別用于存放臨時變量,局部變量,傳遞參數(shù), 和函數(shù)返回地址.


這里需要注意的是, 局部變量存放在堆棧中的處理方式一定要保證堆棧足夠大, 特別是定義了局部數(shù)組變量的情況下,一旦數(shù)組過大,超過了堆棧大小就會發(fā)生堆棧溢出,  如果只是讀取數(shù)據(jù)還好, 一旦寫入數(shù)據(jù),就會破壞堆棧空間以外的數(shù)據(jù), 導(dǎo)致程序時常.



評分

參與人數(shù) 1黑幣 +100 收起 理由
admin + 100 共享資料的黑幣獎勵!

查看全部評分

回復(fù)

使用道具 舉報

ID:98618 發(fā)表于 2017-2-22 01:09 | 顯示全部樓層
能從編譯器級別來進(jìn)行調(diào)試的都是高手啊,佩服樓主的技術(shù),不過平時很少用avr單片機(jī),不知道樓主有51單片機(jī)方面的編譯器優(yōu)化資料嗎?
回復(fù)

使用道具 舉報

ID:69798 發(fā)表于 2018-2-10 00:38 | 顯示全部樓層
難得一見的好文,很有用的文章,分析得很到位,先收藏謝謝樓主了
回復(fù)

使用道具 舉報

您需要登錄后才可以回帖 登錄 | 立即注冊

本版積分規(guī)則

手機(jī)版|小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術(shù)交流QQ群281945664

Powered by 單片機(jī)教程網(wǎng)

快速回復(fù) 返回頂部 返回列表