美文网首页
STM32问题集

STM32问题集

作者: 光头不光还亮 | 来源:发表于2020-05-09 14:28 被阅读0次

    1>usart1.c(28): error: #20: identifier "USART_InitTypeDef" is undefined

    想用stm32的串口,报错usart1.c(28): error: #20: identifier "USART_InitTypeDef" is undefined

    但是右键查定义却能找到

    怎么回事?

    重新百修改了STM32的工程代码

    所有提示未定义的代码都已经确定定义过,度但为什么会出现这知样的情况?

    在stm32f10x_conf.h 里面包含stm32f10x_usart.h头文道件的代码被注释掉了,内把这个注释去掉,重新包含这个头文件就把上面容的问题解决了。

    2>assert_param()函数

    我们在学STM32的时候函数assert_param出现的几率非常大,上网搜索一下,网上一般解释断言机制,做为程序开发调试阶段时使用。下面我就谈一下我对这些应用的看法,学习东西抱着知其然也要知其所以然。

    4 断言机制函数assert_param

    我们在分析库函数的时候,几乎每一个函数的原型有这个函数assert_param();下面以assert_param(IS_GPIO_ALL_PERIPH(GPIOx));为例说一下我的理解,函数的参数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))

    这个宏定义的作用就是检查参数PERIPH,判断参数PERIPH是否为GPIOX(A...G)基址中的一个,只要有一个为真则其值为真,否则为假,不用多说,这是C语言中基本的逻辑运算。当然这个库函数也用的很有意思,看:首先对PERIPH进行取址,也就是求地址,&PERIPH,然后对这个地址强制转化为32位的指针,即前面加(uint32_t *),然后通过*进行访问这个地址(指针)中的内容。不多说了,看几遍就能明白。

    下面我们再回到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

    这是一个预编译文件,若是定义了USE_FULL_ASSERT这个文件,则执行后面的文件,我们在程序中一般都没什么定义,即执行后面这个语句((void)0),这个语句不用多想,没有定义USE_FULL_ASSERT就是什么也不执行。说的明白点,对上面的那个语句IS_GPIO_ALL_PERIPH(GPIOx)不执行任何操作。

    若是定义了USE_FULL_ASSERT它,我们调用这个函数assert_param时,及对参数IS_GPIO_ALL_PERIPH(GPIOx)的正确性进行检查,通过一个C语言中的双目运算符来判断,若是返回1,执行语句(void)0,跟上面一样,若是返回0,则执行后面的函数assert_failed((uint8_t *)__FILE__,__LINE__),函数的作用在库函数中有解释,用来指示出错的行数和文件。注意:__FILE__,__LINE__是标准库函数中的宏定义!切记

    void assert_failed(uint8_t* file, uint32_t line);刚开始没看明白为什么加在这里,仔细一想是在头文件的函数声明。至于函数实体呢?我们从官方文件的模板中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) { }

    } 英文注释也说明了怎么应用,通过输入参数来确定位置,最简单的方法就是串口打印了,这个函数的主要思想是在输入参数有问题的时候,但是有编译不出来,它可以帮你检查参数的有效性,好处不必多言,自己领悟就行。

    继续说明如下: assert_param是怎样包含进去的呢?我们在stm32f10x_conf.h这个头文件中定义的函数声明还是宏定义,怎么在其它文件中都能应用呢?也很多网上朋友在刚开始学习的时候都遇到编译不过去的问题出现,最后通过在文件中添加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语言问题还没有理解透彻,多上网搜一下,或者多看书,很快就搞明白的。

    3>开始自己的SPI工作不正常了经常出现匪夷所思的问题,原来是波特率太高了,我的SPI2在APB1上,SPI_Config中设置是这样的:

    SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_2;

       因为ST说SPI最高18Mb,所以就简单的32M/2=18这么设置的,接着就完蛋了,经常出奇怪的毛病。

       现在设置的是SPI_BaudRatePrescaler_8工作就正常了,可是速度是不是就只有36M/8=4Mb了啊??  

    因为你的SPI是双向的,所以每发一个数据,都会收到数据,然后会产生的OVR标志,该标志不清除,所有的数据都不能进来。

    解决的办法是每发送一个字节,就读一个字节,这样防止OVR标志产生,

    4>STM32系统时钟默认设置

    我们一直都说STM32有一个非常复杂的时钟系统,然而在原子或者野火的例程中,只要涉及到时钟,我们却只能看到类似的库函数调用,如RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1|RCC_APB2Periph_GPIOA, ENABLE);这个仅仅只是起到开启挂载在APB2线上的USART1时钟的作用罢了,APB2的时钟频率是多少我们并不知道”

    在加载启动文件(就是那个.s文件)启动的时候已经调用了SystemInit函数,这里就设置了系统时钟,不需要再调用了

    5>为什么开启串口接收中断和读取串口接收中断标志位发生冲突?

    这两个命令出现冲突,如果注释掉“开启串口接收中断”,则可以正常运行,是因为?

    6 >STM32 JTDO JREST复用为普通IO

    一、开启AFIO的时钟(必须保证先打开AFIO,否则无效)

    RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO, ENABLE);

    二、禁用JTAG,使能SWD

    GPIO_PinRemapConfig(GPIO_Remap_SWJ_JTAGDisable , ENABLE);//释放JTAG引脚

    三、打开GPIO的时钟

    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);

    即可。

    JTDO、JTDI、JTRST都可以这样设置。

    相关文章

      网友评论

          本文标题:STM32问题集

          本文链接:https://www.haomeiwen.com/subject/zepxnhtx.html