欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

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

程序员文章站 2024-02-22 11:59:16
...

串口中断介绍

串口简介

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

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

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

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

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

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

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

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

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

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

串口通信过程

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

空闲中断介绍

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;
}
相关标签: STM32