我們在學STM32的時候函數(shù)assert_param出現(xiàn)的幾率非常大,上網搜索一下,網上一般解釋斷言機制,做為程序開發(fā)調試階段時使用。下面我就談一下我對這些應用的看法,學習東西抱著知其然也要知其所以然。
4 斷言機制函數(shù)assert_param
我們在分析庫函數(shù)的時候,幾乎每一個函數(shù)的原型有這個函數(shù)assert_param();下面以assert_param(IS_GPIO_ALL_PERIPH(GPIOx));為例說一下我的理解,函數(shù)的參數(shù)IS_GPIO_ALL_PERIPH(GPIOx),我們可以尋找到原型
#define IS_GPIO_ALL_PERIPH(PERIPH) (((*(uint32_t*)&(PERIPH)) == GPIOA_BASE)|| \
((*(uint32_t*)&(PERIPH)) == GPIOB_BASE) || \
((*(uint32_t*)&(PERIPH)) == GPIOC_BASE) || \
((*(uint32_t*)&(PERIPH)) == GPIOD_BASE) || \
((*(uint32_t*)&(PERIPH)) == GPIOE_BASE) || \
((*(uint32_t*)&(PERIPH)) == GPIOF_BASE) || \
((*(uint32_t*)&(PERIPH)) == GPIOG_BASE))
這個宏定義的作用就是檢查參數(shù)PERIPH,判斷參數(shù)PERIPH是否為GPIOX(A.。.G)基址中的一個,只要有一個為真則其值為真,否則為假,不用多說,這是C語言中基本的邏輯運算。當然這個庫函數(shù)也用的很有意思,看:首先對PERIPH進行取址,也就是求地址,&PERIPH,然后對這個地址強制轉化為32位的指針,即前面加(uint32_t *),然后通過*進行訪問這個地址(指針)中的內容。不多說了,看幾遍就能明白。
下面我們再回到assert_param這個函數(shù),這個函數(shù)是哪里的呢?在stm32f10x_conf.h尋找到原型如下:
#ifdef USE_FULL_ASSERT
#define assert_param(expr) ((expr) ? (void)0 : assert_failed((uint8_t*)__FILE__, __LINE__))
void assert_failed(uint8_t* file, uint32_t line);
#else
#define assert_param(expr) ((void)0)
#endif
這是一個預編譯文件,若是定義了USE_FULL_ASSERT這個文件,則執(zhí)行后面的文件,我們在程序中一般都沒什么定義,即執(zhí)行后面這個語句((void)0),這個語句不用多想,沒有定義USE_FULL_ASSERT就是什么也不執(zhí)行。說的明白點,對上面的那個語句IS_GPIO_ALL_PERIPH(GPIOx)不執(zhí)行任何操作。
若是定義了USE_FULL_ASSERT它,我們調用這個函數(shù)assert_param時,及對參數(shù)IS_GPIO_ALL_PERIPH(GPIOx)的正確性進行檢查,通過一個C語言中的雙目運算符來判斷,若是返回1,執(zhí)行語句(void)0,跟上面一樣,若是返回0,則執(zhí)行后面的函數(shù)assert_failed((uint8_t *)__FILE__,__LINE__),函數(shù)的作用在庫函數(shù)中有解釋,用來指示出錯的行數(shù)和文件。注意:__FILE__,__LINE__是標準庫函數(shù)中的宏定義!切記
void assert_failed(uint8_t* file, uint32_t line);剛開始沒看明白為什么加在這里,仔細一想是在頭文件的函數(shù)聲明。至于函數(shù)實體呢?我們從官方文件的模板中main.c中可以找到。如下:
void assert_failed(u8* file, u32 line)
{ /* User can add his own implementation to report the file name and linenumber,
ex: printf(“Wrong parameters value: file %s on line %d\r\n”, file,line) */
/* Infinite loop */
while (1) { }
} 英文注釋也說明了怎么應用,通過輸入參數(shù)來確定位置,最簡單的方法就是串口打印了,這個函數(shù)的主要思想是在輸入參數(shù)有問題的時候,但是有編譯不出來,它可以幫你檢查參數(shù)的有效性,好處不必多言,自己領悟就行。
繼續(xù)說明如下: assert_param是怎樣包含進去的呢?我們在stm32f10x_conf.h這個頭文件中定義的函數(shù)聲明還是宏定義,怎么在其它文件中都能應用呢?也很多網上朋友在剛開始學習的時候都遇到編譯不過去的問題出現(xiàn),最后通過在文件中添加USE_STDPERIPH_DRIVER來解決的:
我們可以在整個工程中進行搜索USE_STDPERIPH_DRIVER,通過頭文件可以看出,是使用標準外設文件。在stm32f10x.h文件中我們可以搜索到如下情況:
#if !defined USE_STDPERIPH_DRIVER
/**
* @brief Comment the line below if you will not use the peripherals drivers.
In this case, these drivers will not be included and the application code will
be based on direct access to peripherals registers
*/
#define USE_STDPERIPH_DRIVER
#endif
#ifdef USE_STDPERIPH_DRIVER
#include “stm32f10x_conf.h”
#endif
可以很容易看出來,我們不在那里添加,這個頭文件中也給我們設置了開關,只要把第一個的注釋去掉,就不用在配置中添加USE_STDPERIPH_DRIVER了,在第二個文件中我們可以知道怎樣包含這個控制開關文件了,呵呵。我們也明白為什么我們在寫程序的時候只要包含stm32f10x.h就能很容易的包含所有的文件文件了吧,我們只要在stm32f10x_conf.h配置一下就能包含所需要的庫文件了。
通過以上可以看出,通過頭文件的相互包含,來控制外設以及調試文件的調用,這樣我們理清思路,理解起來就好多了。當然在學習中可能有些C語言問題還沒有理解透徹,多上網搜一下,或者多看書,很快就搞明白的。
PS 2:
STM32中assert_param的使用
在STM32的固件庫和提供的例程中,到處都可以見到assert_param()的使用。如果打開任何一個例程中的stm32f10x_conf.h文件,就可以看到實際上assert_param是一個宏定義;
在固件庫中,它的作用就是檢測傳遞給函數(shù)的參數(shù)是否是有效的參數(shù)。
所謂有效的參數(shù)是指滿足規(guī)定范圍的參數(shù),比如某個參數(shù)的取值范圍只能是小于3的正整數(shù),如果給出的參數(shù)大于3,
則這個assert_param()可以在運行的程序調用到這個函數(shù)時報告錯誤,使程序員可以及時發(fā)現(xiàn)錯誤,而不必等到程序運行結果的錯誤而大費周折。
這是一種常見的軟件技術,可以在調試階段幫助程序員快速地排除那些明顯的錯誤。
它確實在程序的運行上犧牲了效率(但只是在調試階段),但在項目的開發(fā)上卻幫助你提高了效率。
當你的項目開發(fā)成功,使用release模式編譯之后,或在stm32f10x_conf.h文件中注釋掉對USE_FULL_ASSERT的宏定義,所有的assert_param()檢驗都消失了,不會影響最終程序的運行效率。
#define assert_param(expr) ((expr) ? (void)0 : assert_failed((u8 *)__FILE__, __LINE__))
。。。
assert_param(IS_ADC_ALL_PERIPH(ADCx));
。。。
在執(zhí)行assert_param()的檢驗時,如果發(fā)現(xiàn)參數(shù)出錯,它會調用函數(shù)assert_failed()向程序員報告錯誤,在任何一個例程中的main.c中都有這個函數(shù)的模板,如下:
void assert_failed(uint8_t* file, uint32_t line)
{
while (1)
{}
}
你可以按照自己使用的環(huán)境需求,添加適當?shù)恼Z句輸出錯誤的信息提示,或修改這個函數(shù)做出適當?shù)腻e誤處理。
1、STM32F10xD.LIB是DEBUG模式的庫庫文件。
2、STM32F10xR.LIB是Release模式的庫庫文件。
3、要選擇DEBUG和RELEASE模式,需要修改stm32f10x_conf.h的內容。
#define DEBUG 表示DEBUG模式,把該語句注釋掉,則為RELEASE模式。
4、要選擇DEBUG和RELEASE模式,也可以在Options,C/C++,Define里填入DEBUG的預定義。
這樣,就不需要修改stm32f10x_conf.h的內容。
5、如果把庫加入項目,則不需要將ST的庫源文件加入項目,比較方便。
但是,庫的選擇要和DEBUG預定義對應。
評論
查看更多