GD32F103替代STM32F103带来串口不能用的问题

近期,由于需要,先前使用GD32F103的替换STM32F103的芯片,它的系统主频提升到104MHZ, 众所周之,STM32F103的主频是72MHZ,这里没有什么问题,只要改一下SYSTEMCLOCK_CONFIG中的PLL的因子就可以了,原来是9倍,现在换成13倍频,这样,各项功能都正常,包括串口RS485, 但最近,客户要求原来的9600的波特率,改为1200, 这很简单,把波特率在串口的配置函数中修改一下就可以了,我使用了STM CUBEMX来生成的HAL代码库,本想以为就这么简单,但事实上,串口是不能正常执行命令的。换回9600,就能收到。我知道,这可能是串口时钟的原因,仿真一下,串口是能收到中断信号,但收到的数据不对,因此状态机,肯定就不能收到有效命令行了。

串口收到数据错的问题,是因为主频提高了,由72MHZ提到104MHZ,而波特率降低了6分之1,中间的分频可能是溢出了,没有细算。

//RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1;
RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV2;
把这个分频系数改为DIV2, 相当于APB进行了2分频,UART的进钟是取自于APB的,改完成, 1200的波特率命令执行正常。由于我的系统较简单,这个APB时钟分频改了后,可能要评估还会影响其它的硬件资源。
一点小经验,分享带来各位网友,不要在这样的问题上花时间。
以下是修改前后的代码,以供参考。

在这里插入图片描述

  RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV2;
  /* BEGIN: Added by lejianz, 2021/11/20 */
  //RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1;
  RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV2;
void MX_USART2_UART_Init(void)
{

  huart2.Instance = USART2;
  //huart2.Init.BaudRate = 115200;
  //huart2.Init.BaudRate = 9600;
  huart2.Init.BaudRate = 1200;
  huart2.Init.WordLength = UART_WORDLENGTH_8B;
  huart2.Init.StopBits = UART_STOPBITS_1;
  huart2.Init.Parity = UART_PARITY_NONE;
  huart2.Init.Mode = UART_MODE_TX_RX;
  huart2.Init.HwFlowCtl = UART_HWCONTROL_NONE;
  huart2.Init.OverSampling = UART_OVERSAMPLING_16;
  if (HAL_UART_Init(&huart2) != HAL_OK)
  {
    Error_Handler();
  }	
	
void HAL_UART_MspInit(UART_HandleTypeDef* uartHandle)
{

  GPIO_InitTypeDef GPIO_InitStruct;
  if(uartHandle->Instance==USART1)
  {
  /* USER CODE BEGIN USART1_MspInit 0 */

  /* USER CODE END USART1_MspInit 0 */
    /* Peripheral clock enable */
    __HAL_RCC_USART1_CLK_ENABLE();
  
    /**USART1 GPIO Configuration    
    PA9     ------> USART1_TX
    PA10     ------> USART1_RX 
    */
    GPIO_InitStruct.Pin = GPIO_PIN_9;
    GPIO_InitStruct.Mode = GPIO_MODE_AF_PP;
    //GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH;
    GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;
    HAL_GPIO_Init(GPIOA, &GPIO_InitStruct);

    GPIO_InitStruct.Pin = GPIO_PIN_10;
    GPIO_InitStruct.Mode = GPIO_MODE_INPUT;
    GPIO_InitStruct.Pull = GPIO_NOPULL;
    HAL_GPIO_Init(GPIOA, &GPIO_InitStruct);


	// PA8 -----> RS485 Control
    GPIO_InitStruct.Pin = RS485_CTRL_PIN;
    GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP;
    GPIO_InitStruct.Pull = GPIO_NOPULL;
    //GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH;
    GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;
    HAL_GPIO_Init(GPIOA, &GPIO_InitStruct);
	RS485_CTRL_RX();


    /* USART1 interrupt Init */
    HAL_NVIC_SetPriority(USART1_IRQn, 0, 0);
    HAL_NVIC_EnableIRQ(USART1_IRQn);
    }

版权声明:本文为CSDN博主「USB_ABC」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/USB_ABC/article/details/121439426

近期,由于需要,先前使用GD32F103的替换STM32F103的芯片,它的系统主频提升到104MHZ, 众所周之,STM32F103的主频是72MHZ,这里没有什么问题,只要改一下SYSTEMCLOCK_CONFIG中的PLL的因子就可以了,原来是9倍,现在换成13倍频,这样,各项功能都正常,包括串口RS485, 但最近,客户要求原来的9600的波特率,改为1200, 这很简单,把波特率在串口的配置函数中修改一下就可以了,我使用了STM CUBEMX来生成的HAL代码库,本想以为就这么简单,但事实上,串口是不能正常执行命令的。换回9600,就能收到。我知道,这可能是串口时钟的原因,仿真一下,串口是能收到中断信号,但收到的数据不对,因此状态机,肯定就不能收到有效命令行了。

串口收到数据错的问题,是因为主频提高了,由72MHZ提到104MHZ,而波特率降低了6分之1,中间的分频可能是溢出了,没有细算。

//RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1;
RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV2;
把这个分频系数改为DIV2, 相当于APB进行了2分频,UART的进钟是取自于APB的,改完成, 1200的波特率命令执行正常。由于我的系统较简单,这个APB时钟分频改了后,可能要评估还会影响其它的硬件资源。
一点小经验,分享带来各位网友,不要在这样的问题上花时间。
以下是修改前后的代码,以供参考。

在这里插入图片描述

  RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV2;
  /* BEGIN: Added by lejianz, 2021/11/20 */
  //RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1;
  RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV2;
void MX_USART2_UART_Init(void)
{

  huart2.Instance = USART2;
  //huart2.Init.BaudRate = 115200;
  //huart2.Init.BaudRate = 9600;
  huart2.Init.BaudRate = 1200;
  huart2.Init.WordLength = UART_WORDLENGTH_8B;
  huart2.Init.StopBits = UART_STOPBITS_1;
  huart2.Init.Parity = UART_PARITY_NONE;
  huart2.Init.Mode = UART_MODE_TX_RX;
  huart2.Init.HwFlowCtl = UART_HWCONTROL_NONE;
  huart2.Init.OverSampling = UART_OVERSAMPLING_16;
  if (HAL_UART_Init(&huart2) != HAL_OK)
  {
    Error_Handler();
  }	
	
void HAL_UART_MspInit(UART_HandleTypeDef* uartHandle)
{

  GPIO_InitTypeDef GPIO_InitStruct;
  if(uartHandle->Instance==USART1)
  {
  /* USER CODE BEGIN USART1_MspInit 0 */

  /* USER CODE END USART1_MspInit 0 */
    /* Peripheral clock enable */
    __HAL_RCC_USART1_CLK_ENABLE();
  
    /**USART1 GPIO Configuration    
    PA9     ------> USART1_TX
    PA10     ------> USART1_RX 
    */
    GPIO_InitStruct.Pin = GPIO_PIN_9;
    GPIO_InitStruct.Mode = GPIO_MODE_AF_PP;
    //GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH;
    GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;
    HAL_GPIO_Init(GPIOA, &GPIO_InitStruct);

    GPIO_InitStruct.Pin = GPIO_PIN_10;
    GPIO_InitStruct.Mode = GPIO_MODE_INPUT;
    GPIO_InitStruct.Pull = GPIO_NOPULL;
    HAL_GPIO_Init(GPIOA, &GPIO_InitStruct);


	// PA8 -----> RS485 Control
    GPIO_InitStruct.Pin = RS485_CTRL_PIN;
    GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP;
    GPIO_InitStruct.Pull = GPIO_NOPULL;
    //GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH;
    GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;
    HAL_GPIO_Init(GPIOA, &GPIO_InitStruct);
	RS485_CTRL_RX();


    /* USART1 interrupt Init */
    HAL_NVIC_SetPriority(USART1_IRQn, 0, 0);
    HAL_NVIC_EnableIRQ(USART1_IRQn);
    }

版权声明:本文为CSDN博主「USB_ABC」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/USB_ABC/article/details/121439426

生成海报
点赞 0

USB_ABC

我还没有学会写个人说明!

暂无评论

发表评论

相关推荐

STM32 C++编程系列一:STM32 C++编程介绍

一、STM32及其他单片机开发现状 在目前绝大部分的单片机开发当中,C语言占据着主流的地位,但由于C语言本身是一种面向过程的语言,因此在当前利用面向对象思想构建可复用代码为主流的今天显得比较麻烦&#x

课程实习stm32主从蓝牙计算器+温度测量

说明:对于主从蓝牙计算器项目中的代码都是本人经过思考之后自行创作出来的,没有经过任何的网上抄录,由于课程实习的要求不高,所以我就没有对一些出现的bug进行修改(没有删除功能等