ST意法半导体
直播中

1123127317

9年用户 1131经验值
擅长:制造/封装 接口/总线/驱动
私信 关注
[问答]

在外部中断中将printf写入,外部中断LED可以翻转,但串口不可显示,为什么?

在外部中断中将printf写入,外部中断LED可以翻转,但串口不可显示

```/* USER CODE END Header */
/* Includes ------------------------------------------------------------------*/
#include "main.h"
#include "dma.h"
#include "usart.h"
#include "gpio.h"
#include "stdio.h"
/* Private includes ----------------------------------------------------------*/

void SystemClock_Config(void);

int main(void)
{
HAL_Init();
SystemClock_Config();
MX_GPIO_Init();
MX_DMA_Init();
MX_USART1_UART_Init();

while (1)
{

printf("key downrn");

}
/* USER CODE END 3 */
}/**@brief System Clock Configuration@retval None
*/
void SystemClock_Config(void)
{
RCC_OscInitTypeDef RCC_OscInitStruct = {0};
RCC_ClkInitTypeDef RCC_ClkInitStruct = {0};/** Configure the main internal regulator output voltage
*/
HAL_PWREx_ControlVoltageScaling(PWR_REGULATOR_VOLTAGE_SCALE1_BOOST);/** Initializes the RCC Oscillators according to the specified parametersin the RCC_OscInitTypeDef structure.
*/
RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSE;
RCC_OscInitStruct.HSEState = RCC_HSE_ON;
RCC_OscInitStruct.PLL.PLLState = RCC_PLL_ON;
RCC_OscInitStruct.PLL.PLLSource = RCC_PLLSOURCE_HSE;
RCC_OscInitStruct.PLL.PLLM = RCC_PLLM_DIV6;
RCC_OscInitStruct.PLL.PLLN = 85;
RCC_OscInitStruct.PLL.PLLP = RCC_PLLP_DIV2;
RCC_OscInitStruct.PLL.PLLQ = RCC_PLLQ_DIV2;
RCC_OscInitStruct.PLL.PLLR = RCC_PLLR_DIV2;
if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK)
{
Error_Handler();
}/** Initializes the CPU, AHB and APB buses clocks
*/
RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_HCLK|RCC_CLOCKTYPE_SYSCLK
|RCC_CLOCKTYPE_PCLK1|RCC_CLOCKTYPE_PCLK2;
RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_PLLCLK;
RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1;
RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV1;
RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1;if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_4) != HAL_OK)
{
Error_Handler();
}
}/* USER CODE BEGIN 4 *//* USER CODE END 4 *//**@brief  This function is executed in case of error occurrence.@retval None
*/
void Error_Handler(void)
{
/* USER CODE BEGIN Error_Handler_Debug */
/* User can add his own implementation to report the HAL error return state */
__disable_irq();
while (1)
{
}
/* USER CODE END Error_Handler_Debug */
}#ifdef  USE_FULL_ASSERT
/**@brief  Reports the name of the source file and the source line number    where the assert_param error has occurred.
    where the assert_param error has occurred.
@param  file: pointer to the source file name@param  line: assert_param error line source number@retval None
*/
void assert_failed(uint8_t *file, uint32_t line)
{
/* USER CODE BEGIN 6 */
/* User can add his own implementation to report the file name and line number,
ex: printf("Wrong parameters value: file %s on line %drn", file, line) */
/* USER CODE END 6 */
}
#endif /* USE_FULL_ASSERT */
int fputc(int ch,FILE *f)
{
while((USART1->ISR&0X40)==0);//中断状态寄存器
USART3->TDR=(uint8_t)ch;//传输数据寄存器return ch;
}
void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin)
{
UNUSED(GPIO_Pin);
if(bule_key_Pin==GPIO_Pin)
{
HAL_GPIO_TogglePin(led2_GPIO_Port,led2_Pin);

printf("key downrn");
}
}```

```

回帖(4)

丁兵溢

2025-3-13 14:38:43
fputc函数里用了uart1和uart3,怎么这样子?
举报

胡毅秉

2025-3-13 14:38:48
检查一下对应初始化的是哪个串口


判断和发送的串口号不一致

可以参考库中的例程
举报

刘云

2025-3-13 14:38:54
先保证UART的字符能在串口终端,这就要求你所用片内UART跟串口助手TX/RX连接OK,
再就是波特率、帧格式两边一致;


如果要通过重定向做打印输出,写法及用法可以参考相关例程,比方:


STM32Cube_FW_F4_V1.28.1ProjectsSTM32F401RE-NucleoExamplesUARTUART_Printf
STM32Cube_FW_G4_V1.6.0ProjectsNUCLEO-G474REExamplesUARTUART_Printf
STM32Cube_FW_F1_V1.8.6ProjectsSTM3210E_EVALExamplesUARTUART_Printf
......


其实也可以使用sprintf做字符转换后直接调用UART发送。
举报

李玲

2025-3-17 17:37:26

在外部中断中使用 printf 导致串口无输出的问题,通常有以下原因及解决方案:




1. 串口未正确初始化或重定向



  • 问题printf 默认输出到标准输出,需重定向到串口。

  • 检查

    • 确保 USART 已初始化(如 MX_USARTx_UART_Init())。

    • 重定向 printf 到串口:
      // 添加以下代码重定向printf
      int __io_putchar(int ch) {
         HAL_UART_Transmit(&huartx, (uint8_t *)&ch, 1, HAL_MAX_DELAY);
         return ch;
      }

    • 检查是否启用 Use MicroLIB(Keil 中需勾选此选项)。





2. 中断优先级冲突



  • 问题:外部中断优先级高于串口中断,导致 HAL_UART_Transmit 被抢占。

  • 解决

    • 调整中断优先级,确保串口中断优先级不高于外部中断。

    • 在CubeMX中设置:

      • 外部中断(EXTI)的优先级低于串口中断(如USARTx_IRQn)。






3. 在中断中调用阻塞式函数



  • 问题HAL_UART_Transmit 是阻塞函数,可能导致中断超时或死锁。

  • 解决

    • 在中断中使用非阻塞发送(如 HAL_UART_Transmit_IT):
      void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin) {
         if (GPIO_Pin == GPIO_PIN_x) {
             static char msg[] = "Interrupt!rn";
             HAL_UART_Transmit_IT(&huartx, (uint8_t *)msg, strlen(msg));
             HAL_GPIO_TogglePin(LED_GPIO_Port, LED_Pin);
         }
      }

    • 确保启用串口全局中断(USARTx_IRQn)并实现中断服务函数。





4. DMA配置问题



  • 问题:若使用DMA,可能未正确配置或启动。

  • 检查

    • 确认 MX_DMA_Init() 已调用且DMA通道正确。

    • 使用 HAL_UART_Transmit_DMA 并在中断中触发:
      HAL_UART_Transmit_DMA(&huartx, (uint8_t *)msg, strlen(msg));





5. 缓冲区溢出或数据未完成发送



  • 问题:中断中频繁调用 printf 导致缓冲区溢出。

  • 解决

    • 使用静态缓冲区或确保发送完成前不触发新发送:
      void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin) {
         static uint8_t is_busy = 0;
         if (GPIO_Pin == GPIO_PIN_x && !is_busy) {
             is_busy = 1;
             printf("Interrupt!rn");
             HAL_GPIO_TogglePin(LED_GPIO_Port, LED_Pin);
             is_busy = 0;
         }
      }





6. 硬件或接线问题



  • 检查

    • 确认串口线(TX/RX)连接正确。

    • 用示波器或逻辑分析仪检查串口是否有信号输出。





快速测试步骤




  1. main 函数中测试 printf


    printf("Hello Worldrn");

    如果无输出,检查串口重定向和初始化。




  2. 在外部中断中改用 HAL_UART_Transmit_IT


    HAL_UART_Transmit_IT(&huartx, (uint8_t *)"INTrn", 5);

    观察是否正常输出。






总结



  • 关键点:确保串口初始化正确、中断优先级合理,并避免在中断中使用阻塞函数。

  • 推荐方案:在中断中用 HAL_UART_Transmit_IT 替代 printf,并调整优先级。


通过以上步骤排查,通常可以解决串口无输出问题。

举报

更多回帖

发帖
×
20
完善资料,
赚取积分