告别串口不够用:手把手教你用WK2124芯片为树莓派/香橙派扩展4个UART
树莓派/香橙派串口扩展实战WK2124芯片全攻略当你在树莓派或香橙派上连接多个传感器、执行器或通信模块时原生串口数量不足的问题常常成为开发瓶颈。WK2124这颗SPI转4串口芯片能以不到20元的成本完美解决这个痛点。本文将带你从硬件连接到驱动移植再到多路串口数据收发实战彻底掌握WK2124在ARM开发板上的应用技巧。1. 硬件连接与电路设计1.1 WK2124核心特性解析WK2124是专为嵌入式系统设计的串口扩展芯片通过SPI接口可扩展出4个独立的全双工UART通道。每个通道都具备256级收发FIFO大幅降低CPU中断频率独立波特率设置各通道最高支持2Mbps速率灵活的中断配置可编程触发阈值和超时中断多种工作模式支持RS485自动方向控制与同类芯片相比WK2124在Linux内核驱动支持方面表现突出社区资源丰富特别适合树莓派等流行单板计算机。1.2 硬件连接示意图连接WK2124到树莓派需要6根主要信号线树莓派引脚WK2124引脚功能说明SPI0 CS0CS片选信号SPI0 SCLKSCLK时钟信号SPI0 MOSIMOSI主出从入SPI0 MISOMISO主入从出GPIO25IRQ中断输出3.3VVCC电源输入注意IRQ引脚需要10KΩ上拉电阻建议在PCB设计时预留复位电路位置典型连接电路如下图所示此处应有图示实际使用时建议参考数据手册设计# Python代码示例树莓派SPI初始化 import spidev spi spidev.SpiDev() spi.open(0, 0) # 使用SPI0 CS0 spi.max_speed_hz 10000000 # 设置SPI时钟频率2. 驱动移植与内核编译2.1 驱动获取与准备WK2124的Linux驱动通常需要从芯片厂商获取最新版本或从开源社区下载适配版本。驱动主要包含以下关键文件wk2xxx_spi.c核心驱动代码Makefile编译配置文件wk2xxx.h寄存器定义头文件建议创建专用目录存放驱动文件mkdir ~/wk2124_driver cd ~/wk2124_driver # 将驱动文件复制到此目录2.2 内核配置与编译在树莓派上编译驱动需要先准备内核头文件sudo apt update sudo apt install raspberrypi-kernel-headers修改Makefile关键参数KDIR : /lib/modules/$(shell uname -r)/build obj-m : wk2xxx_spi.o编译驱动模块make -C $(KDIR) M$(PWD) modules编译成功后生成wk2xxx_spi.ko文件。2.3 常见编译问题解决在实际移植过程中可能会遇到以下典型问题内核版本兼容性问题解决方案修改驱动中的API调用方式或使用条件编译GPIO中断配置错误// 在驱动中正确设置中断触发方式 irq_set_irq_type(gpio_to_irq(gpio_num), IRQF_TRIGGER_FALLING);SPI通信失败检查树莓派SPI接口是否启用raspi-config # 启用SPI接口3. 驱动加载与设备配置3.1 加载驱动模块将编译好的驱动复制到系统模块目录sudo cp wk2xxx_spi.ko /lib/modules/$(uname -r)/kernel/drivers/tty/serial/ sudo depmod -a手动加载驱动sudo modprobe wk2xxx_spi验证驱动是否加载成功ls /dev/ttysWK* # 应看到ttysWK0到ttysWK3设备节点 dmesg | grep wk2xxx # 查看内核日志3.2 自动加载配置创建udev规则文件sudo nano /etc/udev/rules.d/99-wk2124.rules添加以下内容KERNELttysWK[0-3], MODE0666配置开机自动加载echo wk2xxx_spi | sudo tee -a /etc/modules-load.d/wk2124.conf4. 多路串口应用开发4.1 Python串口通信实例使用pyserial库操作扩展串口import serial import threading def serial_reader(port_name): with serial.Serial(port_name, 115200, timeout1) as ser: while True: data ser.read(100) if data: print(f{port_name} received: {data.decode()}) # 创建四个串口读取线程 ports [f/dev/ttysWK{i} for i in range(4)] threads [] for port in ports: t threading.Thread(targetserial_reader, args(port,)) t.daemon True t.start() threads.append(t) # 主线程写入数据 ser serial.Serial(/dev/ttysWK0, 115200) while True: message input(Enter message to send: ) ser.write(message.encode())4.2 C语言高效通信实现对于性能要求高的场景可以使用原生C接口#include stdio.h #include fcntl.h #include termios.h int open_uart(const char *device) { int fd open(device, O_RDWR | O_NOCTTY); struct termios options; tcgetattr(fd, options); cfsetispeed(options, B115200); cfsetospeed(options, B115200); options.c_cflag | (CLOCAL | CREAD); options.c_cflag ~PARENB; options.c_cflag ~CSTOPB; options.c_cflag ~CSIZE; options.c_cflag | CS8; tcsetattr(fd, TCSANOW, options); return fd; } void uart_loopback_test() { int fd open_uart(/dev/ttysWK0); char buffer[256]; while(1) { int n read(fd, buffer, sizeof(buffer)); if(n 0) { write(fd, buffer, n); // 回传接收到的数据 } } close(fd); }4.3 性能优化技巧中断负载均衡将不同串口的中断绑定到不同CPU核心echo 1 /proc/irq/irq_num/smp_affinityDMA缓冲区设置// 在驱动中增加DMA缓冲区大小 spi-master-dma_tx dma_request_slave_channel(spi-dev, tx);实时性调优sudo nice -n -20 python3 serial_app.py # 提高进程优先级5. 项目实战环境监测系统5.1 系统架构设计利用WK2124的4个串口连接不同传感器UART0空气质量传感器PMS5003UART1气象站WS-3000UART2土壤湿度探头RS485接口UART3LoRa无线模块系统架构如下图所示此处应有架构图实际项目中建议使用专业绘图工具5.2 数据采集实现多线程数据采集示例from collections import deque import serial import threading import time class SensorReader: def __init__(self): self.data_buffers { air_quality: deque(maxlen100), weather: deque(maxlen100), soil: deque(maxlen100) } def read_pms5003(self): ser serial.Serial(/dev/ttysWK0, 9600) while True: data ser.read(32) # PMS5003数据包长度 self.data_buffers[air_quality].append(self.parse_pms5003(data)) def read_weather_station(self): ser serial.Serial(/dev/ttysWK1, 19200) while True: line ser.readline() self.data_buffers[weather].append(line.decode().strip()) def start_all(self): threads [ threading.Thread(targetself.read_pms5003), threading.Thread(targetself.read_weather_station) ] for t in threads: t.daemon True t.start()5.3 常见问题排查数据丢失问题检查FIFO阈值设置增加看门狗定时器监控波特率不匹配# 精确设置波特率 ser serial.Serial(/dev/ttysWK0, baudrate115200, bytesize8, parityN, stopbits1)电磁干扰处理在信号线上加磁珠滤波使用双绞线连接长距离设备6. 进阶应用与性能测试6.1 高负载压力测试模拟多路高速数据传输import serial import threading import time def stress_test(port_name): ser serial.Serial(port_name, 921600) test_data bX * 1024 # 1KB测试数据 start_time time.time() for _ in range(1000): # 发送1000次 ser.write(test_data) elapsed time.time() - start_time print(f{port_name} 吞吐量: {1000*1024/elapsed/1024:.2f} KB/s) # 启动四个端口测试 for i in range(4): t threading.Thread(targetstress_test, args(f/dev/ttysWK{i},)) t.start()6.2 低延迟优化方案内核参数调整echo 1000000 /proc/sys/kernel/sched_rt_period_us echo 950000 /proc/sys/kernel/sched_rt_runtime_us实时内核补丁sudo apt install linux-image-rt-rpi-v7中断合并禁用echo 0 /sys/module/wk2xxx_spi/parameters/irq_coalesce6.3 扩展应用场景工业自动化同时连接多个PLC设备实现Modbus RTU多主机通信机器人控制驱动多个伺服电机控制器接收多路编码器反馈物联网网关汇聚多种串口传感器数据通过WiFi/4G上传云端
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2548257.html
如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!