从CuteCom到代码手把手教你用I.MX6ULL实现串口双向通信附完整工程源码在嵌入式开发中串口通信是最基础也最常用的调试手段之一。无论是简单的数据收发还是复杂的协议交互串口都能提供稳定可靠的通信方式。本文将带你从零开始在Ubuntu系统上使用CuteCom工具与I.MX6ULL开发板建立完整的串口通信链路实现双向数据传输。1. 搭建Ubuntu串口调试环境1.1 CuteCom安装与配置CuteCom是Linux下轻量级的串口调试工具相比minicom等工具它的图形界面更加友好直观。在Ubuntu系统中安装非常简单sudo apt update sudo apt install cutecom安装完成后建议使用root权限运行避免权限问题sudo cutecom首次使用时需要正确配置串口参数设备选择通常USB转串口设备会显示为/dev/ttyUSB0如果未自动列出可以手动输入波特率设置与嵌入式设备保持一致常见值为115200数据格式通常为8位数据位、无校验位、1位停止位8N1提示如果遇到权限问题可以临时修改设备权限sudo chmod 666 /dev/ttyUSB01.2 常见问题排查在实际使用中可能会遇到以下问题设备未识别检查dmesg | grep tty输出确认设备节点乱码问题确保双方波特率、数据格式完全一致收发异常检查硬件连接特别是TX/RX是否交叉连接2. I.MX6ULL串口硬件基础2.1 UART硬件架构I.MX6ULL的UART控制器具有以下特点特性说明工作模式全双工异步通信波特率可编程最高支持5MbpsFIFO32字节收发FIFO中断支持多种中断源可配置DMA支持可配合DMA减轻CPU负担2.2 引脚复用配置I.MX6ULL的UART1默认引脚TXDUART1_TX_DATA (GPIO1_IO16)RXDUART1_RX_DATA (GPIO1_IO17)需要正确配置IOMUXC寄存器将引脚功能设置为UART模式// 设置引脚复用为UART功能 IOMUXC_SetPinMux(IOMUXC_UART1_TX_DATA_UART1_TX, 0); IOMUXC_SetPinMux(IOMUXC_UART1_RX_DATA_UART1_RX, 0); // 配置电气特性 IOMUXC_SetPinConfig(IOMUXC_UART1_TX_DATA_UART1_TX, 0x10B0); IOMUXC_SetPinConfig(IOMUXC_UART1_RX_DATA_UART1_RX, 0x10B0);3. 串口驱动实现3.1 初始化流程完整的UART初始化包含以下步骤关闭UARTUCR1[0]0执行软复位UCR2[0]0配置数据格式字长、停止位、校验设置波特率使能UARTUCR1[0]1关键寄存器配置示例void uart_init(void) { // 1. 关闭UART UART1-UCR1 ~(10); // 2. 软复位 UART1-UCR2 ~(10); while((UART1-UCR2 0x1) 0); // 3. 配置数据格式 UART1-UCR2 | (114) | (15) | (12) | (11); // 8N1 // 4. 设置波特率(115200 80MHz) UART1-UFCR 57; // 分频设置 UART1-UBIR 71; UART1-UBMR 3124; // 5. 使能UART UART1-UCR1 | (10); }3.2 数据收发实现发送函数需要检查发送FIFO状态void uart_putc(char c) { while(!(UART1-USR2 (13))); // 等待发送就绪 UART1-UTXD c; } void uart_puts(const char *s) { while(*s) { uart_putc(*s); } }接收函数需要检查接收FIFO状态char uart_getc(void) { while(!(UART1-USR2 0x1)); // 等待数据到达 return UART1-URXD 0xFF; }4. 构建完整通信系统4.1 命令解析框架实现一个简单的命令解析器可以扩展功能typedef struct { const char *cmd; void (*handler)(void); } uart_cmd_t; static uart_cmd_t cmd_table[] { {led_on, led_on_handler}, {led_off, led_off_handler}, {read_temp, read_temp_handler}, {NULL, NULL} }; void uart_cmd_process(void) { static char buf[64]; static int idx 0; char c uart_getc(); if(c \r || c \n) { buf[idx] \0; parse_cmd(buf); idx 0; } else { buf[idx] c; } }4.2 工程组织建议合理的工程目录结构serial_demo/ ├── drivers/ │ ├── uart.c │ └── uart.h ├── app/ │ ├── main.c │ └── cmd.c ├── build/ └── Makefile示例Makefile关键内容CROSS_COMPILE arm-linux-gnueabihf- CC $(CROSS_COMPILE)gcc LD $(CROSS_COMPILE)ld OBJCOPY $(CROSS_COMPILE)objcopy OBJS drivers/uart.o app/main.o app/cmd.o %.o: %.c $(CC) -c $ -o $ serial_demo.elf: $(OBJS) $(LD) -T imx6ull.lds $^ -o $ $(OBJCOPY) -O binary $ serial_demo.bin4.3 调试技巧回显测试实现接收后立即发送的回显功能验证基本通信状态指示灯用LED指示通信状态如闪烁表示数据接收超时机制为接收函数添加超时判断避免死等数据校验简单实现如累加和校验提高可靠性5. 进阶应用协议设计5.1 简单文本协议示例命令格式SET LED1 ON\r\n GET TEMP\r\n解析实现void parse_protocol(const char *buf) { char cmd[16], obj[16], param[16]; if(sscanf(buf, %s %s %s, cmd, obj, param) 3) { if(strcmp(cmd, SET) 0) { handle_set(obj, param); } } else if(sscanf(buf, %s %s, cmd, obj) 2) { if(strcmp(cmd, GET) 0) { handle_get(obj); } } }5.2 二进制协议设计更高效的二进制协议框架#pragma pack(1) typedef struct { uint8_t head; // 0xAA uint8_t cmd; uint16_t len; uint8_t data[0]; uint16_t crc; } uart_frame_t; #pragma pack() void send_frame(uint8_t cmd, const void *data, uint16_t len) { uart_frame_t *frame malloc(sizeof(uart_frame_t) len); frame-head 0xAA; frame-cmd cmd; frame-len len; memcpy(frame-data, data, len); frame-crc calc_crc(frame, sizeof(uart_frame_t)len-2); uart_send((char*)frame, sizeof(uart_frame_t)len); free(frame); }6. 性能优化技巧6.1 FIFO使用优化利用32字节FIFO减少中断次数// 批量发送 void uart_send_bulk(const char *data, int len) { while(len--) { while(UART1-UTS (15)); // 等待TXFIFO有空位 UART1-UTXD *data; } }6.2 中断驱动实现配置接收中断提高响应速度// 中断初始化 void uart_irq_init(void) { UART1-UCR1 | (15); // 使能接收中断 UART1-UCR4 | (10); // 设置FIFO触发阈值 // 注册中断处理函数 request_irq(UART1_IRQn, uart_isr); } // 中断服务程序 void uart_isr(void) { if(UART1-USR1 (15)) { // 接收中断标志 while(UART1-USR2 0x1) { // 有数据 char c UART1-URXD; ringbuf_put(rx_buf, c); } } }6.3 DMA传输配置使用DMA进一步降低CPU占用void uart_dma_init(void) { // 配置DMA通道 DMA1-TCD[0].SADDR UART1-UTXD; DMA1-TCD[0].DADDR tx_buffer; DMA1-TCD[0].CITER BUF_SIZE; // 使能UART DMA请求 UART1-UCR3 | (12); }