串口空闲中断收发不定长数据

这里使用的是stm32f103芯片, 使用串口向单片机发送数据,然后串口原样发送给PC。

1.变量:

uart.h中声明以下变量,方便在主函数中调用。其中RxMax为选定合适的最大接收值,用于计算接收个数,同时防止数组越界。

extern UART_HandleTypeDef UART1;
extern uint8_t RX_STATE;
extern uint8_t TX_DATA[256],RX_DATA[256];
extern uint8_t RxCounter;

#define RxMax 256

uart.c中变量定义如下:串口的初始化以及很多函数都需要用到串口句柄,不妨使用一个全局变量,同时定义一个合适大小的收发缓冲区。

UART_HandleTypeDef UART1;
uint8_t TX_DATA[256],RX_DATA[256];
uint8_t RX_STATE; //TX_DATA写入标志
uint8_t RxCounter; //用于计算收到数据的长度

2.串口初始化:

void  UART1_Init(uint32_t bandrate){ //串口初始化,返回值:无,参数:波特率
	UART1.Instance = USART1;
	UART1.Init.BaudRate =bandrate;
	UART1.Init.HwFlowCtl =UART_HWCONTROL_NONE  ;
	UART1.Init.Mode = UART_MODE_TX_RX;
	UART1.Init.Parity =UART_PARITY_NONE ;
	UART1.Init.StopBits =USART_STOPBITS_1;
	UART1.Init.WordLength =UART_WORDLENGTH_8B;
	HAL_UART_Init(&UART1);
	
	__HAL_UART_ENABLE_IT(&UART1,UART_IT_IDLE); //使能空闲中断
	HAL_UART_Receive_IT(&UART1,RX_DATA,RxMax); //使能接收
	
}

void HAL_UART_MspInit(UART_HandleTypeDef *huart){ //串口相关硬件的初始化

	if(huart->Instance == USART1){
	
	__HAL_RCC_GPIOA_CLK_ENABLE();//打开gpioa的时钟 
	__HAL_RCC_USART1_CLK_ENABLE();//打开串口1的时钟
			
	GPIO_InitTypeDef GPIO_Init;
		
	GPIO_Init.Mode = GPIO_MODE_AF_PP ;
	GPIO_Init.Pin = GPIO_PIN_9;//串口1的TX
	GPIO_Init.Speed =GPIO_SPEED_FREQ_MEDIUM;
	HAL_GPIO_Init(GPIOA,&GPIO_Init);
		
	GPIO_Init.Mode = GPIO_MODE_AF_INPUT ;
	GPIO_Init.Pin = GPIO_PIN_10;//串口1的RX
	GPIO_Init.Pull = GPIO_NOPULL; //上下拉只有输入模式有效,输出模式上下拉电阻被禁止
	HAL_GPIO_Init(GPIOA,&GPIO_Init);
		
	HAL_NVIC_SetPriority(USART1_IRQn ,3,0);//配置  中断优先级
	HAL_NVIC_EnableIRQ(USART1_IRQn);//使能中断
	}
}

3.中断处理:

如果使用简单的串口中断接收方式,预定接收10个字节,但是如果只收到5个字节,那么串口的接收计数器RxXfercounter不能见到0,也就无法调用中断接收完成的回调函数。但是,在收到五个字节后并且没有继续收到数据,串口会出现空闲帧,我们在串口的总中断处理函数中,判断是否发生了空闲中断,然后清除中断标志位,并进行想要的操作。

void USART1_IRQHandler(void){
	HAL_UART_IRQHandler(&UART1);  //这个函数处理UART中断请求。
	if(__HAL_UART_GET_FLAG(&UART1,UART_FLAG_IDLE)){ //如果产生了空闲中断
		__HAL_UART_CLEAR_IDLEFLAG(&UART1);
		RxCounter = 0;
		RxCounter += (RxMax - UART1.RxXferCount);
		HAL_UART_AbortReceive_IT(&UART1);	
		
	} 
}

以上程序在发生空闲中断的条件下,计算收到的字节,并调用中断停止接收(HAL_UART_AbortReceive_IT),该函数会调用中断停止接收回调函数(HAL_UART_AbortReceiveCpltCallback)。 它是一个弱声明函数,我们需要对它进行强声明。

void HAL_UART_AbortReceiveCpltCallback(UART_HandleTypeDef *huart){
	if(huart->Instance == USART1){	//停止接收中断回调函数
		memcpy(TX_DATA,RX_DATA,RxCounter); 
		RX_STATE=1;		
	}
	HAL_UART_Receive_IT(&UART1,RX_DATA,RxMax);
}

在这个函数中,我们把收到的数拷贝到TX_DATA中,并进行置位。 主函数如下,实现把数据发回电脑。 

int main(){
	
	HAL_Init();
	MyRcc_Init(); //时钟初始化,读者需要更改为自己写的时钟初始化函数
	UART1_Init(9600);
	LED_Init();
	
	while(1){
		 if(RX_STATE==1){
		 RX_STATE=0;
		 HAL_UART_Transmit_IT(&UART1,TX_DATA,RxCounter);		 
		 }
}
}

4.完整代码

uart.c

#include "uart.h"


UART_HandleTypeDef UART1;
uint8_t TX_DATA[256],RX_DATA[256];
uint8_t RX_STATE; //TX_DATA写入标志
uint8_t RxCounter; //用于计算收到数据的长度
 
void  UART1_Init(uint32_t bandrate){ //串口初始化
	UART1.Instance = USART1;
	UART1.Init.BaudRate =bandrate;
	UART1.Init.HwFlowCtl =UART_HWCONTROL_NONE  ;
	UART1.Init.Mode = UART_MODE_TX_RX;
	UART1.Init.Parity =UART_PARITY_NONE ;
	UART1.Init.StopBits =USART_STOPBITS_1;
	UART1.Init.WordLength =UART_WORDLENGTH_8B;
	HAL_UART_Init(&UART1);
	
	__HAL_UART_ENABLE_IT(&UART1,UART_IT_IDLE); //使能空闲中断
	HAL_UART_Receive_IT(&UART1,RX_DATA,RxMax); //使能接收
	
}

void HAL_UART_MspInit(UART_HandleTypeDef *huart){

	if(huart->Instance == USART1){
	
	__HAL_RCC_GPIOA_CLK_ENABLE();//打开gpioa的时钟 
	__HAL_RCC_USART1_CLK_ENABLE();//打开串口1的时钟
			
	GPIO_InitTypeDef GPIO_Init;
		
	GPIO_Init.Mode = GPIO_MODE_AF_PP ;
	GPIO_Init.Pin = GPIO_PIN_9;//串口1的TX
	GPIO_Init.Speed =GPIO_SPEED_FREQ_MEDIUM;
	HAL_GPIO_Init(GPIOA,&GPIO_Init);
		
	GPIO_Init.Mode = GPIO_MODE_AF_INPUT ;
	GPIO_Init.Pin = GPIO_PIN_10;//串口1的RX
	GPIO_Init.Pull = GPIO_NOPULL; //上下拉只有输入模式有效,输出模式上下拉电阻被禁止
	HAL_GPIO_Init(GPIOA,&GPIO_Init);
		
	HAL_NVIC_SetPriority(USART1_IRQn ,3,0);//配置  中断优先级
	HAL_NVIC_EnableIRQ(USART1_IRQn);//使能中断
	}
}
 

void USART1_IRQHandler(void){
	HAL_UART_IRQHandler(&UART1);  //这个函数处理UART中断请求。
	if(__HAL_UART_GET_FLAG(&UART1,UART_FLAG_IDLE)){ //如果产生了空闲中断
		__HAL_UART_CLEAR_IDLEFLAG(&UART1);
		RxCounter = 0;
		RxCounter += (RxMax - UART1.RxXferCount);
		HAL_UART_AbortReceive_IT(&UART1);	
		
	} 
}


void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart){

	if(huart->Instance == USART1){  
	
 
	}
}

void HAL_UART_TxCpltCallback(UART_HandleTypeDef *huart){
	if(huart->Instance == USART1){ //发送完成中断回调函数
	
	
	}

}

void HAL_UART_ErrorCallback(UART_HandleTypeDef *huart){
	if(huart->Instance == USART1){	//错误中断回调函数
	
	
	}
	
}
void HAL_UART_AbortReceiveCpltCallback(UART_HandleTypeDef *huart){
	if(huart->Instance == USART1){	//停止接收中断回调函数
		memcpy(TX_DATA,RX_DATA,RxCounter); 
		RX_STATE=1;		
	//	HAL_UART_Receive_IT(&UART1,RX_DATA,RxCounter);
	}
	HAL_UART_Receive_IT(&UART1,RX_DATA,RxMax);
}

uart.h

#ifndef   __UART_H
#define   __UART_H
#include "stdint.h"
#include "string.h"
#include "stm32f1xx_hal.h"

extern UART_HandleTypeDef UART1;
extern uint8_t RX_STATE;
extern uint8_t TX_DATA[256],RX_DATA[256];
extern uint8_t RxCounter;


#define RxMax 256

void  UART1_Init(uint32_t bandrate);
 
#endif

main.c

#include "stm32f1xx_hal.h"
#include "myrcc.h" //读者需要更换为自己的时钟文件,或者拷备以下代码
#include "uart.h"

 

int main(){
	
	HAL_Init();
	MyRcc_Init();
	UART1_Init(9600);
	LED_Init();
	
	while(1){
		 if(RX_STATE==1){
		 RX_STATE=0;
		 HAL_UART_Transmit_IT(&UART1,TX_DATA,RxCounter);		 
		 }
}
}

myrcc.c,

#include "stm32f1xx_hal.h"
#include "myrcc.h"

void MyRcc_Init(void){ //实现使用外部时钟源通过pll倍频
/*时钟初始化
	1.初始化RCC振荡器
	2.初始化时钟总线*/
	RCC_OscInitTypeDef RCC_OscInitType;
	RCC_OscInitType.OscillatorType = RCC_OSCILLATORTYPE_HSE  ;//选择振荡电路;外部高速时钟
	RCC_OscInitType.HSEState = RCC_HSE_ON; //打开hse(外部高速时钟
	RCC_OscInitType.HSEPredivValue = RCC_HSE_PREDIV_DIV1; //预分频
	RCC_OscInitType.PLL.PLLState = RCC_PLL_ON; //pll倍频开启
	RCC_OscInitType.PLL.PLLSource = RCC_PLLSOURCE_HSE; //pll时钟源:HSE
	RCC_OscInitType.PLL.PLLMUL = RCC_PLL_MUL9;
	
	HAL_RCC_OscConfig(&RCC_OscInitType); //参数中指定的参数初始化RCC振荡器
	
	RCC_ClkInitTypeDef RCC_ClkInitStruct;
	RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1; //AHB分频系数
	RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV2; //APB1分频系数
	RCC_ClkInitStruct.APB2CLKDivider =RCC_HCLK_DIV1; //APB2分频系数
	RCC_ClkInitStruct.ClockType =RCC_CLOCKTYPE_SYSCLK | RCC_CLOCKTYPE_HCLK | RCC_CLOCKTYPE_PCLK1 | RCC_CLOCKTYPE_PCLK2; //需要配置的时钟
	RCC_ClkInitStruct.SYSCLKSource =RCC_SYSCLKSOURCE_PLLCLK; //系统时钟输入源
	 
	HAL_RCC_ClockConfig(&RCC_ClkInitStruct,FLASH_LATENCY_2);//根据指定初始化CPU、AHB和APB总线时钟,RCC_ClkInitStruct中的*参数。

 
}

myrcc.h

#ifndef   __MYRCC_H
#define   __MYRCC_H

 void MyRcc_Init(void);

#endif

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值