0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

教大家如何寫出一個(gè)現(xiàn)場(chǎng)保存函數(shù)以及如何調(diào)用該函數(shù)

冬至子 ? 來(lái)源:嵌入式BugMaker ? 作者:BugMaker ? 2023-08-10 11:33 ? 次閱讀

在進(jìn)行CortexM系列MCU開(kāi)發(fā)時(shí),大家應(yīng)該都或多或少觸發(fā)過(guò)不可屏蔽中斷,例如人見(jiàn)人愛(ài)的Hardfault。

對(duì)于嵌入式工程師來(lái)說(shuō),能夠穩(wěn)定復(fù)現(xiàn)的Bug最好解決,再不濟(jì)挨個(gè)打斷點(diǎn)也能定位到問(wèn)題代碼位置。

可是異常如果是不定時(shí),無(wú)規(guī)律的發(fā)生在日常的運(yùn)行中,那打斷點(diǎn)的方式無(wú)異于守株待兔。

本篇將提供小技巧簡(jiǎn)述CortexM的異常壓棧機(jī)制,教大家如何寫出一個(gè)現(xiàn)場(chǎng)保存函數(shù)以及如何調(diào)用該函數(shù)。

如下圖所示,觸發(fā)中斷的時(shí)候,CortexM?系列芯片會(huì)按照該排列將相關(guān)現(xiàn)場(chǎng)數(shù)據(jù)進(jìn)行地址偏移壓棧,N即此時(shí)的SP數(shù)值,通過(guò)SP中存放的地址,我們可以獲取到如圖所示的數(shù)據(jù)。

而要知道觸發(fā)異常的地址則需要關(guān)注LR(Link Register)的數(shù)值,它存放著異常觸發(fā)點(diǎn)的函數(shù)地址。

圖片

那么按照壓棧的排列,我們可以編寫一個(gè)結(jié)構(gòu)體以及相關(guān)的記錄現(xiàn)場(chǎng)函數(shù),如下圖所示,結(jié)構(gòu)體的成員完全是按照上圖的壓棧順序排列,那下一步就是在哪里調(diào)用這個(gè)接口來(lái)獲取到案發(fā)時(shí)的第一手資料

圖片

?要想獲得第一手?jǐn)?shù)據(jù),就要去中斷函數(shù)的執(zhí)行處,很多廠商SDK都會(huì)在名為XXX_it.c文件里為開(kāi)發(fā)者們寫好各個(gè)中斷的Handler,由于S文件中都是弱定義(弱定義即在C文件中有同名的函數(shù),則以C文件中的函數(shù)優(yōu)先編譯,則可能造成觸發(fā)中斷時(shí)并不會(huì)運(yùn)行S文件處的函數(shù))所以建議可以將其屏蔽或者自己重新編寫一個(gè)新的函數(shù)名。

由下圖可見(jiàn),通過(guò)IMPORT的形式調(diào)用外部現(xiàn)場(chǎng)保存函數(shù),將SP寄存器賦值給R0寄存器是因?yàn)镽0通常時(shí)作為函數(shù)的第一個(gè)參數(shù)進(jìn)行調(diào)用的,可看作是將SP數(shù)值作為參數(shù)傳入StackSave函數(shù),BL的意思是帶返回的跳轉(zhuǎn),即使跳轉(zhuǎn)運(yùn)行完成StackSave函數(shù)后還返回此處。

圖片

完成上述函數(shù)編寫以及調(diào)用后,即可進(jìn)行測(cè)試了。本次范例通過(guò)SPI初始化時(shí)傳入錯(cuò)誤的外設(shè)寄存器地址來(lái)觸發(fā)HardFault中斷,由下圖可看出,此時(shí)記錄到的LR數(shù)值為0x8000e83

圖片

?查詢后可知,0x8000e83的位置為SPI初始化函數(shù)中,符合預(yù)設(shè)的異常觸發(fā)位置。在非仿真環(huán)境下可通過(guò)map文件找到大致的函數(shù)地址。

圖片

綜上,我們就知道了如何獲取到觸發(fā)異常中斷時(shí)的程序運(yùn)行地址,通過(guò)將獲取到的信息存入到片上Flash指定地址的形式將此時(shí)的現(xiàn)場(chǎng)信息進(jìn)行保存,并通過(guò)固件和map一一對(duì)應(yīng)管控。

這樣在漫長(zhǎng)運(yùn)行過(guò)程中,如果不慎觸發(fā)異常異常中斷,通過(guò)對(duì)應(yīng)的map文件即可迅速定位BUG所在大致位置?

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問(wèn)題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • 寄存器
    +關(guān)注

    關(guān)注

    31

    文章

    5290

    瀏覽量

    119786
  • 仿真器
    +關(guān)注

    關(guān)注

    14

    文章

    1012

    瀏覽量

    83586
  • 串口中斷
    +關(guān)注

    關(guān)注

    0

    文章

    64

    瀏覽量

    13845
  • Flash單片機(jī)
    +關(guān)注

    關(guān)注

    0

    文章

    111

    瀏覽量

    9377
  • MCU芯片
    +關(guān)注

    關(guān)注

    3

    文章

    246

    瀏覽量

    11348
收藏 人收藏

    評(píng)論

    相關(guān)推薦

    C函數(shù)調(diào)用機(jī)制與棧幀原理詳解

    當(dāng)個(gè)C函數(shù)調(diào)用時(shí),函數(shù)的參數(shù)如何傳遞、堆棧指針如何變化、棧幀是如何被建立以及如何被消除的,
    發(fā)表于 06-08 10:49 ?1158次閱讀
    C<b class='flag-5'>函數(shù)</b><b class='flag-5'>調(diào)用</b>機(jī)制與棧幀原理詳解

    文詳解python調(diào)用函數(shù)

    函數(shù)被定義后,本身是不會(huì)自動(dòng)執(zhí)行的,只有在被調(diào)用后,函數(shù)才會(huì)被執(zhí)行,得到相應(yīng)的結(jié)果。但是在 Python 中我們要注意個(gè)關(guān)鍵點(diǎn),就是Pyt
    發(fā)表于 10-01 10:45 ?616次閱讀

    如何查看及更改函數(shù)/函數(shù)塊的調(diào)用環(huán)境

    模塊化設(shè)計(jì)的思想是把些相似的功能(比如電機(jī)控制、閥控制)設(shè)計(jì)成函數(shù)函數(shù)塊,這樣就可以反復(fù)調(diào)用。其優(yōu)點(diǎn)是:使程序架構(gòu)更加清晰,避免重復(fù)編寫相似功能的代碼。不過(guò)可能會(huì)產(chǎn)生
    的頭像 發(fā)表于 11-17 09:08 ?826次閱讀
    如何查看及更改<b class='flag-5'>函數(shù)</b>/<b class='flag-5'>函數(shù)</b>塊的<b class='flag-5'>調(diào)用</b>環(huán)境

    如何寫出多串口共用printf函數(shù)語(yǔ)句呢

    如何寫出多串口共用printf函數(shù)語(yǔ)句呢?求大神解答
    發(fā)表于 11-17 06:37

    C++教程之函數(shù)的遞歸調(diào)用

    C++教程之函數(shù)的遞歸調(diào)用 在執(zhí)行函數(shù) f 的過(guò)程中,又要調(diào)用 f 函數(shù)本身,稱為函數(shù)的遞歸
    發(fā)表于 05-15 18:00 ?35次下載

    當(dāng)函數(shù)執(zhí)行完畢后,如何返回調(diào)用處?

    當(dāng)函數(shù)執(zhí)行完畢后,如何返回調(diào)用處呢?由于該函數(shù)可能會(huì)被多次調(diào)用,且每次調(diào)用的地方很可能不樣,這
    的頭像 發(fā)表于 09-14 14:27 ?1.7w次閱讀
    當(dāng)<b class='flag-5'>函數(shù)</b>執(zhí)行完畢后,如何返回<b class='flag-5'>調(diào)用</b>處?

    帶你了解嵌入式C語(yǔ)言函數(shù)調(diào)用

    大家都知道函數(shù)調(diào)用是通過(guò)棧來(lái)實(shí)現(xiàn)的,而且知道在棧中存放著該函數(shù)的局部變量。但是對(duì)于棧的實(shí)現(xiàn)細(xì)節(jié)可能不定清楚。
    發(fā)表于 07-12 17:08 ?2066次閱讀
    帶你了解嵌入式C語(yǔ)言<b class='flag-5'>函數(shù)</b><b class='flag-5'>調(diào)用</b>棧

    C語(yǔ)言函數(shù)的回調(diào)函數(shù)

    ,如果過(guò)后再執(zhí)行,則稱之為異步回調(diào)。 回調(diào)函數(shù)就是個(gè)通過(guò)函數(shù)指針調(diào)用函數(shù)。如果你把
    的頭像 發(fā)表于 09-11 09:57 ?4087次閱讀

    傳遞函數(shù)是怎么定義的以及如何寫出傳遞函數(shù)?

    本來(lái)這期應(yīng)該講Buck的尖峰的,不過(guò)遇到些問(wèn)題,所以就往后拖拖吧,這次來(lái)個(gè)簡(jiǎn)單點(diǎn)兒的。 來(lái)看看傳遞函數(shù),也就是我們經(jīng)??吹降腍(s)。 傳遞函數(shù)是怎么定義的呢? 百科是這么定義的: 【傳遞
    的頭像 發(fā)表于 06-30 11:23 ?7.7w次閱讀
    傳遞<b class='flag-5'>函數(shù)</b>是怎么定義的<b class='flag-5'>以及</b><b class='flag-5'>如何寫出</b>傳遞<b class='flag-5'>函數(shù)</b>?

    如何寫要被C調(diào)用的匯編函數(shù)

    問(wèn):我該怎么寫個(gè)將被C調(diào)用的匯編函數(shù)? 答:個(gè)最簡(jiǎn)單的學(xué)習(xí)怎么寫匯編的方式是,先用C寫這個(gè)代
    的頭像 發(fā)表于 01-22 14:54 ?1227次閱讀

    C語(yǔ)言函數(shù)調(diào)用的形式及過(guò)程

    C語(yǔ)言函數(shù)調(diào)用時(shí)的數(shù)據(jù)傳遞 在調(diào)用有參函數(shù)時(shí),主調(diào)函數(shù)和被調(diào)函數(shù)之間有數(shù)據(jù)傳遞關(guān)系。
    的頭像 發(fā)表于 03-10 14:28 ?1567次閱讀

    什么是函數(shù)調(diào)用?

    函數(shù)調(diào)用,就是使用我們已經(jīng)定義好的函數(shù),或者C語(yǔ)言自帶的庫(kù)函數(shù)。
    的頭像 發(fā)表于 04-04 17:21 ?5560次閱讀

    python定義函數(shù)調(diào)用函數(shù)的順序

    定義函數(shù)調(diào)用函數(shù)的順序 函數(shù)被定義后,本身是不會(huì)自動(dòng)執(zhí)行的,只有在被調(diào)用后,函數(shù)才會(huì)被執(zhí)行,得
    的頭像 發(fā)表于 10-04 17:17 ?1234次閱讀

    python函數(shù)函數(shù)之間的調(diào)用

    ) 運(yùn)行結(jié)果: 無(wú)結(jié)果 分析:因?yàn)榈?行中的x(f)中的f沒(méi)有帶括號(hào),f只是個(gè)普通的參數(shù),所以程序只調(diào)用執(zhí)行了x(f)函數(shù),沒(méi)有調(diào)用執(zhí)行f
    的頭像 發(fā)表于 10-04 17:17 ?547次閱讀

    OP-TEE的內(nèi)核初始化函數(shù)調(diào)用

    init_primary_helper函數(shù)來(lái)完成系統(tǒng)運(yùn)行環(huán)境的建立 ,如果系統(tǒng)支持ATF,則該函數(shù)會(huì)返回OP-TEE的處理句柄,該處理句柄主要包含 ? 各種安全監(jiān)控模式調(diào)用的處理函數(shù)、
    的頭像 發(fā)表于 11-02 18:18 ?685次閱讀
    OP-TEE的內(nèi)核初始化<b class='flag-5'>函數(shù)</b><b class='flag-5'>調(diào)用</b>