STM32 MDK編程中__weak的使用及避坑
STM32 HAL庫中有很多使用__weak修飾的函數(shù)。比如在"stm32f4xx_hal_spi.c"中有一處函數(shù)定義:
本文引用地址:http://www.biyoush.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 */}
此處的函數(shù)修飾符__weak其實(shí)是一個(gè)宏定義,定義如下:
#define __weak __attribute__((weak))
__attribute__((weak))用于告訴鏈接器此處的函數(shù)為弱引用,這樣一來,如果在其地方有同樣的函數(shù)定義,則鏈接器會(huì)選擇沒用__attribute__((weak))修飾的那個(gè)函數(shù)來鏈接。這個(gè)特性對(duì)于需要使用回調(diào)函數(shù)的地方非常友好,可以在函數(shù)庫中用__weak實(shí)現(xiàn)一個(gè)最小化的回調(diào)函數(shù),這樣用戶可以根據(jù)是否需要回調(diào)而決定是否自己實(shí)現(xiàn)回調(diào)函數(shù),在用戶不需要實(shí)現(xiàn)自己的回調(diào)函數(shù)時(shí),也不會(huì)因?yàn)槿鄙俸瘮?shù)定義而報(bào)錯(cuò)。
在使用__weak時(shí),遇到過2個(gè)坑,下面給大家一個(gè)參考。
1.使用__weak和不使用__weak修飾的函數(shù)不能放在同一個(gè)源文件中,否則會(huì)報(bào)函數(shù)重復(fù)定義的錯(cuò)誤。不過這個(gè)也好理解,因?yàn)開_weak是給鏈接器做指示用的而非編譯器 。當(dāng)二者存在于同一源文件中,編譯器會(huì)報(bào)錯(cuò)。
2.在__weak修飾的函數(shù)中,不能使用while(1)來阻塞程序。如果使用了while(1),編譯能通過,但在存在非__weak修飾的函數(shù)情況下,程序也依舊無法繼續(xù)向下執(zhí)行。這應(yīng)該是MDK的一個(gè)bug。解決方法是先定義一個(gè)值為1的局部變量,然后while這個(gè)局部變量。
uint8_t tmp = 1;while(tmp);
評(píng)論