STM32空闲中断-USART_IT_IDLE使用(不使用DMA方式)

串口中断介绍

串口简介

UART(Universal Asynchronous Receiver/Transmitter)通用异步收发传输器,UART 作为异步串口通信协议的一种,工作原理是将传输数据的每个字符一位接一位地传输。是在应用程序开发过程中使用频率最高的数据总线。

UART 串口的特点是将数据一位一位地顺序传送,只要 2 根传输线就可以实现双向通信,一根线发送数据的同时用另一根线接收数据。UART 串口通信有几个重要的参数,分别是波特率、起始位、数据位、停止位和奇偶检验位,对于两个使用 UART 串口通信的端口,这些参数必须匹配,否则通信将无法正常完成。UART 串口传输的数据格式如下图所示:

在这里插入图片描述

  • 起始位:表示数据传输的开始,电平逻辑为 “0” 。

  • 数据位:可能值有 5、6、7、8、9,表示传输这几个 bit 位数据。一般取值为 8,因为一个 ASCII 字符值为 8 位。

  • 奇偶校验位:用于接收方对接收到的数据进行校验,校验 “1”
    的位数为偶数(偶校验)或奇数(奇校验),以此来校验数据传送的正确性,使用时不需要此位也可以。

  • 停止位: 表示一帧数据的结束。电平逻辑为 “1”。

  • 波特率:串口通信时的速率,它用单位时间内传输的二进制代码的有效位(bit)数来表示,其单位为每秒比特数

  • bit/s(bps)。常见的波特率值有 4800、9600、14400、38400、115200等,数值越大数据传输的越快,波特率为

  • 115200 表示每秒钟传输 115200 位数据。

串口通信过程

在这里插入图片描述

空闲中断介绍

STM32的IDLE的中断在串口无数据接收的情况下,是不会一直产生的,产生的条件是这样的,当清除IDLE标志位后,必须有接收到第一个数据后,才开始触发,一断接收的数据断流,没有接收到数据,即产生IDLE中断。IDLE位不会再次被置高直到RXNE位被置起(即又检测到一次空闲总线)。RXNE接收中断可以不用开启,减少进中断的次数。

代码案列

串口配置

/* Config IDLE Interrupt To LPUART1  */
if(uart->UartHandle.Instance == LPUART1)
{ 
    __HAL_UART_ENABLE_IT(&uart->UartHandle, UART_IT_IDLE);
}

中断接收

void LPUART1_IRQHandler(void)
{
    struct stm32_uart *uart;
    int ch;	
	uart = &lpuart1;
        
     /* enter interrupt */
  //  rt_interrupt_enter();
	
    if((l_serial.parent.open_flag & RT_DEVICE_FLAG_INT_RX))
	{   
        HAL_UART_IRQHandler(&uart->UartHandle);	
        ch = -1;     
        ch = uart->UartHandle.Instance->RDR & 0xff;
        lpuart_recv_buff[recv_len++] = ch;  
        
        /* 空闲中断来时 */
        if(__HAL_UART_GET_FLAG(&uart->UartHandle,UART_FLAG_IDLE)!=RESET)
    	{
			recv_len--;
    	    __HAL_UART_CLEAR_IDLEFLAG(&uart->UartHandle);
    	 	uart_rx_idle_callback(lpuart_recv_buff,recv_len);
           // rt_memset(lpuart_recv_buff,0,sizeof(lpuart_recv_buff));
            recv_len = 0;
    	} 
			
    }
	
	if(recv_len >= 300)
	{
        recv_len = 0;
    }
    /* leave interrupt */
	//rt_interrupt_leave();
}

回调函数

void uart_rx_idle_callback(uint8_t *pData,int size)
{
	
    /* If recvive byte > 5, judge valid */
    if(size > 5)
    {
        debug_uart_recv_data_parse(pData,size);      
    }
   
}

数据处理

/*
 * lpuart Serial interrupt receive callback function
 * Format:HEAD(1B)+ CMD(1B)+ LEN(2B)+ DATA + CS(1B)
 */
bool debug_uart_recv_data_parse( uint8_t *pData, uint16_t Size)
{
    debug_uart_package_t debug_frame = {0};	
    rz_reset_info_t *pobj = rz_reset_info_get();

  //  printf_debug_log("recv_data_parse",pData,Size);

    /* Step1: Judge the length of received data*/
    debug_frame.recv_buf = pData;	   
    if(debug_frame.recv_buf == NULL) {
        rt_kprintf("recv_buff NULL error \r\n");
		return false;
    }   
	debug_frame.recv_len = Size;
	if(debug_frame.recv_len > MAX_EFR_LENGTH ) {
		rt_kprintf("Receive data length error \r\n");
		return false;
	}
    
#if 1   
	debug_frame.head = debug_frame.recv_buf[0];
	debug_frame.check_sum = rz_cs_check_sum_data(debug_frame.recv_buf,debug_frame.recv_len-1);

     /* Step2: Judge the content of received data */
	if(debug_frame.head != EFR_HEAD && debug_frame.check_sum != debug_frame.recv_buf[debug_frame.recv_len] ) {
		rt_kprintf("recv head error || check sum error\r\n");
		return false;
	}

    /* Step3:Check Mudule */
    if(pobj->init == NULL ){
        rt_kprintf(" Module Init failed\r\n");      
        return false;
	}
#endif  
    
    recvive_complete_flag = true;    
    g_lpuart_obj.lpuart_recv_len = debug_frame.recv_len;
    rt_memcpy (g_lpuart_obj.lpuart_recv_buf, debug_frame.recv_buf , debug_frame.recv_len);
   
	return true;
}
发布了117 篇原创文章 · 获赞 60 · 访问量 1万+
展开阅读全文

没有更多推荐了,返回首页

©️2019 CSDN 皮肤主题: 技术工厂 设计师: CSDN官方博客

分享到微信朋友圈

×

扫一扫,手机浏览