当前位置: 首页 > news >正文

告别串口不够用:手把手教你用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 常见编译问题解决

在实际移植过程中可能会遇到以下典型问题:

  1. 内核版本兼容性问题

    • 解决方案:修改驱动中的API调用方式,或使用条件编译
  2. GPIO中断配置错误

    // 在驱动中正确设置中断触发方式 irq_set_irq_type(gpio_to_irq(gpio_num), IRQF_TRIGGER_FALLING);
  3. 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

添加以下内容:

KERNEL=="ttysWK[0-3]", MODE="0666"

配置开机自动加载:

echo "wk2xxx_spi" | sudo tee -a /etc/modules-load.d/wk2124.conf

4. 多路串口应用开发

4.1 Python串口通信实例

使用pyserial库操作扩展串口:

import serial import threading def serial_reader(port_name): with serial.Serial(port_name, 115200, timeout=1) 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(target=serial_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 性能优化技巧

  1. 中断负载均衡

    • 将不同串口的中断绑定到不同CPU核心
    echo 1 > /proc/irq/<irq_num>/smp_affinity
  2. DMA缓冲区设置

    // 在驱动中增加DMA缓冲区大小 spi->master->dma_tx = dma_request_slave_channel(&spi->dev, "tx");
  3. 实时性调优

    sudo nice -n -20 python3 serial_app.py # 提高进程优先级

5. 项目实战:环境监测系统

5.1 系统架构设计

利用WK2124的4个串口连接不同传感器:

  • UART0:空气质量传感器(PMS5003)
  • UART1:气象站(WS-3000)
  • UART2:土壤湿度探头(RS485接口)
  • UART3:LoRa无线模块

系统架构如下图所示(此处应有架构图,实际项目中建议使用专业绘图工具):

5.2 数据采集实现

多线程数据采集示例:

from collections import deque import serial import threading import time class SensorReader: def __init__(self): self.data_buffers = { 'air_quality': deque(maxlen=100), 'weather': deque(maxlen=100), 'soil': deque(maxlen=100) } 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(target=self.read_pms5003), threading.Thread(target=self.read_weather_station) ] for t in threads: t.daemon = True t.start()

5.3 常见问题排查

  1. 数据丢失问题

    • 检查FIFO阈值设置
    • 增加看门狗定时器监控
  2. 波特率不匹配

    # 精确设置波特率 ser = serial.Serial('/dev/ttysWK0', baudrate=115200, bytesize=8, parity='N', stopbits=1)
  3. 电磁干扰处理

    • 在信号线上加磁珠滤波
    • 使用双绞线连接长距离设备

6. 进阶应用与性能测试

6.1 高负载压力测试

模拟多路高速数据传输:

import serial import threading import time def stress_test(port_name): ser = serial.Serial(port_name, 921600) test_data = b'X' * 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(target=stress_test, args=(f"/dev/ttysWK{i}",)) t.start()

6.2 低延迟优化方案

  1. 内核参数调整

    echo 1000000 > /proc/sys/kernel/sched_rt_period_us echo 950000 > /proc/sys/kernel/sched_rt_runtime_us
  2. 实时内核补丁

    sudo apt install linux-image-rt-rpi-v7
  3. 中断合并禁用

    echo 0 > /sys/module/wk2xxx_spi/parameters/irq_coalesce

6.3 扩展应用场景

  1. 工业自动化

    • 同时连接多个PLC设备
    • 实现Modbus RTU多主机通信
  2. 机器人控制

    • 驱动多个伺服电机控制器
    • 接收多路编码器反馈
  3. 物联网网关

    • 汇聚多种串口传感器数据
    • 通过WiFi/4G上传云端
http://www.jsqmd.com/news/690399/

相关文章:

  • 如何5分钟搭建个人番茄小说图书馆:终极离线阅读解决方案
  • 告别网络依赖:手把手教你下载并本地配置Mermaid.js(附完整HTML模板)
  • 别再只盯着编译器版本!解决ARMCC A1163E报错,关键在Keil这个隐藏设置
  • 2026年比较好的TI型号/TI汽车级芯片生产厂家推荐 - 品牌宣传支持者
  • 2026固安网站建设品牌选型指南:房山区,怀柔区,延庆区固安外贸网站制作,固安外贸网站建设,优选指南! - 优质品牌商家
  • 海思使用sdl+sdl_ttf+freetype生成位图叠加osd
  • RISC-V IDE混战,我为什么最终选择了Segger Embedded Studio?
  • 电脑小白自救指南:当你的Win10被2345、小鸟壁纸攻占后,除了重装还能做什么?
  • 逆向知乎x-zse-96参数时,我踩过的那些‘环境坑’:从Canvas到Window原型链的完整避坑指南
  • 股市学习心得-股市的一天
  • 从TOPS到实际吞吐量:解码AI芯片推理效率的四大关键指标
  • 超表面信道优化:原理、对抗机制与5G应用
  • 3个步骤解锁图表数据:WebPlotDigitizer让科研图表“开口说话“
  • 【模拟IC设计实战】从源极负反馈到Cascode OTA:增益、线性度与带宽的权衡艺术
  • 深入浅出AUTOSAR通信栈:用一张图讲清楚CAN、CANIF、PDUR、COM、CANTP之间的数据流转
  • Godot游戏资源提取:3分钟学会PCK文件解包技巧
  • 现代内容创作:模板工具降低视觉制作成本的策略与实践
  • 别再只会用库了!用C语言手搓I2C驱动OLED(SH1106/SSD1306)的底层逻辑与调试技巧
  • 编码基础:ASCII、Unicode、UTF-8 区别与原理
  • 联发科Genio 700处理器:中端AIoT市场的性能与能效平衡
  • 从华为3COM到H3C再到紫光:一个网络设备品牌的“前世今生”与认证体系变迁
  • 第19篇:注意力机制初探——让AI学会“聚焦”关键信息(概念入门)
  • 全面掌握QtScrcpy:高效实现Android设备屏幕镜像与控制的终极指南
  • 终极网盘直链下载助手:八大平台一键解析,告别限速烦恼
  • 新手也能看懂的CTF逆向入门:从UPX脱壳到pyc反编译实战(附flag获取全流程)
  • 为什么陶瓷PCB“仿真没问题”,实际却频繁失效?3个容易忽略的细节
  • 从驱动器内部架构看SSI编码器:为什么高端伺服都爱用FPGA来处理?
  • 元学习驱动的图像融合新范式:ReFusion如何通过可学习损失实现自适应融合
  • 从零到一:深入解析torch.optim.SGD的动量与正则化实战
  • 别再死记硬背了!用Python算算你的摄像头到底需要多大带宽(附分辨率/帧率/格式计算脚本)