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

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

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

為什么中斷處理函數(shù)不能直接調(diào)用不可重入函數(shù)

strongerHuang ? 來源:strongerHuang ? 作者:C語言與CPP編程 ? 2021-02-17 09:33 ? 次閱讀

1 前言

最近在公司維護的項目中碰到一個解決了定位很久的 bug , bug 找到的時候發(fā)現(xiàn)犯了很低級的錯誤——在中斷處理函數(shù)中調(diào)用了 printf 函數(shù),因為中斷處理函數(shù)的調(diào)用了不可重入函數(shù),導(dǎo)致中斷丟失和系統(tǒng)位置錯誤,這里直接導(dǎo)致嵌入式 linux 系統(tǒng)應(yīng)用進程中的所有線程停掉,進而導(dǎo)致看門狗進程得不到喂狗,設(shè)備重啟。

那什么是不可重入函數(shù)呢?

為什么中斷處理函數(shù)不能直接調(diào)用不可重入函數(shù)?

怎樣寫可重入函數(shù)?

就以上三個問題展開小短文:

2 什么是不可重入函數(shù)?

可重入函數(shù)主要用于多任務(wù)環(huán)境中,一個可重入的函數(shù)簡單來說就是可以被中斷的函數(shù),也就是說,可以在這個函數(shù)執(zhí)行的任何時刻中斷它,轉(zhuǎn)入 OS 調(diào)度下去執(zhí)行另外一段代碼,而返回控制時不會出現(xiàn)什么錯誤;而不可重入的函數(shù)由于使用了一些系統(tǒng)資源,比如全局變量區(qū),中斷向量表等,所以它如果被中斷的話,可能會出現(xiàn)問題,這類函數(shù)是不能運行在多任務(wù)環(huán)境下的。

滿足下列條件的函數(shù)多數(shù)是不可重入的:

函數(shù)體內(nèi)使用了靜態(tài)(static)的數(shù)據(jù)結(jié)構(gòu);

函數(shù)體內(nèi)調(diào)用了 malloc() 或者 free() 函數(shù);

函數(shù)體內(nèi)調(diào)用了標(biāo)準 I/O 函數(shù);

A. 可重入函數(shù)

o4YBAF_2ZTuAc1xyAAAeslwkBXY186.jpg

B. 不可重入函數(shù)1

o4YBAF_2ZU2AR4Z7AAAkUu9cugE344.jpg

C. 不可重入函數(shù)2

pIYBAF_2ZV6AJsskAAAo5ZMEv4M490.jpg

3 為什么中斷處理函數(shù)不能直接調(diào)用不可重入函數(shù)?

在多任務(wù)系統(tǒng)下,中斷可能在任務(wù)執(zhí)行的任何時間發(fā)生;如果一個函數(shù)的執(zhí)行期間被中斷后,到重新恢復(fù)到斷點進行執(zhí)行的過程中,函數(shù)所依賴的環(huán)境沒有發(fā)生改變,那么這個函數(shù)就是可重入的,否則就不可重入。

在中斷前后不都要保存和恢復(fù)上下文嗎,怎么會出現(xiàn)函數(shù)所依賴的環(huán)境發(fā)生改變了呢?我們知道中斷時確實保存一些上下文,但是僅限于返回地址,cpu 寄存器等之類的少量上下文,而函數(shù)內(nèi)部使用的諸如全局或靜態(tài)變量,buffer 等并不在保護之列,所以如果這些值在函數(shù)被中斷期間發(fā)生了改變,那么當(dāng)函數(shù)回到斷點繼續(xù)執(zhí)行時,其結(jié)果就不可預(yù)料了。

在中斷處理函數(shù)中調(diào)用有互斥鎖保護的全局變量,如果恰好該變量正在被另一個線程調(diào)用,會導(dǎo)致中斷處理函數(shù)不能及時返回,導(dǎo)致中斷丟失等嚴重問題。

并且在多線程環(huán)境中使用,在沒有加鎖的情況下,對同一段內(nèi)存塊進行并發(fā)讀寫,就會造成 segmentfault/coredump 之類的問題。

總而言之,中斷處理函數(shù)做的事情越簡單越好。

4 如何寫出可重入的函數(shù)?

在函數(shù)體內(nèi)不訪問那些全局變量;

如果必須訪問全局變量,記住利用互斥信號量來保護全局變量?;蛘哒{(diào)用該函數(shù)前關(guān)中斷,調(diào)用后再開中斷;

不使用靜態(tài)局部變量;

堅持只使用缺省態(tài)(auto)局部變量;

在和硬件發(fā)生交互的時候,切記關(guān)閉硬件中斷。完成交互記得打開中斷,在有些系列上,這叫做“進入/退出核心”或者用 OS_ENTER_KERNAL/OS_EXIT_KERNAL 來描述;

不能調(diào)用任何不可重入的函數(shù);

謹慎使用堆棧。最好先在使用前先 OS_ENTER_KERNAL;

責(zé)任編輯:xj

原文標(biāo)題:中斷函數(shù)調(diào)用不可重入函數(shù)的后果

文章出處:【微信公眾號:strongerHuang】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

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

    關(guān)注

    180

    文章

    7604

    瀏覽量

    136860
  • 函數(shù)
    +關(guān)注

    關(guān)注

    3

    文章

    4331

    瀏覽量

    62629
  • 中斷函數(shù)
    +關(guān)注

    關(guān)注

    0

    文章

    13

    瀏覽量

    5287

原文標(biāo)題:中斷函數(shù)調(diào)用不可重入函數(shù)的后果

文章出處:【微信號:strongerHuang,微信公眾號:strongerHuang】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

收藏 人收藏

    評論

    相關(guān)推薦

    HAL庫的函數(shù)調(diào)用示例

    HAL(Hardware Abstraction Layer,硬件抽象層)庫是STM32等微控制器中常用的庫,它為開發(fā)者提供了訪問和控制硬件設(shè)備的接口。以下是一些常用的HAL庫函數(shù)及其調(diào)用示例: 一
    的頭像 發(fā)表于 12-02 14:01 ?389次閱讀

    定時器回調(diào)函數(shù)不能用ICACHE_FLASH_ATTR定義?

    非 OS SDK 在中斷處理函數(shù)中,請勿使用任何 ICACHE_FLASH_ATTR 定義的函數(shù)。 請問: 1、定時器和hw定時器的回調(diào)函數(shù)
    發(fā)表于 07-22 06:33

    FreeRTOS如何在中斷調(diào)用內(nèi)存分配函數(shù)?

    最近在玩FreeRTOS,遇到一個問題,就是不知如何在中斷調(diào)用內(nèi)存分配函數(shù)。pvPortMalloc函數(shù)中會調(diào)用xTaskResumeAl
    發(fā)表于 05-08 08:25

    IAR庫函數(shù)中斷服務(wù)程序中無法調(diào)用函數(shù)是為什么?

    為什么在中斷服務(wù)函數(shù)里無法調(diào)用?定義的變量也不能用!求大神幫忙解決問題!子函數(shù)怎么寫,寫到哪,才能用?實在是不知道怎么辦了。
    發(fā)表于 05-08 08:01

    STVD中斷函數(shù)如何調(diào)用才能放到主函數(shù)使用?

    我用的是STVD,一直研究定時器4中斷,網(wǎng)上找了很多資料,現(xiàn)在是定時器4中斷可以正常工作,但是中斷函數(shù)只能寫在stm8_interrupt_vector.c這個里面,如果寫到主
    發(fā)表于 04-26 06:25

    STM32中斷相應(yīng)函數(shù)不能持續(xù)嗎?

    在main.c中的測試函數(shù)可以使DAC芯片正常輸出多路幅值不同的方波電壓,將其放到中斷it.c文件中,發(fā)現(xiàn)只是輸出直線型電壓,而非方波,請問中斷相應(yīng)函數(shù)
    發(fā)表于 04-08 07:05

    函數(shù)多層調(diào)用的主要注意事項分析

    應(yīng)用方案設(shè)計中,開發(fā)者經(jīng)常會碰到某個子函數(shù)需要多次多級調(diào)用的情況。
    的頭像 發(fā)表于 03-27 15:36 ?890次閱讀
    子<b class='flag-5'>函數(shù)</b>多層<b class='flag-5'>調(diào)用</b>的主要注意事項分析

    回調(diào)函數(shù)(callback)是什么?回調(diào)函數(shù)的實現(xiàn)方法

    回調(diào)函數(shù)是一種特殊的函數(shù),它作為參數(shù)傳遞給另一個函數(shù),并在被調(diào)用函數(shù)執(zhí)行完畢后被調(diào)用?;卣{(diào)
    發(fā)表于 03-12 11:46 ?2948次閱讀

    STM32cubeIDE PA0口外部中斷改變LED燈狀態(tài)時,GPIO翻轉(zhuǎn)函數(shù)放在外部中斷回調(diào)函數(shù)中不被調(diào)用怎么解決?

    STM32cubeIDE PA0口外部中斷改變LED燈狀態(tài)時,GPIO翻轉(zhuǎn)函數(shù)放在外部中斷回調(diào)函數(shù)中不被調(diào)用,放在EXTI0_IRQHand
    發(fā)表于 03-12 06:32

    函數(shù)指針與回調(diào)函數(shù)的應(yīng)用實例

    通常我們說的指針變量是指向一個整型、字符型或數(shù)組等變量,而函數(shù)指針是指向函數(shù)。 函數(shù)指針可以像一般函數(shù)一樣,用于調(diào)用
    的頭像 發(fā)表于 03-07 11:13 ?402次閱讀
    <b class='flag-5'>函數(shù)</b>指針與回調(diào)<b class='flag-5'>函數(shù)</b>的應(yīng)用實例

    內(nèi)聯(lián)函數(shù)定義 為什么需要內(nèi)聯(lián)函數(shù)

    inline關(guān)鍵字是C99標(biāo)準的型關(guān)鍵字,其作用是將函數(shù)展開,把函數(shù)的代碼復(fù)制到每一個調(diào)用處。
    的頭像 發(fā)表于 02-19 12:20 ?548次閱讀

    verilog中函數(shù)和任務(wù)對比

    在verilog中,函數(shù)和任務(wù)均用來描述共同的代碼段,并且在模式內(nèi)任意位置被調(diào)用,提高代碼效率,讓代碼更加的直觀,提高代碼可讀性。但是在實際使用的過程中,函數(shù)和任務(wù)也存在諸多的不同,下面將對而這進行
    的頭像 發(fā)表于 02-12 18:43 ?892次閱讀

    中斷調(diào)用函數(shù)IfxCcu6_PwmBc_updateHallPattern時,寄存器MCMOUTS的值為什么不能更新到MCMOUT中?

    中斷調(diào)用函數(shù)IfxCcu6_PwmBc_updateHallPattern時,寄存器MCMOUTS的值為什么不能更新到MCMOUT中
    發(fā)表于 02-05 06:32

    linux用gdb調(diào)試遇到函數(shù)調(diào)用怎么辦?

    linux用gdb調(diào)試遇到函數(shù)調(diào)用怎么辦? 在Linux上使用GDB調(diào)試時,遇到函數(shù)調(diào)用是一個常見的情況。函數(shù)
    的頭像 發(fā)表于 01-31 10:33 ?719次閱讀

    GD32 MCU是如何進入中斷函數(shù)

    用過GD32 MCU的小伙伴們都知道,程序是順序執(zhí)行的,但當(dāng)有中斷來的時候程序會跳轉(zhuǎn)到中斷函數(shù),執(zhí)行完中斷函數(shù)后程序又繼續(xù)回到原來的位置繼續(xù)
    的頭像 發(fā)表于 01-30 09:45 ?1100次閱讀
    GD32 MCU是如何進入<b class='flag-5'>中斷</b><b class='flag-5'>函數(shù)</b>的