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

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

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

應用筆記 | 淺談STM32庫里的回調函數(shù)

STM32單片機 ? 來源:未知 ? 2023-09-14 17:10 ? 次閱讀

wKgZomUCzxeAaQAcAAHUHXdxAL4057.gif

關鍵字:回調函數(shù),HAL庫

目錄預覽

1.回調函數(shù)

2.STM32固件庫里的回調函數(shù)

3.STM32庫函數(shù)里的回調機制及觸發(fā)事件

4.常見問題

01 回調函數(shù)

有人對STM32固件庫里的回調函數(shù)有些好奇甚至糾結,這里簡單地介紹下,以供參考。其實從用法及功能上講他們并沒有什么特別的,跟其他函數(shù)一樣,也是實現(xiàn)特定功能的代碼段。一般來講,所謂回調函數(shù),泛指基于事件觸發(fā)而被調用執(zhí)行的函數(shù),簡單點說,就是條件滿足了就調用的函數(shù),往往會跟函數(shù)指針結合起來通過函數(shù)指針實現(xiàn)調用。

經(jīng)常會有人基于類似下面的代碼介紹回調函數(shù):

wKgZomUCzxeAeJQJAAE3ih5ygAI645.png

wKgZomUCzxeAcQ0qAACzLxsXWNI996.png

在上面代碼中,那四個有關加減乘除的函數(shù)可以看成回調函數(shù),具體何時被調用,根據(jù)函數(shù)Compute(float a,float b,float(*Action)(float a,float b))里的函數(shù)指針的賦值情況來定,被賦予哪個回調函數(shù)的地址就調用哪個回調函數(shù)。當然,使用函數(shù)指針并不是回調函數(shù)的核心特征,因事件驅動而被調用才是其核心特征。

生活中我們有時會對某人說,回頭再談、回頭再聊。潛臺詞往往就是等時機成熟了、條件滿足了再來具體交涉。這里就充滿著濃濃的回調意味。

回調函數(shù)可以理解為事件響應函數(shù)或者說事件驅動函數(shù)。即使相同的事件、基于不同的場景可能會有不同應對處理,從軟件代碼角度講就對應不同的回調函數(shù)代碼。

我們不妨看個生活中的例子。生活中有人中了六合彩了,針對這一事件,中獎人可能有下面諸多舉動之一【這里簡化下,多選一】。但這件事發(fā)生在不同人身上,右邊的選擇很可能不盡一樣。換言之,中獎了,到底會選擇右邊哪一項還得結合具體的人來定。

wKgZomUCzxeAFlJnAACj-D4y6D8586.png

圖1 中六合彩的可能后續(xù)行為

我們再切換到STM32的嵌入式開發(fā)中來,以UART接收完成事件為例。針對這一事件,不同的應用場景的應對處理往往也是五花八門、五彩繽紛。

wKgZomUCzxeAEWUJAACtWY4vIss183.png

圖2 UART接收完成后可能后續(xù)動作

顯然,特定的應用場景對應著特定的回調函數(shù),一般來講,沒法簡單地僅僅基于事件就擬定一段既能適用于各種場景而又富有針對性的代碼。

結合上面的描述,稍微小結下?;卣{函數(shù)除了具有基于事件的觸發(fā)而被調用執(zhí)行的特征外,還具有相同事件因應不同應用場景可能需要不同的回調函數(shù)之特征,即基于特定應用場景的回調函數(shù)其內(nèi)容具有特定性。

02

STM32固件庫里的回調函數(shù)

說到這里,我們具體結合STM32外設固件庫里回調函數(shù)來聊聊。

首先,作為一個函數(shù)庫,除了個別初始化函數(shù)外,里面不存在現(xiàn)存的完整的回調函數(shù)。結合前面的介紹,我們知道回調函數(shù)需要結合具體場景而擬定,作為函數(shù)庫根本做不到這一點,它沒法事先知曉發(fā)生某個事件時不同的應用會需要采取怎樣的操作。

其次,STM32庫函數(shù)的確采用了回調機制,并基于可能的各種事件為STM32開發(fā)者預留了只有函數(shù)定義而無具體內(nèi)容的回調函數(shù),或者是只定義了一些基于各類事件的函數(shù)指針,具體的回調函數(shù)需要用戶完成并將函數(shù)地址賦給相應的函數(shù)指針而被調用。簡單點說,函數(shù)庫給我們事先預留了眾多的回調函數(shù)接口

STM32固件庫里的回調函數(shù)采用了兩種調用方式:

第一種是legacy方式,傳統(tǒng)的回調方式,庫以weak方式定義了各種空的回調函數(shù),像下面這些。STM32庫里都給我們準備好了。【下面是有關UART部分事件的弱回調函數(shù)體,內(nèi)容為空】

wKgZomUCzxeADzDJAANeplyjz2o647.png

圖3 UART傳輸事件相關弱回調函數(shù)定義

具體開發(fā)時,我們根據(jù)事件和應用場景基于類似上面的weak函數(shù)進行重寫,重寫時拿掉weak,庫里預留的弱定義函數(shù)盡量不用動它。比方像下面這些都是最終的用戶回調函數(shù)。

wKgZomUCzxiAT000AATeo_OfHWQ133.png

圖4 UART傳輸事件相關的用戶回調函數(shù)

另外一種就是指針方式,或稱注冊方式。即函數(shù)庫里事先基于各類事件定義好了各種回調函數(shù)指針,具體的回調函數(shù)由用戶基于不同事件和應用需求撰寫,然后將函數(shù)地址賦給函數(shù)指針,這個動作我們稱之為回調函數(shù)進行注冊,之后回調函數(shù)就可以通過函數(shù)指針而被適時調用。

比方下面是UART外設里定義的一些函數(shù)指針:【星號所指的是與UART傳輸完成事件有關的回調函數(shù)所用的指針】

wKgZomUCzxiAS8UPAAfoAb6dGr8950.png

圖5 UART傳輸事件相關的回調函數(shù)指針

當我們將回調函數(shù)寫好后,將函數(shù)地址賦給函數(shù)指針即可在相應事件發(fā)生時被調用。比方類似下面的操作代碼。紅星標所指代碼就是在做回調函數(shù)的注冊。

wKgZomUCzxiAABYkAADrvcGtRLk282.png

圖6 UART傳輸完成事件用戶回調函數(shù)及注冊

給函數(shù)指針賦地址可以直接賦地址或通過調用庫函數(shù)xxx_RegisterCallback完成【見上圖星標代碼】。

這種指針方式需要我們對C語言中的結構體、函數(shù)指針有相應的了解,庫只是給我們提供了相應的函數(shù)指針,具體的用戶回調函數(shù)由用戶根據(jù)需要來編寫,將其地址賦給相應的函數(shù)指針以供調用。

而前面介紹的傳統(tǒng)型回調函數(shù),庫則幫我們把可能涉及到的回調函數(shù)全部以弱定義的方式都準備好了,我們按需針對性選用,去掉weak填空重寫。使用起來相對更直觀些,無需我們對函數(shù)指針有太多了解。

目前STM32庫回調機制中,作為用戶到底使用上面的哪種回調方式呢?在每個系列的固件庫的配置頭文件中有針對各個外設事件回調函數(shù)使用方式的選擇,比方以STM32F4系列為例,這里有個stm32f4xx_hal_conf.h的頭文件,我們可以看到基于各個外設事件回調函數(shù)使用方式選擇的宏。

wKgZomUCzxiAGgUlAAUt6fbPy1Q732.png

圖7 回調函數(shù)調用方式的選擇配置

若我們不對該頭文件的相應外設事件的回調函數(shù)調用方式的宏定義做調整,則默認傳統(tǒng)回調方式,即legacy方式,非指針方式。若將相應的宏值改為1,則該外設事件相關回調函數(shù)采用指針注冊方式。

03 STM32庫函數(shù)里的回調機制及觸發(fā)事件

整體上講,STM32外設庫里的API函數(shù)大體由三部分組成,分別是:

初始化函數(shù)

啟動型執(zhí)行函數(shù)

回調函數(shù)【弱定義函數(shù)或回調函數(shù)指針,最終靠用戶具體完成編寫】

這樣的安排,讓整個工程代碼結構比較清晰,可以讓人快速了解庫結構,同時現(xiàn)存的API函數(shù)大大減少開發(fā)工作量,預留的回調函數(shù)接口一方面給開發(fā)者提供了便利,另一方面讓用戶基于不同應用場景自由組織代碼而又不破壞整個軟件架構。

對于回調函數(shù),可以由哪些事件觸發(fā)呢?大致分三類,分別是外設初始化操作、外設處理完成【中斷】事件、外設出錯【中斷】事件。我們關注最多是外設處理完成中斷事件相關的回調函數(shù)。

wKgZomUCzxmAMpSLAAGwSBkNnWY201.png

圖8 回調函數(shù)觸發(fā)事件的分類

04 常見問題

4.1 STM32庫函數(shù)里的回調函數(shù)是什么,有何用?

回調函數(shù)終究乃用戶所編寫,是用戶基于特定事件和應用需求而編寫的功能模塊,與其他函數(shù)并無本質區(qū)別。形式上講,STM32庫預先為用戶做了回調函數(shù)的弱定義或基于事件的函數(shù)指針的定義。因基于特定條件發(fā)生后被調用執(zhí)行而被冠以回調稱號。

嚴格來講,庫函數(shù)里沒有完整的回調函數(shù),只有基于各類事件的弱定義的不具備實際功能的空回調函數(shù),或者是針對各類事件而定義的各種用于調用回調函數(shù)的函數(shù)指針。我們的程序監(jiān)測相應條件或事件往往是有的放矢,當相應事件出現(xiàn)時我們需要做相應的處理,這正是回調函數(shù)要實現(xiàn)的功能,也是其功用所在。

4.2 STM32工程里的回調函數(shù)與中斷函數(shù)有什么區(qū)別?

STM32外設庫里的回調函數(shù)的確多數(shù)時候跟中斷事件及中斷服務程序息息相關,往往在中斷服務程序中基于特定事件調用相應的用戶回調函數(shù)。很多時候,我們完全可以將用戶回調函數(shù)看成中斷函數(shù)的一個調用模塊或延伸。

一個中斷服務程序里可以因不同事件而調用不同的回調函數(shù),即一個中斷服務程序里可能包含多個不同的回調函數(shù)。比方,我們在定時器中斷服務程序里可以涉及多個事件及相應的用戶回調函數(shù),定時器中斷服務程序可能涉及更新事件、不同通道的比較事件或捕獲事件,相應的用戶回調函數(shù)往往因應用場景而異。

當然,回調函數(shù)的調用還可以是中斷事件以外的其他事件觸發(fā)調用,比方可以基于初始化操作來調用相應初始化回調函數(shù)。當然,在庫里對某個外設的初始化可能有些默認操作,但這個默認操作很難是放之四海而皆準的操作,這時我們就得根據(jù)實際應用針對性編寫初始化代碼,即初始化型回調函數(shù)。

4.3 STM32庫函數(shù)里的回調函數(shù)是否可以不用?

STM32庫函數(shù)里的回調機制是庫設計者為了便于軟件框架清晰、減少開發(fā)者工作量等因素事先準備的函數(shù)聲明及接口,用戶使用時只需根據(jù)具體應用編寫相關函數(shù)體。當然,你如果不想理睬這些回調函數(shù)聲明及定義也是可以的,你根據(jù)具體應用自行組織代碼完全可行。

4.4 STM32庫函數(shù)里似乎存在著類似半成品的庫回調函數(shù)?

STM32庫函數(shù)里的確準備了一些包含用戶回調函數(shù)的由庫定義的回調函數(shù),是庫設計者基于各類特定事件而準備的回調函數(shù),它會針對特定事件做一些基本而必要的操作,比方狀態(tài)的檢查、標志監(jiān)測及清除,但它沒有辦法徹底寫完整,因為它無法知道該事件發(fā)生后用戶的真實需求是什么,該如何操作,所以它終究還是需要調用真正的用戶回調函數(shù)。這樣做的目的還是為了給開發(fā)者減少開發(fā)工作量、以及減少出錯等。

我們不妨具體看個實例。下面的回調函數(shù)采樣的指針注冊方式,我們看看UART的DMA傳輸完成中斷里傳輸完成的回調函數(shù)的調用過程。

首先,在UART的DMA啟動函數(shù)HAL_UART_Transmit_ DMA()里有這樣一部分內(nèi)容:

wKgZomUCzxmAenU3AALdRrAC7tY951.png

圖9 外設啟動運行代碼中庫回調函數(shù)的賦值

庫里就DMA傳輸事件準備了幾個回調函數(shù)【傳輸完成、半完成、出錯】,即上圖中紅線標示出來的。其實這幾個回調函數(shù)還不算完整的用戶回調函數(shù),是庫定義的并會做一些在它看來用戶必定需要完成的一些操作,它事先幫助完成,之后才調用最終的戶回調函數(shù)。我們以傳輸完成事件為例來看看,上圖星號所標的函數(shù)。

wKgZomUCzxmAfylfAAQP-ZU9bwQ322.png

圖10 庫回調函數(shù)進一步調回用戶回調函數(shù)

在這個庫定義的UART_DMATransmitCplt()函數(shù)里,它對DMA的傳輸模式做了判斷,如果是Normal模式,就將UART的傳輸數(shù)據(jù)長度設置為0,禁止DMA后續(xù)傳輸功能,使能UART傳輸完成中斷的使能。然后才來調用用戶回調函數(shù)【上圖中箭頭所指】。如果DMA工作在循環(huán)模式,代碼進到UART_DMATransmitCplt()函數(shù)后就直接調用最終的用戶回調函數(shù)。也就說這些庫定義的回調函數(shù)在用戶回調函數(shù)的基礎上做了些必要操作,用戶回調函數(shù)可以看成這類庫回調函數(shù)的子集。

4.5 基于STM32庫來組織用戶回調函數(shù)要注意什么?

前面提過了,用戶回調函數(shù)主要基于初始化事件或中斷事件而組織的代碼。那些中斷事件的回調函數(shù)的調用基本都是在中斷服務程序里發(fā)生的。所以,我們在編寫回調函數(shù)時要結合具體情況靈活地組織代碼。要考慮中斷優(yōu)先級、具體事件響應的實時性等。具體點說,我們在組織回調函數(shù)時,要考慮是否一定要一股腦地全寫在中斷服務程序里,會不會影響別的中斷響應。對于有些不緊急而又耗時的事件響應代碼,可以考慮只在回調函數(shù)里設置相應標志,真正的處理代碼放到主循環(huán)去完成。

還提醒一點,STM32庫設計者主動給我們準備了弱定義回調函數(shù)或基于各個事件的回調函數(shù)指針,盡管很豐富了,但未必能包羅萬象,必要時我們可能還得根據(jù)具體情況來額外組織些類似回調函數(shù)的事件/中斷響應代碼。

關于STM32 HAL庫里的回調函數(shù)就簡單介紹到這里,希望能幫到一些STM32開發(fā)者。

完整內(nèi)容請點擊“閱讀原文”下載原文檔。

wKgZomUCzxmARlaPAAOyOP2Y2vs804.png訂閱號

關注STM32

wKgZomUCzxmAEEnTAACDSIYrXK4469.jpg視頻

wKgZomUCzxmAZ1vaAAAfRB2s2NQ304.pngB站賬號

點擊“閱讀原文”,可下載原文檔


原文標題:應用筆記 | 淺談STM32庫里的回調函數(shù)

文章出處:【微信公眾號:STM32單片機】歡迎添加關注!文章轉載請注明出處。


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

    關注

    6023

    文章

    44376

    瀏覽量

    628342
  • STM32
    +關注

    關注

    2257

    文章

    10828

    瀏覽量

    352459

原文標題:應用筆記 | 淺談STM32庫里的回調函數(shù)

文章出處:【微信號:STM32_STM8_MCU,微信公眾號:STM32單片機】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

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

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

    esp8266怎么找到調函數(shù)被調用的地方?

    esp8266里的程序怎么運行? 在user_init里注冊了espconn_regist_sentcb,espconn_regist_recvcb這幾個調函數(shù),怎么找到這幾個
    發(fā)表于 07-10 08:24

    芯海通用 MCU應用筆記 :在 IAR 及 MDK 開發(fā)環(huán)境下使用 printf 函數(shù)重定向移植差異指南

    本應用筆記,旨在幫助客戶在使用不同 IDE(MDK Keil 或 IAR)時,對使用 printf 函數(shù)來打印輸出UART 串口信息時遇到的無法打印、打印亂碼等移植問題作出梳理,理清不同 IDE
    發(fā)表于 05-16 11:56

    stm32中斷調函數(shù)改變的變量在while中不變怎么解決?

    關于stm32 中斷調函數(shù)改變的變量在while中不變問題
    發(fā)表于 04-26 07:44

    在uCGUI的調函數(shù)里加了行代碼,stm32無法啟動怎么解決?

    我是一個初學者。最近在寫一個uCGUI上的小應用的時候遇到了這樣的問題,我在MULTIEDIT控件的調函數(shù)里加了一行代碼,運行我自己寫的一個用于計算的函數(shù),在沒有加這一行代碼的時候運
    發(fā)表于 04-24 07:06

    STM32H750VBT6 ADC1,ADC2,ADC3,加DMA為什么ADC3調函數(shù)不能正常工作?

    STM32H750VBT6ADC1,ADC2,ADC3,加DMA為什么ADC3調函數(shù)不能正常工作 ADC1,ADC2
    發(fā)表于 03-29 07:23

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

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

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

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

    函數(shù)指針與調函數(shù)的應用實例

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

    ??嵌入式中調函數(shù)的實現(xiàn)方法

    調函數(shù)的命名規(guī)范沒有固定的標準,但是根據(jù)通用慣例和編碼規(guī)范,調函數(shù)的命名應該能夠反映
    發(fā)表于 03-04 14:49 ?518次閱讀

    函數(shù)指針的五大作用

    1,調函數(shù)機制函數(shù)指針常用于實現(xiàn)調函數(shù)。通過將
    的頭像 發(fā)表于 12-06 08:00 ?1562次閱讀
    <b class='flag-5'>函數(shù)</b>指針的五大作用

    一文解析BLE觀察者模式調機制

    nRF5 SDK從版本14開始,對事件調機制做了更新,引入了觀察者模式,以解耦不同BLE Layer對BLE事件的調函數(shù)。
    的頭像 發(fā)表于 11-27 10:07 ?775次閱讀
    一文解析BLE觀察者模式<b class='flag-5'>回</b><b class='flag-5'>調</b>機制

    hal庫的中斷調函數(shù)可以重入嗎?

    能不能實現(xiàn)中斷調函數(shù)的重入呢
    發(fā)表于 10-15 11:32

    AN2784應用筆記

    這個應用筆記說明了如何使用大容量的STM32F10xxx的FSMC(靈活的靜態(tài)存儲器控制器)驅動一組外部的存儲器。文中首先簡要地介紹了STM32F10xxx的FSMC控制器,然后給出了包含典型
    發(fā)表于 10-10 07:45

    用筆記 | STM32U575/585 MCU 硬件開發(fā)入門

    用筆記 | STM32U575/585 MCU 硬件開發(fā)入門
    的頭像 發(fā)表于 09-28 18:13 ?3141次閱讀
    應<b class='flag-5'>用筆記</b> | <b class='flag-5'>STM32</b>U575/585 MCU 硬件開發(fā)入門