STM32串口溢出错误Overrun使用不当导致的串口死机

1. STM32串口默认是打开Overrun、DMA on RX Error

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

2. 使用HAL库,打开Overrun

如果出现错误,HAL库函数会关闭接收,再调用错误回调函数(用户需实现,不然串口接收不再工作,从外部看串口死机)

void HAL_UART_IRQHandler(UART_HandleTypeDef *huart)
{
      errorcode = huart->ErrorCode;
      if ((HAL_IS_BIT_SET(huart->Instance->CR3, USART_CR3_DMAR)) ||
          ((errorcode & (HAL_UART_ERROR_RTO | HAL_UART_ERROR_ORE)) != 0U))
      {
        /* Blocking error : transfer is aborted
           Set the UART state ready to be able to start again the process,
           Disable Rx Interrupts, and disable Rx DMA request, if ongoing */
        UART_EndRxTransfer(huart); 	//关闭接收

HAL_UART_ErrorCallback(huart);	//调用错误回调函数
}
}

HAL_StatusTypeDef HAL_UART_Init(UART_HandleTypeDef *huart)
{
  if (huart->AdvancedInit.AdvFeatureInit != UART_ADVFEATURE_NO_INIT)
  {
    UART_AdvFeatureConfig(huart);
  }
}

所以需在错误回调中重新开启接收

uint8_t buf[10];
void HAL_UART_ErrorCallback(UART_HandleTypeDef *huart)
{
	if(huart == &huart1)
	{
		__HAL_UNLOCK(huart);
		HAL_UART_Receive_IT(&huart1,buf,1);
}
}


/**
  * @brief Configure the UART peripheral advanced features.
  * @param huart UART handle.
  * @retval None
  */
void UART_AdvFeatureConfig(UART_HandleTypeDef *huart)
{
  /* Check whether the set of advanced features to configure is properly set */
  assert_param(IS_UART_ADVFEATURE_INIT(huart->AdvancedInit.AdvFeatureInit));



  /* if required, configure RX overrun detection disabling */
  if (HAL_IS_BIT_SET(huart->AdvancedInit.AdvFeatureInit, UART_ADVFEATURE_RXOVERRUNDISABLE_INIT))
  {
    assert_param(IS_UART_OVERRUN(huart->AdvancedInit.OverrunDisable));
    MODIFY_REG(huart->Instance->CR3, USART_CR3_OVRDIS, huart->AdvancedInit.OverrunDisable);
  }

  /* if required, configure DMA disabling on reception error */
  if (HAL_IS_BIT_SET(huart->AdvancedInit.AdvFeatureInit, UART_ADVFEATURE_DMADISABLEONERROR_INIT))
  {
    assert_param(IS_UART_ADVFEATURE_DMAONRXERROR(huart->AdvancedInit.DMADisableonRxError));
    MODIFY_REG(huart->Instance->CR3, USART_CR3_DDRE, huart->AdvancedInit.DMADisableonRxError);
  }

 
}
3. 在不使用HAL库来实现串口接收时需禁止Overrun,不然出现串口接收溢出,则串口自动关闭接收,从外部看来串口死机。
  • 15
    点赞
  • 59
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
好的,我可以帮你解决这个问题。首先,你需要在 STM32 上配置 ADC,并获取 ADC 的值。然后,你需要使用串口发送这个值。这个过程可以分为以下几个步骤: 1. 配置 ADC:你需要配置 ADC 的通道和采样率等参数。这可以使用 STM32 的 HAL 库来实现。 2. 获取 ADC 的值:一旦 ADC 配置完成,你可以通过使用 HAL 库提供的函数来获取 ADC 的值。 3. 将 ADC 的值转换为字符串:由于串口发送的是字符串,你需要将 ADC 的值转换为字符串。可以使用 sprintf 函数实现。 4. 发送字符串:最后,你可以使用 HAL 库提供的函数将字符串发送到串口。 下面是一个简单的示例代码,它演示了如何发送 ADC 值到串口: ``` #include "stm32f4xx_hal.h" #include <stdio.h> UART_HandleTypeDef huart; void HAL_ADC_ConvCpltCallback(ADC_HandleTypeDef* hadc) { uint32_t adc_value = HAL_ADC_GetValue(hadc); char str[10]; sprintf(str, "%d\r\n", adc_value); HAL_UART_Transmit(&huart, (uint8_t*)str, strlen(str), HAL_MAX_DELAY); } int main(void) { HAL_Init(); // 初始化串口 huart.Instance = USART2; huart.Init.BaudRate = 115200; huart.Init.WordLength = UART_WORDLENGTH_8B; huart.Init.StopBits = UART_STOPBITS_1; huart.Init.Parity = UART_PARITY_NONE; huart.Init.Mode = UART_MODE_TX_RX; huart.Init.HwFlowCtl = UART_HWCONTROL_NONE; huart.Init.OverSampling = UART_OVERSAMPLING_16; HAL_UART_Init(&huart); // 初始化 ADC ADC_HandleTypeDef hadc; hadc.Instance = ADC1; hadc.Init.ClockPrescaler = ADC_CLOCK_ASYNC_DIV1; hadc.Init.Resolution = ADC_RESOLUTION_12B; hadc.Init.DataAlign = ADC_DATAALIGN_RIGHT; hadc.Init.ScanConvMode = DISABLE; hadc.Init.EOCSelection = ADC_EOC_SINGLE_CONV; hadc.Init.LowPowerAutoWait = DISABLE; hadc.Init.LowPowerAutoPowerOff = DISABLE; hadc.Init.ContinuousConvMode = ENABLE; hadc.Init.DiscontinuousConvMode = DISABLE; hadc.Init.ExternalTrigConv = ADC_SOFTWARE_START; hadc.Init.ExternalTrigConvEdge = ADC_EXTERNALTRIGCONVEDGE_NONE; hadc.Init.DMAContinuousRequests = DISABLE; hadc.Init.Overrun = ADC_OVR_DATA_OVERWRITTEN; hadc.Init.OversamplingMode = DISABLE; HAL_ADC_Init(&hadc); // 配置 ADC 通道 ADC_ChannelConfTypeDef sConfig; sConfig.Channel = ADC_CHANNEL_0; sConfig.Rank = ADC_REGULAR_RANK_1; sConfig.SamplingTime = ADC_SAMPLETIME_3CYCLES; sConfig.SingleDiff = ADC_SINGLE_ENDED; sConfig.OffsetNumber = ADC_OFFSET_NONE; sConfig.Offset = 0; HAL_ADC_ConfigChannel(&hadc, &sConfig); // 启动 ADC 转换 HAL_ADC_Start_IT(&hadc); while (1) { // 主循环 } } ``` 这个示例代码通过使用 HAL 库来配置 ADC 和串口,并使用 sprintf 函数将 ADC 值转换为字符串。在 ADC 转换完成后,会触发 HAL_ADC_ConvCpltCallback 函数,该函数将 ADC 值发送到串口
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值