探究STM32G0串口(LL库)接收中断一直触发的原因

单片机型号:STM32G030C8T6
在初始化完串口后,开启串口接收中断,程序就一直在执行串口中断(此时串口无任何数据),通过在线查看串口外设的相关寄存器(如下图)发现RDR寄存器一直没有被清零,导致ISR寄存器的RXNE位一直有效,串口中断就一直处于触发的状态。
在这里插入图片描述
通过查看STM32G0x0 Datasheet章节 26.8.10 Bit5 RXNE位,数据收据里面说到可以通过读取RDR寄存器或者往RXFRQ寄存器写1来清除ISR寄存器的RXNE标志位。
在这里插入图片描述
LL库串口初始化及中断接收代码如下:

void Usart2_Init(u32 baudrate)
{
	LL_USART_DeInit(USART2);
	LL_USART_InitTypeDef USART_InitStruct = {0};
	LL_GPIO_InitTypeDef GPIO_InitStruct = {0};
	
	/* Peripheral clock enable */
	LL_APB1_GRP1_EnableClock(LL_APB1_GRP1_PERIPH_USART2);
	LL_IOP_GRP1_EnableClock(LL_IOP_GRP1_PERIPH_GPIOA);
	
	GPIO_InitStruct.Pin = LL_GPIO_PIN_2;
	GPIO_InitStruct.Mode = LL_GPIO_MODE_ALTERNATE;
	GPIO_InitStruct.Speed = LL_GPIO_SPEED_FREQ_HIGH;
	GPIO_InitStruct.OutputType = LL_GPIO_OUTPUT_PUSHPULL;
	GPIO_InitStruct.Pull = LL_GPIO_PULL_UP;
	GPIO_InitStruct.Alternate = LL_GPIO_AF_1;
	LL_GPIO_Init(GPIOA, &GPIO_InitStruct);
	
	GPIO_InitStruct.Pin = LL_GPIO_PIN_3;
	GPIO_InitStruct.Mode = LL_GPIO_MODE_ALTERNATE;
	GPIO_InitStruct.Speed = LL_GPIO_SPEED_FREQ_HIGH;
	GPIO_InitStruct.OutputType = LL_GPIO_OUTPUT_PUSHPULL;
	GPIO_InitStruct.Pull = LL_GPIO_PULL_UP;
	GPIO_InitStruct.Alternate = LL_GPIO_AF_1;
	LL_GPIO_Init(GPIOA, &GPIO_InitStruct);
	
	NVIC_SetPriority(USART2_IRQn, USART_PREPRIORITY);
	NVIC_EnableIRQ(USART2_IRQn);
	
	USART_InitStruct.PrescalerValue = LL_USART_PRESCALER_DIV1;
	USART_InitStruct.BaudRate = baudrate;  //波特率
	USART_InitStruct.DataWidth = LL_USART_DATAWIDTH_8B; //8位数据
	USART_InitStruct.StopBits = LL_USART_STOPBITS_2;  //2位停止位
	USART_InitStruct.Parity = LL_USART_PARITY_NONE;  //无奇偶校验
	USART_InitStruct.HardwareFlowControl = LL_USART_HWCONTROL_NONE;
	USART_InitStruct.OverSampling = LL_USART_OVERSAMPLING_16;
	USART_InitStruct.TransferDirection = LL_USART_DIRECTION_TX_RX;
	LL_USART_Init(USART2, &USART_InitStruct);
	
	LL_USART_SetTXFIFOThreshold(USART2, LL_USART_FIFOTHRESHOLD_1_8);//设置TX缓存阈值为深度的1/8
	LL_USART_SetRXFIFOThreshold(USART2, LL_USART_FIFOTHRESHOLD_1_8);//设置RX缓存阈值为深度的1/8
	
	LL_USART_EnableIT_RXNE_RXFNE(USART2); //使能串口接收中断
	LL_USART_Enable(USART2);
}

void USART2_IRQHandler(void)
{
    u8 data;
    if (LL_USART_IsActiveFlag_RXNE_RXFNE(USART2)) { //判断接收数据寄存器是否有数据
   		 //清除接收中断标志方式一:读取RDR寄存器以自动清除RXNE标志
		//清除接收中断标志方式二:往RXFRQ寄存器写1(刷新数据)以清除RXNE中断标志
        data = LL_USART_ReceiveData8(USART2); //读接收到的字节,同时相关标志自动清除
        LL_USART_RequestRxDataFlush(USART2);
    }
}
  • 1
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
嗨!对于 STM32G0 系列微控制器的串口中断发送和接收,你可以按照以下步骤进行配置和编程: 1. 配置串口: - 首先,打开串口时钟使能。例如,对于 USART1,可以使用 `RCC_APB2ENR_USART1EN` 来使能时钟。 - 然后,配置串口的引脚,选择正确的引脚作为串口的收发引脚。例如,对于 USART1,可以使用 `GPIOx_AFRL` 或 `GPIOx_AFRH` 来配置引脚的复用功能。 - 接下来,配置串口的参数,如波特率、数据位、停止位和校验位等。可以使用 `USART_CR1` 和 `USART_CR2` 寄存器来进行配置。 2. 配置中断: - 首先,打开串口接收中断使能。可以使用 `USART_CR1` 寄存器的 `RXNEIE` 位来使能接收中断。 - 然后,打开串口的发送中断使能。可以使用 `USART_CR1` 寄存器的 `TXEIE` 位来使能发送中断。 - 最后,配置串口中断优先级。可以使用 `NVIC_SetPriority()` 函数来设置中断优先级,并通过 `NVIC_EnableIRQ()` 函数使能中断。 3. 编写中断处理函数: - 对于接收中断,你可以在中断处理函数中读取接收到的数据,并进行相应的处理。可以使用 `USART_DR` 寄存器来读取接收到的数据。 - 对于发送中断,你可以在中断处理函数中发送需要发送的数据。可以使用 `USART_DR` 寄存器来写入需要发送的数据。 需要注意的是,在中断处理函数中,要及时清除中断标志位,以免重复触发中断。可以使用 `USART_SR` 寄存器的相应位来清除中断标志位。 这只是一个简单的概述,实际的代码实现可能会有所不同,具体还需根据你的具体应用和需求进行调整。希望对你有所帮助!如果你有其他问题,请随时提问。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值