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