MaixCAM MaixPy UART 串口使用介绍

更新历史
日期 版本 作者 更新内容
2025-08-08 1.2.0 Neucrack 添加 MaixCAM2 支持
2024-08-01 1.1.0 Neucrack 优化文档,更多详细内容
2024-03-07 1.0.0 Neucrack 初版文档

前置知识

请先学会使用pinmap 模块设置引脚功能。

要让一个引脚能使用 UART 功能,先用pinmap设置对应引脚功能为UART

串口简介

串口是一种通信方式,包含了硬件和通信协议的定义。

  • 硬件包括:
    • 3 个引脚: GNDRXTX,通信双发交叉连接 RX TX, 即一方 TX 发送到另一方的 RX, 双方 GND 连接到一起。
    • 控制器,一般在芯片内部,也叫 UART 外设,一般一个芯片有一个或者多个 UART 控制器,每个控制器有相对应的引脚。
  • 串口通信协议: 为了让双方能顺利通信,规定了一套协议,即以什么样的时序通信,具体可以自行学习,常见的参数有 波特率 校验位等,波特率是我们用得最多的参数。

通过板子的串口,可以和其它单片机或者 SOC 进行数据通信,比如可以在 MaixCAM 上实现人体检测功能,检测到坐标后通过串口发送给 STM32/Arduino 单片机。

选择合适的 I2C 使用

首先我们需要知道设备有哪些引脚和 I2C,如图:

设备型号 引脚简图 引脚复用说明
MaixCAM 板子丝印比如A19是引脚名,UART1_TX是功能名
MaixCAM-Pro maixcam_pro_io 第一个名如A19是引脚名,对应UART1_TX是功能名
MaixCAM2 maixcam2_io 第一个名如IO0_A21是引脚名,对应UART4_TX是功能名

需要注意的是,引脚默认可能用做其它用途,最好避开这些引脚,请看pinmap 文档中的说明。

MaixCAM/MaixCAM-Pro 串口使用注意点:

  • 默认从 USB 口引出了一个串口0,可以插上配套的 Type-C 转接小板,就能直接使用上面的串口引脚,也可以不用转接板,直接使用板子上的 A16(TX)A17(RX)引脚, 和 USB 口引出的是同样的引脚,是等效的。
  • 对于 MaixCAM 使用 USB 引出的串口时需要注意,Typc-C 正插和反插,转接小板上的 RXTX会交换(默认 Type-C 母口朝前和丝印符合),所以当你发现无法通信时,有可能就是 RX TX 反了,可以尝试将 Type-C 翻转一面插再看看通信是否正常。这个算是设计缺陷,不过一般也不会经常拔插所以适应一下也能接受。
  • MaixCAM 的串口0 在开机时会打印一部分开机日志, 启动完毕后会打印serial ready字样,如果和单片机通信需要注意丢弃这部分信息,如果出现系统启动出现问题也可以通过查看串口0的开机打印来诊断问题。
  • MaixCAM 的 TX(UART0) 引脚也是启动模式检测引脚之一,所以在串口0开机时不能是被拉低的状态,不然会导致无法开机,是芯片的特性,如果你在做 3.3v5v 的电平转换电路要十分注意不要默认拉低请保持浮空(可以考虑使用电平转换芯片)。以及如果你发现无法开机,也可以先检查一下 TX 是否被拉低了。
  • 综上,UART0 如果你遇到了问题不好解决,建议使用其它串口比如UART1
  • UART0 也是系统默认maix protocol 串口。

MaixCAM2 使用注意点

  • MaixCAM2引出的串口很多,有UART0 / UART1 / UART2 / UART3 / UART4共 4 组串口,使用时不要混淆。
  • UART0是系统终端和日志串口。

波特率限制

注意,不是所有波特率都能使用的,推荐没有特殊需求使用 115200,这是芯片都支持的,其它波特率误码率可能会很高或者驱动未支持导致数据传输丢失。

这里列举不同设备的常见测试可用波特率(欢迎PR):

  • MaixCAM / MaixCAM-Pro: 115200
  • MaixCAM2: 115200。理论最高能到4000000 bits/s,底层baud = uart_clk /(小数分频 * 16),默认uart_clk200000000,小数分频值整数部分uart_clk / (baud * 16),小数部分round((uart_clk % (baud * 16)) * 16 / (baud * 16)) / 16,比如115200计算出来分频器设置为108.5,实际验证精度(115200 - (uart_clk / (108.5 * 16))) / 115200 = 0.0064%,所以如果你切换波特率,也可以根据这个公式进行计算精度。

串口硬件接线

两个设备通信, 接三个引脚,GNDRXTX,通信双发交叉连接 RX TX, 即一方 TX 发送到另一方的 RX, 双方 GND 连接到一起即可。

MaixPy 中使用串口

将两个通信的板子双方连接好后(通信双发交叉连接 RX TX, 即一方 TX 发送到另一方的 RX, 双方 GND 连接到一起),就可以使用软件了。

通过 MaixPy 使用串口,核心代码:

from maix import uart

serial_dev = uart.UART("/dev/ttyS0", 115200)
serial_dev.write_str("Hello MaixPy")

/dev/ttyS0 是串口设备,可以通过print(uart.list_devices())看到所有串口设备。
一般/dev/ttyS*,*就是串口号。

对于默认对应引脚就是串口功能的串口可以这样直接使用,其它串口则需要先用pinmap映射引脚功能为对应的UART功能,然后创建UART对象即可:

from maix import uart, pinmap, time, sys, err

# ports = uart.list_devices() # 列出所有串口

# get pin and UART number according to device id
device_id = sys.device_id()
if device_id == "maixcam2":
    pin_function = {
        "IO0_A21": "UART4_TX",
        "IO0_A22": "UART4_RX"
        # "IO1_A0": "UART2_TX",
        # "IO1_A1": "UART2_RX"
    }
    device = "/dev/ttyS4"
    # device = "/dev/ttyS2"
else:
    pin_function = {
        "A16": "UART0_TX",
        "A17": "UART0_RX"
        # "A19": "UART1_TX"
        # "A18": "UART1_RX",

    }
    device = "/dev/ttyS0"
    # device = "/dev/ttyS1"

for pin, func in pin_function.items():
    err.check_raise(pinmap.set_pin_function(pin, func), f"Failed set pin{pin} function to {func}")

# Init UART
serial_dev = uart.UART(device, 115200)
serial_dev.write_str("Hello MaixPy")

串口连接电脑

  • 有开发者可能会问:为什么插上 USB 电脑没出现串口设备?

答: 因为设备的 USB 口是 USB 功能,不是 USB 转串口功能,而且默认会虚拟成 USB 网卡,如果要访问设备的终端,请使用 ssh 连接。

  • 我想电脑和板子串口通信,怎么办?

答: 现代电脑一般只有 USB 口,所以想电脑和板子的 UART 通信,中间需要一个 USB 转 UART 的转接板,比如这个,USB 连接电脑,串口连接开发板的串口就能通信了。

  • 我想在电脑看板子串口终端打印的日志,或者终端交互,怎么操作?

答:一般推荐通过网络 ssh 进行终端交互,如果遇到问题,可以用以下方法进入串口终端:

  • 对于 MaixCAM/MaixCAM-Pro: 将 USB 转 UART 的转接板和板子的串口0A16(TX)A17(RX))引脚交叉相连,提前将系统内/boot/uEnv.txt中的consoledev一行前面加#号注释掉或者直接删掉使能串口0作为终端,然后重启系统就能通过串口0看到开机日志和终端交互了。
  • 对于 MaixCAM2: 将 USB 转 UART 的转接板和板子的串口0(U0T/U0R)引脚交叉相连,就可以进行串口终端交互了,开机也会打印日志。

发送数据

主要有两个函数write_strwrite函数。

write_str函数来发送字符串,write用来发送字节流,即strbytes类型,两者可以互相转换,比如:

  • "A" 调用encode()方法变成b"A",反过来b"A"调用decode()方法变成"A"
  • str 没法显示一些不可见字符比如 ASCII 码中的值0,在字符串中也是\0一般作为结束符,在bytes类型中就可以用b"\x00"来储存。
  • 对于非 ASCII 编码的字符串更有用,比如UTF-8编码中中文是由三个字节\xe5\xa5\xbd来表示的,我们可以通过"好".encode("utf-8")得到b"\xe5\xa5\xbd",也可以通过b'\xe5\xa5\xbd'.decode("utf-8)得到"好"

所以如果我们需要发送字节数据,则用write()方法发送即可, 比如:

bytes_content = b'\x01\x02\x03'
serial.write(bytes_content)

所以对于 str 类型,也可以不用write_str,而是使用serial.write(str_content.encode()) 来发送。

如果你有其它类型的数据,想将它们变成一个字符串发送,可以使用Python 字符串格式化来创建一个字符串,比如:
想发送I have xxx apple,这里xxx 想用一个整型变量,则:

num = 10
content = "I have {} apple".format(num)
content2 = f"I have {num} apple"
content3 = "I have {:04d} apple".format(num)
content4 = f"I have {num:d} apple"
print(content)
print(content2)
print(content3)
print(content4)
print(type(content))
serial.write_str(content)

另外你也可以把数据编码成二进制流数据发送,比如前 4 个字节是十六进制的 AABBCCDD,中间发送一个 int 类型的数值,最后再加一个0xFF结尾,使用struct.pack来进行编码(看不懂可以看后文的介绍):

from struct import pack
num = 10
bytes_content = b'\xAA\xBB\xCC\xDD'
bytes_content += pack("<i", num)
bytes_content += b'\xFF'
print(bytes_content, type(bytes_content))
serial.write(bytes_content)

这里 pack("<i", num)num编码为int类型即4字节的有符号数,<符号意思是小端编码,低位在前,这里num = 10,十六进制 4 字节表示就是0x0000000A,小端就是把低字节0x0A放在前面,得到一个b'\x0A\x00\x00\x00'的字节类型数据。

这里只举例使用i编码int类型的数据,还有其它类型比如B表示unsigned char等等,更多的struct.pack格式化用法可以自行搜索python struct pack

这样最终发送的就是AA BB CC DD 0A 00 00 00 FF二进制数据了。

接收

使用read方法进行读取数据,直接:

while not app.need_exit():
    data = serial.read()
    if data:
        print(data)
    time.sleep_ms(1)

同样,read方法获得的数据也是bytes类型,这里read会读取对方一次性发送的一串数据,如果没有数据就是b''即空字节。
这里用了time.sleep_ms(1)进行睡眠了1ms,用来释放 CPU,不让这个线程占用所有 CPU 资源,而且1ms不影响我们程序的效率,特别是在多线程时有用。

另外read函数有两个参数:

  • len:代表想接收的最大长度,默认-1代表缓冲区有多少就返回多少,传>0的值则代表最多返回这个长度的数据。
  • timeout
    • 默认 0 代表从缓冲区读取数据立马返回数据,如果len-1则返回所有数据,如果指定了len则返回长度不超过len 的数据。
    • <0 代表一直等待直到接收到了数据才返回,如果len-1则等待到接收到数据才返回(一串连续接收到的数据,即阻塞式读取所有数据),如果指定了len则等待接收数量达到len才返回。
    • >0 代表无论有没有接收到数据,超过这个时间就会返回。

看起来有点复杂,常见的参数组合:

  • read(): 即read(-1, 0),从缓冲区读取收到的数据,通常是对方一次性发来的一串数据,等到对方没有发送(一个字符的发送时间内没有再发)就立刻返回。
  • read(len = -1, timeout = -1): 阻塞式读取一串数据,等到对方发送了数据并且一个字符的发送时间内没有再发才返回。
  • read(len = 10, timeout = 1000): 阻塞式读取 10 个字符,读取到 10 个字符或者 超过 1000ms 还没收到就返回已经收到的数据。

设置接收回调函数

在 MCU 开发中,串口收到数据通常会有中断事件发生, MaixPy 已经在底层处理好了中断,开发者无需再处理中断。
如果你想在接收到数据时调用一个回调函数,可以用set_received_callback设置回调函数:


from maix import uart, app, time

def on_received(serial : uart.UART, data : bytes):
    print("received:", data)
    # send back
    serial.write(data)

device = "/dev/ttyS0"

serial = uart.UART(device, 115200)
serial.set_received_callback(on_received)

serial0.write_str("hello\r\n")
print("sent hello")
print("wait data")

while not app.need_exit():
    time.sleep_ms(100) # sleep to make CPU free

在接收到数据后会在另外一个线程里调用设置的回调函数,因为是在另外的线程里调用的,所以不像中断函数要尽快退出函数,你可以在回调函数里处理一些事务再退出也是可以的,注意多线程常见问题。

使用回调函数的方式接收数据请不要再使用read函数读取,否则会读取出错。

更多串口例程

MaixPy examples

串口API 文档

更多 API 看 UART API 文档

应用层通信协议

概念和字符协议

串口只是规定了保证硬件通信的时序,为了让接收方知道发送方发送的字符流的含义,我们一般会规定一个应用通信协议。
比如发送放需要发送一个坐标,包含了x, y两个整型值,为了让接收方能理解我们发送的字节流的含义,我们规定:

  • 帧头:当我开始发送$符号时,就代表我要开始发送有效的数据啦。

内容:设计一个开头符号的原因是串口是流式传输,比如发送两次12345有可能在某个时刻接收到了12345123这样的数据,第二帧的45还没有接收到,我们可以根据起始和结尾符号来判断一个完整的数据帧。

  • x, y 的取值范围是 0~65535, 即两个字节的无符号短整型(unsinged short),我会先发 x 再发 y,用逗号隔开,比如10,20
  • 帧尾:最后我会再发一个*标记来代表我这次数据发送完成了。

这样发送一次数据就类似$10,20*这样一个字符串,对方如果用 C 语言接收和解析:

// 1. 接收数据
// 2. 根据帧头帧尾判断是否接收完毕了,并将完整的一帧数据存到 buff 数组里面
// 3. 解析一帧数据
uint16_t x, y;
sscanf(buff, "$%d,%d*", &x, &y);

这样我们就制定了最简单的字符通信协议,具有一定的可靠性。
但是由于我们串口一般用的参数是115200 8 N 1,这里的N就是无奇偶校验,我们可以在自己的协议里面加一个校验值放在末尾,比如:

  • 这里我们规定 x,y 后还有一个校验值,取值范围是 0 到 255,它的值为前面所有字符加起来的和对 255 取余。
  • 这里以 $10,20举例,在Python只需要使用sum函数就可以sum(b'$10,20') % 255 --> 20,最终发送$10,20,20*
  • 接收放接收到数据后读取到校验值20,然后自己也同样的方式计算一遍$10,20的校验值,如果也是20说明数据传输没有发生错误,如果不相同我们则可以认为数据传输过程中发生了错误,可以丢弃等下一个数据包。

比如在 MaixPy 中,我们需要编码一个字符协议,直接使用 Python 的字符串格式化功能即可:

x = 10
y = 20
content = "${},{}*".format(x, y)
print(content)

二进制通信协议

上面的字符协议有个很明显的特征,我们都是用可见字符的方式在传输数据,传输数据时有点就是简单,人眼能直接看懂;
缺点就是占用字符数量不固定,数据量比较大,比如$10,20*$1000,2000*,同样的格式,数值不同长度不同,这里1000用了 4 个字符也就是4个字节,我们都知道一个无符号短整型(uint16)类型的数据只需要两个字节就能表示0~65535的取值范围,用这种表示方法可以少传输数据。
我们也知道可见字符可以通过ASCII码表转换成二进制表示形式,比如$1000查找ASCII码表换成二进制表示就是0x24 0x31 0x30 0x30 0x30一共 5 个字节,也就是我们实际传输数据的时候传输的二进制内容,如果现在我们用二进制的方式直接编码1000,即0x03E8,就可以直接发送0x24 0x03 0xE8,最终只需要发送 3 个字节,减少了通信开销。

另外这里0x03E8两个字节低位是0xE8,先发送低位0xE8我们称之为小端编码,反之则是大端编码,两个皆可,双方规定一致即可。

在 MaixPy 中,要将一个数值转换成 bytes 类型也很简单,使用struct.pack函数即可,比如这里的0x03E8也就是十进制的1000,我们用

from struct import pack
b = pack("<H", 1000)
print(b)

这里<H表示小端编码,H表示一个 uint16类型的数据,最终得到b'\xe8\x03'的 bytes 类型数据。

同样的,二进制协议也可以有 帧头,数据内容,校验值,帧尾等,也可以不要帧尾,而是设计一个帧长的字段,看个人喜好即可。

MaixPy MaixPy 内置通信协议

另外 MaixPy 也内置了一个通信协议可以直接使用,使用这个协议可以实现串口甚至 TCP 来切换应用、控制应用、获取应用发出的数据等。

比如 AI 检测应用检测到物体后发出的坐标就可以通过这个协议解析到。

其它教程