當前位置:首頁 > 單片機 > 單片機
[導讀]大家在用stm32庫函數(shù)的時候幾乎都會發(fā)現(xiàn)assert_param()這個函數(shù),這個函數(shù)是判斷參數(shù)有沒有錯誤,具體是什么錯誤呢,我會在后面貼圖的。assert_param()這個函數(shù)在stm32f10x_conf.h中定義:#ifdef USE_FULL_ASSERT#de

大家在用stm32庫函數(shù)的時候幾乎都會發(fā)現(xiàn)assert_param()這個函數(shù),這個函數(shù)是判斷參數(shù)有沒有錯誤,具體是什么錯誤呢,我會在后面貼圖的。


assert_param()這個函數(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

以上代碼就是stm32f10x_conf.h中的一部分,我們再看下面的代碼:


RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB|RCC_APB2Periph_GPIOE, ENABLE);

這個函數(shù)是使能GPIOB和GPIOE端口的時鐘的。GPIOB和GPIOE是屬于APB2外設(shè)的所以用函數(shù)RCC_APB2PeriphClockCmd( xxx , xxx) 來使能,我們在進RCC_APB2PeriphClockCmd( xxx , xxx)函數(shù)里面看看:

void RCC_APB2PeriphClockCmd(uint32_t RCC_APB2Periph, FunctionalState NewState)

{

assert_param(IS_RCC_APB2_PERIPH(RCC_APB2Periph));

assert_param(IS_FUNCTIONAL_STATE(NewState));

if (NewState != DISABLE)

{

RCC->APB2ENR |= RCC_APB2Periph;

}

else

{

RCC->APB2ENR &= ~RCC_APB2Periph;

}

}

在這個函數(shù)里面就可以看見我們今天要學習的函數(shù)了,在這里我們就分析函數(shù)1,以點帶面

1、assert_param(IS_RCC_APB2_PERIPH(RCC_APB2Periph));

2、assert_param(IS_FUNCTIONAL_STATE(NewState));

函數(shù)1里面的參數(shù)是IS_RCC_APB2_PERIPH(RCC_APB2Periph),我們在進入這個函數(shù)里面看看:

這是一個宏定義

#define IS_RCC_APB2_PERIPH(PERIPH) ((((PERIPH) & 0xFFC00002) == 0x00) && ((PERIPH) != 0x00))

我們來計算一下,PERIPH 是我們傳遞的參數(shù) RCC_APB2Periph_GPIOB|RCC_APB2Periph_GPIOE,

RCC_APB2Periph_GPIOB|RCC_APB2Periph_GPIOE=0x00000008|0x00000040

(PERIPH) & 0xFFC00002 = 0x00,所以((((PERIPH) & 0xFFC00002) == 0x00) && ((PERIPH) != 0x00)) = 1

所以就相當于assert_param(1);

我們反過來在看stm32f10x_conf.h代碼的那部分,如果定義USE_FULL_ASSERT,那么就會定義

#define assert_param(expr) ((expr) ? (void)0: assert_failed((uint8_t *)__FILE__, __LINE__))

如果沒定義USE_FULL_ASSERT,那么就會定義#define assert_param(expr) ((void)0),這是為什么呢,是因為用戶在代碼調(diào)試階段很可能會輸入錯誤的參數(shù)而出現(xiàn)錯誤,一般這種錯誤不會察覺到,所以當我們想檢查這種錯誤的時候就定義USE_FULL_ASSERT,當我們完成工程項目開始投入生產(chǎn)以后,代碼肯定是沒有這方面的錯誤了,我們不需要程序在檢查我們的參數(shù)了我們就不用定義USE_FULL_ASSERT。這樣我們的代碼會小一點。

所以我們在stm32f10x_conf.h中打開注釋,這樣就會檢查我們代碼中要求檢查的參數(shù)了,當我們參數(shù)沒有錯誤時,就相當于assert_param(1);,進入函數(shù)assert_param();觀察,就會執(zhí)行(void)0,意思就是什么也不執(zhí)行,因為此時參數(shù)沒有錯誤。

我們現(xiàn)在需要制造出一個錯誤,在觀察函數(shù)是怎么執(zhí)行的。

現(xiàn)在我們傳遞一個錯誤的參數(shù)

RCC_APB2PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE);

我們在計算一下,RCC_APB1Periph_TIM3=0x00000002

(PERIPH) & 0xFFC00002) != 0x00

所以IS_RCC_APB2_PERIPH(PERIPH)返回0

所以相當于assert_param(0);

進入assert_param();函數(shù)觀察

#define assert_param(expr) ((expr) ? (void)0 : assert_failed((uint8_t *)__FILE__, __LINE__))

就會執(zhí)行這個函數(shù)assert_failed((uint8_t *)__FILE__, __LINE__),我們解釋一下這個函數(shù)是干什么的,這個函數(shù)是用戶自己發(fā)揮的,想在這個函數(shù)里面干什么就干什么,它的原型程序里面沒有,總之我沒找到,后來我在網(wǎng)上看見一文章,他說他在官方例子main.c里面找到的,所以我就拷貝到我的main.c中,函數(shù)原型如下:

void assert_failed(u8* file, u32 line)
{
//User can add his own implementation to report the file name and line number,

// ex: printf("Wrong parameters value: file %s on line %drn", file, line)
//用戶可以在這里添加錯誤信息:比如打印出出錯的文件名和行號

// Infinite loop
while (1)
{
}
}


這個函數(shù)就是在程序運行時,可以打印出我們參數(shù)錯誤的文件和行號,我把這個函數(shù)修改為:

void assert_failed(u8* file, u32 line)

{

printf("Wrong parametersvalue: file %s on line %drn", file, line);

}

你也可以添加別的函數(shù),總之能顯示錯誤信息就ok,不一定用串口顯示。

如果參數(shù)有錯誤的話,就會向串口打印出錯誤處的文件和行號,看下圖是串口打印出的錯誤信息:


錯誤文件為stm32f10x_rcc.c 行號1098,我們在看一下程序是不是這里出現(xiàn)了參數(shù)錯誤,如下圖:


還真是這里出現(xiàn)了參數(shù)錯誤,因為我們傳遞的參數(shù)是

RCC_APB2PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE);


這里我就基本介紹完了,下面在介紹幾點。


1、assert_failed((uint8_t *)__FILE__, __LINE__)這個函數(shù)的__FILE__, __LINE__形參,可能是c語言自帶的把,具體是怎么獲取錯誤參數(shù)的文件和行號我也不知道。


2、加入我們傳遞的參數(shù)是RCC_APB2PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE);雖然RCC_APB1Periph_TIM2是屬于APB1的,則在程序運行時是打印不出錯誤參數(shù)的,因為RCC_APB1Periph_TIM2和RCC_APB2Periph_AFIO都等于0x00000001,大家可以去stm32f10x_rcc.h文件去看。所以大家在使用參數(shù)檢查功能時還要自己注意一下參數(shù)有沒有錯誤。官方這個查錯功能不是萬能的。


本站聲明: 本文章由作者或相關(guān)機構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點,本站亦不保證或承諾內(nèi)容真實性等。需要轉(zhuǎn)載請聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請及時聯(lián)系本站刪除。
換一批
延伸閱讀

9月2日消息,不造車的華為或?qū)⒋呱龈蟮莫毥谦F公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關(guān)鍵字: 阿維塔 塞力斯 華為

加利福尼亞州圣克拉拉縣2024年8月30日 /美通社/ -- 數(shù)字化轉(zhuǎn)型技術(shù)解決方案公司Trianz今天宣布,該公司與Amazon Web Services (AWS)簽訂了...

關(guān)鍵字: AWS AN BSP 數(shù)字化

倫敦2024年8月29日 /美通社/ -- 英國汽車技術(shù)公司SODA.Auto推出其旗艦產(chǎn)品SODA V,這是全球首款涵蓋汽車工程師從創(chuàng)意到認證的所有需求的工具,可用于創(chuàng)建軟件定義汽車。 SODA V工具的開發(fā)耗時1.5...

關(guān)鍵字: 汽車 人工智能 智能驅(qū)動 BSP

北京2024年8月28日 /美通社/ -- 越來越多用戶希望企業(yè)業(yè)務能7×24不間斷運行,同時企業(yè)卻面臨越來越多業(yè)務中斷的風險,如企業(yè)系統(tǒng)復雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務連續(xù)性,提升韌性,成...

關(guān)鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據(jù)媒體報道,騰訊和網(wǎng)易近期正在縮減他們對日本游戲市場的投資。

關(guān)鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會開幕式在貴陽舉行,華為董事、質(zhì)量流程IT總裁陶景文發(fā)表了演講。

關(guān)鍵字: 華為 12nm EDA 半導體

8月28日消息,在2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會上,華為常務董事、華為云CEO張平安發(fā)表演講稱,數(shù)字世界的話語權(quán)最終是由生態(tài)的繁榮決定的。

關(guān)鍵字: 華為 12nm 手機 衛(wèi)星通信

要點: 有效應對環(huán)境變化,經(jīng)營業(yè)績穩(wěn)中有升 落實提質(zhì)增效舉措,毛利潤率延續(xù)升勢 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務引領(lǐng)增長 以科技創(chuàng)新為引領(lǐng),提升企業(yè)核心競爭力 堅持高質(zhì)量發(fā)展策略,塑強核心競爭優(yōu)勢...

關(guān)鍵字: 通信 BSP 電信運營商 數(shù)字經(jīng)濟

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺與中國電影電視技術(shù)學會聯(lián)合牽頭組建的NVI技術(shù)創(chuàng)新聯(lián)盟在BIRTV2024超高清全產(chǎn)業(yè)鏈發(fā)展研討會上宣布正式成立。 活動現(xiàn)場 NVI技術(shù)創(chuàng)新聯(lián)...

關(guān)鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會上,軟通動力信息技術(shù)(集團)股份有限公司(以下簡稱"軟通動力")與長三角投資(上海)有限...

關(guān)鍵字: BSP 信息技術(shù)
關(guān)閉
關(guān)閉