STM32F4 USART1 无法正常工作并返回垃圾数据

Posted

技术标签:

【中文标题】STM32F4 USART1 无法正常工作并返回垃圾数据【英文标题】:STM32F4 USART1 doesn't work correctly and return garbage data 【发布时间】:2019-01-23 23:31:12 【问题描述】:

我正在尝试使用 STM32F407 USART1 外设,但它无法正常工作。我已经阅读了很多次数据表,但我找不到任何解决方案。

我使用 FTDI232 进行通信。我的连接是正确的,我在 MSP430 上使用这个连接。在代码中,我暂时不使用 DMA,因为我目前不知道如何配置 DMA(用于多重通信)。

FTDI232 RX Pin ---> PA9 (STM32407 USART1 TX)
FTDI232 TX Pin ---> PA10 (STM32407 USART1 RX)
FTDI232 Ground ---> STM32F407 Ground

我为几乎所有行添加了 cmets。 我的错误在哪里?你能帮帮我吗?

#include "stm32f4xx.h"                  // Device header
#include "stm32f4xx_hal.h"              // Keil::Device:STM32Cube HAL:Common

/************************************************************ 
    CPU  Frequency  168Mhz
    AHB  Frequency  168MHz
    APB1 Frequency  42MHz
    APB2 Frequency  84MHz
************************************************************/

/************************* PLL Parameters *************************************/
/* PLL_VCO = (HSE_VALUE or HSI_VALUE / PLL_M) * PLL_N */
#define PLL_M      8
#define PLL_N      336

/* SYSCLK = PLL_VCO / PLL_P */
#define PLL_P      2

/* USB OTG FS, SDIO and RNG Clock =  PLL_VCO / PLLQ */
#define PLL_Q      7

/******************************************************************************/

#define CPU_PLL_CLK             168000000UL         // 168 MHz
#define AHB_BUS_CLK             168000000UL         // 168 MHz
#define APB1_BUS_CLK            42000000UL          // 42  MHz
#define APB2_BUS_CLK            84000000UL          // 84  MHz

// USART1 Functions
void            USART1_Init(uint32_t baudrate);
void            USART1_WriteChar(uint8_t data);

// Delay with MS
// Every Cycle = 5.95 ns (for 168 MHz)
void delay_ms(uint32_t delay_time)
   
    delay_time *= (CPU_PLL_CLK / 1000) / 5.95;
    while(delay_time--);


/* PLL_VCO = (HSE_VALUE or HSI_VALUE / PLL_M) * PLL_N
     SYSCLK = PLL_VCO / PLL_P
*/ 
void System_Config(void)


    RCC->CFGR        = 0x00000000;                  // CFGR All Cleared - System Clock HSI
    RCC->CFGR           |=  (4    << 10);           // PPRE1 - APB1 - AHB clock divided by 2
    RCC->CFGR           |=  (4    << 13);           // PPRE2 - APB2 - AHB clock divided by 2
    RCC->CR             |=  (1    << 16);           // HSEON - HSE External Oscillator

    while (!(RCC->CR & 0x00020000));        // HSERDY - Wait HSE active

    RCC->PLLCFGR    |=  (8      <<  0);         // PLLM = 8;
    RCC->PLLCFGR    |=  (336    <<  6);         // PLLN = 336;
    RCC->PLLCFGR    &= ~(3    << 16);           // PLLP = 2;    For 2, write 00
    RCC->PLLCFGR    |=  (7      << 24);         // PLLQ = 7;

    RCC->CR             |=  (1    << 24);           // PLLON - PLL Active

    while (!(RCC->CR & 0x02000000));        // PLLRDY - Wait PLL active

    FLASH->ACR      |=  (5    <<  0);           // Wait State = 5
    FLASH->ACR      |=  (1    <<  9);           // Data Cache active
    FLASH->ACR      |=  (1    << 10);           // Instruction Cache active

    RCC->CFGR       |=  (2    <<  0);     // System Clock PLL

    while ((RCC->CFGR & 0x0000000F) != 0x0000000A); // Wait until load



int main(void)


    System_Config();
    USART1_Init(115200);        // Parameter Not Used Now

    while(1)
    
        USART1_WriteChar('a');
        delay_ms(1000);
    

    return 0;


/*
    Tx/Rx baud = fCLK / (8 * (2 - OVER8) * USARTDIV)
                                OVER8 -> CR1 Register 15. Bit (Over Sampling)
                                fCLK = SystemClock
    8000000 / (16 * 9600)
*/
void USART1_Init(uint32_t baudrate)


    /* GPIOA clock enable */
    RCC->AHB1ENR        |= (1UL << 0);                                              //  RCC_AHB1ENR_GPIOAEN;

    // AFR[0] = AFRL Register   - Address 0x20
    // AFR[1] = AFRH Register   - Address   0x21
    // GPIOA->AFR[1] |= (0x00000110);   

    GPIOA->MODER    |=  (2UL << 18);                                                // GPIO_AFRH_AFRH0 - PA9  - Alternate State 
    GPIOA->MODER    |=  (2UL << 20);                                                // GPIO_AFRH_AFRH1 - PA10 - Alternate State

    GPIOA->PUPDR &= ~((3 << 18) | (3 << 20));                               // No Pull UP/DOWN
    GPIOA->PUPDR |=   (1 << 18) | (1 << 20);                                // Pull UP

    GPIOA->AFR[1]   &= ~(15UL << 4);                                                // Clear AF Mode
    GPIOA->AFR[1]   &= ~(15UL << 8);                                                // Clear AF Mode
    GPIOA->AFR[1]   |=  (7UL << 4);                                                 // GPIO_AFRH_AFRH0 - PA9
    GPIOA->AFR[1]   |=  (7UL << 8);                                                 // GPIO_AFRH_AFRH1 - PA10

    // USART1 clock enable
    RCC->APB2ENR        |= (1UL << 4);                                                  // Enable USART1 - RCC_APB2ENR_USART1EN;
    //RCC->APB2RSTR     |= (1UL << 4);  
    //RCC->APB2RSTR     &= ~(1UL << 4);                                             // USART1 Disable Reset Mode    

    USART1->CR2 &= ~(3 << 12);      // USART_STOPBITS_1

    USART1->CR1 &= ~(1 << 12);      //  USART_WORDLENGTH_8B
    USART1->CR1 &= ~(1 << 10);      //  USART_PARITY_NONE
    USART1->CR1 |=  (1 <<  7);      //  USART_CR1_TXEIE
    USART1->CR1 |=  (1 <<  5);      //  USART_CR1_RXNEIE
    USART1->CR1 |=  (1 <<  3);      //  USART_MODE_TX
    USART1->CR1 |=  (1 <<  2);      //  USART_MODE_RX

    //USART1->BRR     = SystemCoreClock / (16 * 115200);            // 115200 baud
    //USART1->BRR  = 84000000 / (16 * 115200);
    USART1->BRR  = 0x2D9;                     // 115200

    /* Enable the USART */
    USART1->CR1 |=  (1 <<  13);     //  USART_CR1_UE

    NVIC->ISER[1] = 1 << (USART1_IRQn - 32);        // USART1 Global Interrupt Enable
    //NVIC->ISER[1]  |= 0x20;                           // USART1 Interrupt Enable      

    //USART1->SR &= ~(0x40);                          // Status Register



void USART1_WriteChar(uint8_t data)

    while(!(USART1->SR & 0x80));      // USART_SR_TXE - USART_FLAG_TXE
    USART1->DR = data;  

【问题讨论】:

你怎么知道它不起作用?您是否在 PC 上运行串行监视器程序?你在用示波器吗?你看到了什么? 我想用 FTDI232 将 STM 中的字符发送到我的 PC 上的串行监视器。我不使用示波器。我的串行监视器上没有看到任何内容。 你检查过你的串口监视器是否工作吗?您可以通过连接 FTDI232 的 TXRX 线来做到这一点。然后通过 FTDI 发送一个字符,您应该会收到它。 是的,我查过了。我的 FTDI 正在工作。我试过用 MSP430,它可以工作,但不能用 STM32 你的中断文件在哪里?你在哪里处理接收中断? 【参考方案1】:

这也可能是由于与您的程序无关的问题。如果您将 Discovery 板与 USART1 一起使用,请注意一些必需的引脚 (PA9 / PA10) 已在使用中。

更多信息here。

【讨论】:

是的,我的问题是探索板的 PA9/PA10 引脚。该引脚与 USB 一起使用。为此,如果要使用 USART1,必须选择 PB6/PB7 作为 Discovery Board。谢谢你:)【参考方案2】:

如果您尝试了@elasticman 指出的解决方案,并且再次获得一堆垃圾数据,那么检查以下建议也很好。借自this链接。

(**) HSE_VALUE 是在 stm32f4xx.h 文件中定义的常量 (默认值 25 MHz),用户必须确保 HSE_VALUE 与 所用晶体的实际频率。否则, 这个函数可能有错误的结果。

STM32F407发现板中使用的晶振为8MHz,因此务必将stm32f4xx.h文件中的HSE_VALUE(默认为25MHz)修正为8MHz,以确保USART正常工作。

【讨论】:

【参考方案3】:

请尝试将 AHB 时钟更改为 64 MHz,然后添加更多预标量以使 APB1 低于 42MHz 我不知道原因,但超过 64MHz 则不起作用。

【讨论】:

【参考方案4】:

我想我有一个解决办法。如果你拉起你的 gpio pin

GPIO_InitStruct.GPIO_PuPd = GPIO_PuPd_UP;    

您可以让 USART1(连接到我板上的 st-link)工作,而不管 ST 实现的外设过载。

(编辑):我看到你已经将 PUPDR 设置为 pullup,抱歉,如果不能解决这个问题,但它确实解决了我的问题。顺便说一句,我正在使用 STM32F429Disc1 板。

【讨论】:

以上是关于STM32F4 USART1 无法正常工作并返回垃圾数据的主要内容,如果未能解决你的问题,请参考以下文章

STM32F4的USB时钟必须工作在48MHZ吗?如果不是48M会有啥问题

STM32F4 SPI1 工作,SPI5 不工作?

STM32F4串口打印 while函数出错

16 . USART 串口通信实验

16 . USART 串口通信实验

16 . USART 串口通信实验