新聞中心

        EEPW首頁 > 嵌入式系統(tǒng) > 設計應用 > keilc51可重入函數(shù)及模擬棧淺析

        keilc51可重入函數(shù)及模擬棧淺析

        作者: 時間:2012-04-20 來源:網(wǎng)絡 收藏

         1、關于可重入(可再入)和堆棧(仿真堆棧)

        本文引用地址:http://www.104case.com/article/149187.htm

          “可重入可以被一個以上的任務調用,而不必擔心數(shù)據(jù)被破壞。可重入函數(shù)任何時候都可以被中斷,一段時間以后又可以運行,而相應的數(shù)據(jù)不會丟失。”(摘自嵌入式實時操作系統(tǒng)uC/OS-II)

          在理解上述概念之前,必須先說一下的“覆蓋技術”。(采用該技術的原因請看附錄中一網(wǎng)友的解釋)

          (1)局部變量存儲在全局RAM空間(不考慮擴展外部存儲器的情況);

          (2)在編譯鏈接時,即已經(jīng)完成局部變量的定位;

          (3)如果各函數(shù)之間沒有直接或間接的調用關系,則其局部變量空間便可覆蓋。

          正是由于以上的原因,在Keil C51環(huán)境下,純粹的函數(shù)如果不加處理(如增加一個棧),是無法重入的。舉個例子:

        程序

        程序

          在上面的代碼中,TaskA與TaskB并不存在直接或間接的調用關系,因而它們的局部變量a與b便是可以被互相覆蓋的,即它們可能都被定位于某一個相同的RAM空間。這樣,當TaskA運行一段時間,改變了a后,TaskB取得CPU控制權并運行時,便可能會改變b。由于a和b指向相同的RAM空間,導致TaskA重新取得CPU控制權時,a的值已經(jīng)改變,從而導致程序運行不正確,反過來亦然。另一方面,func()與TaskB有直接的調用關系,因而其局部變量b與c不會被互相覆蓋,但也不能保證func的局部變量c不會與TaskA或其他任務的局部變量形成可覆蓋關系。

          根據(jù)上述分析我們很容易就能夠判斷出TaskA和TaskB這兩個函數(shù)是不可重入的(當然,func也不可重入)。那么如何讓函數(shù)成為可重入函數(shù)呢?C51編譯器采用了一個擴展關鍵字reentrant作為定義函數(shù)時的選項,需要將一個函數(shù)定義為可重入函數(shù)時,只要在函數(shù)后面加上關鍵字reentrant即可。

          與非可重入函數(shù)的參數(shù)傳遞和局部變量的存儲分配方法不同,C51編譯器為可重入函數(shù)生成一個棧(相對于系統(tǒng)堆棧或是硬件堆棧來說),通過這個模擬棧來完成參數(shù)傳遞和存放局部變量。模擬棧以全局變量?C_IBP、?C_PBP和?C_XBP作為棧指針(系統(tǒng)堆棧棧頂指針為SP),這些變量定義在DATA地址空間,并且可在文件startup.a51中進行初始化。根據(jù)編譯時采用的存儲器模式,模擬棧區(qū)可位于內部(IDATA)或外部(PDATA或XDATA)存儲器中。如表1所示:

        表1

        根據(jù)編譯時采用的存儲器模式

          注意:51系列單片機的系統(tǒng)堆棧(也叫硬件堆棧或常規(guī)棧)總是位于內部數(shù)據(jù)存儲器中(SP為 8位寄存器,只能指向內部),而且是“向上生長”型的(從低地址向高地址),而模擬棧是“向下生長”型的。

          1、可重入函數(shù)參數(shù)傳遞過程剖析

          在進入剖析之前,先簡單講講c51函數(shù)調用時參數(shù)是如何傳遞的。簡單來說,參數(shù)主要是通過寄存器R1~R7來傳遞的,如果在調用時,參數(shù)無寄存器可用或是采用了編譯控制指令“NOREGPARMS”,則參數(shù)的傳遞將發(fā)生在固定的存儲器區(qū)域,該存儲器區(qū)域稱為參數(shù)傳遞段,其地址空間取決于編譯時所選擇的存儲器模式。利用51單片機的工作寄存器最多傳遞3個參數(shù),如表2所示。

        表二

        利用51單片機的工作寄存器最多傳遞3個參數(shù)

          舉兩個例子:

          func1(int a):“a”是第一個參數(shù),在R6,R7中傳遞;

          func2(int b,int c, int *d):“b”在R6,R7中傳遞,“c”在R4,R5中傳遞,“*d”則在R1,R2,R3中傳遞。

          至于函數(shù)的返回值通過哪些寄存器或是什么方法傳遞這里就不說了,大家可以看看c51的相關文檔或是書籍。

          好了,接下來我們開始剖析一個簡單的程序,代碼如下:

        程序

          程序很簡單,廢話少說,下面跟我一起看看c51翻譯成的匯編語言是什么樣子的(大存儲模式下large XDATA)。

        程序

          說明:模擬棧指針最初在startup.a51中初始化為0xFFFF+1;由以上匯編代碼可以看出參數(shù)是從右往左掃描的。

          接下來看看fun的匯編代碼:(很長,大家耐心看吧,有些可以跳過的)

        程序


        上一頁 1 2 下一頁

        評論


        相關推薦

        技術專區(qū)

        關閉
        主站蜘蛛池模板: 西畴县| 建德市| 吴桥县| 建湖县| 东台市| 定安县| 石嘴山市| 东辽县| 弥勒县| 迁西县| 宝应县| 舟山市| 渑池县| 阳山县| 恩平市| 台江县| 绥江县| 大渡口区| 台南市| 昌宁县| 英山县| 濮阳市| 宁波市| 桓仁| 岑溪市| 建德市| 礼泉县| 安远县| 塔城市| 灵川县| 吉木萨尔县| 神池县| 平昌县| 沁源县| 石首市| 岳池县| 玉林市| 广南县| 江安县| 美姑县| 贡觉县|