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

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

暂无评论

发表评论

相关推荐

C#上位机开发(二) 串口使用

系列文章目录 代码下载 前言 关于C#上位机软件的制作,是我通过学习网络上的博主代码并自己进行了一些实战后总结验证以后的,一套自己的代码风格,引入了面相对象编程等思路 C#上位机开发(