Stm32中的assert_param()函数

大家在用stm32库函数的时候几乎都会发现assert_param()这个函数,这个函数是判断参数有没有错误,具体是什么错误呢,我会在后面贴图的。
 
assert_param()这个函数在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);
 
这个函数是使能GPIOB和GPIOE端口的时钟的。GPIOB和GPIOE是属于APB2外设的所以用函数RCC_APB2PeriphClockCmd ( xxx , xxx) 来使能,我们在进RCC_APB2PeriphClockCmd ( xxx , xxx)函数里面看看:
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;
}
}
 
在这个函数里面就可以看见我们今天要学习的函数了,在这里我们就分析函数1,以点带面
1、assert_param(IS_RCC_APB2_PERIPH(RCC_APB2Periph));
2、assert_param(IS_FUNCTIONAL_STATE(NewState));
 
函数1里面的参数是IS_RCC_APB2_PERIPH(RCC_APB2Periph),我们在进入这个函数里面看看:
这是一个宏定义
#define   IS_RCC_APB2_PERIPH(PERIPH)   ((((PERIPH) & 0xFFC00002) == 0x00) && ((PERIPH) != 0x00))
我们来计算一下,PERIPH   是我们传递的参数   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),这是为什么呢,是因为用户在代码调试阶段很可能会输入错误的参数而出现错误,一般这种错误不会察觉到,所以当我们想检查这种错误的时候就定义USE_FULL_ASSERT,当我们完成工程项目开始投入生产以后,代码肯定是没有这方面的错误了,我们不需要程序在检查我们的参数了我们就不用定义USE_FULL_ASSERT。这样我们的代码会小一点。
所以我们在stm32f10x_conf.h中打开注释,这样就会检查我们代码中要求检查的参数了,当我们参数没有错误时,就相当于assert_param(1);,进入函数assert_param();观察,就会执行(void)0,意思就是什么也不执行,因为此时参数没有错误。
 
我们现在需要制造出一个错误,在观察函数是怎么执行的。
 
现在我们传递一个错误的参数
RCC_APB2PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE);
我们在计算一下,RCC_APB1Periph_TIM3 = 0x00000002
(PERIPH) & 0xFFC00002) != 0x00
所以IS_RCC_APB2_PERIPH(PERIPH)返回0
所以相当于assert_param(0);
进入assert_param();函数观察
#define assert_param(expr) ((expr) ? (void)0 : assert_failed((uint8_t *)__FILE__, __LINE__))
就会执行这个函数assert_failed((uint8_t *)__FILE__, __LINE__),我们解释一下这个函数是干什么的,这个函数是用户自己发挥的,想在这个函数里面干什么就干什么,它的原型程序里面没有,总之我没找到,后来我在网上看见一文章,他说他在官方例子main.c里面找到的,所以我就拷贝到我的main.c中,函数原型如下:
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 %d\r\n", file, line) 
  //用户可以在这里添加错误信息:比如打印出出错的文件名和行号
  // Infinite loop 
  while (1)
  {
  }
}
 
这个函数就是在程序运行时,可以打印出我们参数错误的文件和行号,我把这个函数修改为:
void assert_failed(u8* file, u32 line) 
printf("Wrong parameters value: file %s on line %d\r\n", file, line);
}
你也可以添加别的函数,总之能显示错误信息就ok,不一定用串口显示。
如果参数有错误的话,就会向串口打印出错误处的文件和行号,看下图是串口打印出的错误信息:
 
 
错误文件为stm32f10x_rcc.c  行号1098,我们在看一下程序是不是这里出现了参数错误,如下图:

还真是这里出现了参数错误,因为我们传递的参数是
RCC_APB2PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE);
 
这里我就基本介绍完了,下面在介绍几点。
 
1、assert_failed((uint8_t *)__FILE__, __LINE__)这个函数的__FILE__, __LINE__形参,可能是c语言自带的把,具体是怎么获取错误参数的文件和行号我也不知道。
 
2、加入我们传递的参数是RCC_APB2PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE);   虽然RCC_APB1Periph_TIM2是属于APB1的,则在程序运行时是打印不出错误参数的,因为RCC_APB1Periph_TIM2 和 RCC_APB2Periph_AFIO 都等于0x00000001,大家可以去stm32f10x_rcc.h文件去看。所以大家在使用参数检查功能时还要自己注意一下参数有没有错误。官方这个查错功能不是万能的。
  • 3
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值