STM32 MDK編程中__weak的使用及避坑
STM32 HAL庫中有很多使用__weak修飾的函數。比如在"stm32f4xx_hal_spi.c"中有一處函數定義:
本文引用地址:http://www.ex-cimer.com/article/202312/453925.htm/** * @brief Initialize the SPI MSP. * @param hspi pointer to a SPI_HandleTypeDef structure that contains * the configuration information for SPI module. * @retval None */__weak void HAL_SPI_MspInit(SPI_HandleTypeDef *hspi){ /* Prevent unused argument(s) compilation warning */ UNUSED(hspi); /* NOTE : This function should not be modified, when the callback is needed, the HAL_SPI_MspInit should be implemented in the user file */}
此處的函數修飾符__weak其實是一個宏定義,定義如下:
#define __weak __attribute__((weak))
__attribute__((weak))用于告訴鏈接器此處的函數為弱引用,這樣一來,如果在其地方有同樣的函數定義,則鏈接器會選擇沒用__attribute__((weak))修飾的那個函數來鏈接。這個特性對于需要使用回調函數的地方非常友好,可以在函數庫中用__weak實現一個最小化的回調函數,這樣用戶可以根據是否需要回調而決定是否自己實現回調函數,在用戶不需要實現自己的回調函數時,也不會因為缺少函數定義而報錯。
在使用__weak時,遇到過2個坑,下面給大家一個參考。
1.使用__weak和不使用__weak修飾的函數不能放在同一個源文件中,否則會報函數重復定義的錯誤。不過這個也好理解,因為__weak是給鏈接器做指示用的而非編譯器 。當二者存在于同一源文件中,編譯器會報錯。
2.在__weak修飾的函數中,不能使用while(1)來阻塞程序。如果使用了while(1),編譯能通過,但在存在非__weak修飾的函數情況下,程序也依舊無法繼續向下執行。這應該是MDK的一個bug。解決方法是先定義一個值為1的局部變量,然后while這個局部變量。
uint8_t tmp = 1;while(tmp);
評論