树莓派3:64位系统串口(UART)使用问题的解决方法

ops/2024/11/29 2:55:05/

前言

当我们要使用串口进行zigbee的短距离通信时,发现无法使用串口.

原因

树莓派3bCPU内部有两个串口,一个硬件串口(就是我们平时使用的UART),还有一个迷你串口(mini-uart),在老版本的树莓派中把硬件串口分配在GPIO上,可以单独使用.但是在新的树莓派中官方把硬件串口给了蓝牙模块上,而将一个没有时钟源,必须使用内核提供时钟参考源的mini串口,分发给GPIO的串口,这样以来由于内核的频率本身是在一直变换的,导致"mini串口"的速率不稳定,出现不能使用的情况.(我这边甚至没有显示出来(就是没有/dev/serial0 ->ttyS0)),工程师也知道这个bug了就没直接启动mini串口,只准我们用蓝牙.

解决只有serial1 -> ttyAMA0的情况

这是没打开GPIO串口的情况,只有serial1(蓝牙)使用的是ttyAMA0(硬件串口):

这里我试了很多方法,大体分为两种:
1,sudo  nano  /boot/config.txt 去里面打开写enable_uart=1,或者其他这就要看(cd /boot/overlays/#提供了README)这个文件的README了.
下面就是README的全部有关UART的解释

  krnbt_baudrate          Set the baudrate of the PL011 UART when usedwith krnbt=onuart0                   Set to "off" to disable uart0 (default "on")uart1                   Set to "on" or "off" to enable or disable uart1(default varies)
Name:   disable-bt
Info:   Disable onboard Bluetooth on Pi 3B, 3B+, 3A+, 4B and Zero W, restoringUART0/ttyAMA0 over GPIOs 14 & 15.N.B. To disable the systemd service that initialises the modem so itdoesn't use the UART, use 'sudo systemctl disable hciuart'.
Load:   dtoverlay=disable-bt
Params: <None>Name:   dpi18
Info:   Overlay for a generic 18-bit DPI displayThis uses GPIOs 0-21 (so no I2C, uart etc.), and activates the output2-3 seconds after the kernel has started.
Load:   dtoverlay=dpi18
Params: <None>Name:   dpi18cpadhi
Info:   Overlay for a generic 18-bit DPI display (in 'mode 6' connection scheme)This uses GPIOs 0-9,12-17,20-25 (so no I2C, uart etc.), and activatesthe output 3-3 seconds after the kernel has started.
Load:   dtoverlay=dpi18cpadhi
Params: <None>Name:   dpi24
Info:   Overlay for a generic 24-bit DPI displayThis uses GPIOs 0-27 (so no I2C, uart etc.), and activates the output2-3 seconds after the kernel has started.
Load:   dtoverlay=dpi24
Params: <None>Name:   midi-uart0
Info:   Configures UART0 (ttyAMA0) so that a requested 38.4kbaud actually gets31.25kbaud, the frequency required for MIDI
Load:   dtoverlay=midi-uart0
Params: <None>Name:   midi-uart1
Info:   Configures UART1 (ttyS0) so that a requested 38.4kbaud actually gets31.25kbaud, the frequency required for MIDI
Load:   dtoverlay=midi-uart1
Params: <None>Name:   midi-uart2
Info:   Configures UART2 (ttyAMA1) so that a requested 38.4kbaud actually gets31.25kbaud, the frequency required for MIDI
Load:   dtoverlay=midi-uart2
Params: <None>Name:   midi-uart3
Info:   Configures UART3 (ttyAMA2) so that a requested 38.4kbaud actually gets31.25kbaud, the frequency required for MIDI
Load:   dtoverlay=midi-uart3
Params: <None>Name:   midi-uart4
Info:   Configures UART4 (ttyAMA3) so that a requested 38.4kbaud actually gets31.25kbaud, the frequency required for MIDI
Load:   dtoverlay=midi-uart4
Params: <None>Name:   midi-uart5
Info:   Configures UART5 (ttyAMA4) so that a requested 38.4kbaud actually gets31.25kbaud, the frequency required for MIDI
Load:   dtoverlay=midi-uart5
Params: <None>Name:   miniuart-bt
Info:   Switch the onboard Bluetooth function on Pi 3B, 3B+, 3A+, 4B and Zero Wto use the mini-UART (ttyS0) and restore UART0/ttyAMA0 over GPIOs 14 &15. Note that this may reduce the maximum usable baudrate.N.B. It is also necessary to edit /lib/systemd/system/hciuart.serviceand replace ttyAMA0 with ttyS0, unless using Raspbian or anotherdistribution with udev rules that create /dev/serial0 and /dev/serial1,in which case use /dev/serial1 instead because it will always becorrect. Furthermore, you must also set core_freq and core_freq_min tothe same value in config.txt or the miniuart will not work.
Load:   dtoverlay=miniuart-bt,<param>=<val>
Params: krnbt                   Set to "on" to enable autoprobing of Bluetoothdriver without need of hciattach/btattachName:   pi3-miniuart-bt
Info:   This overlay has been renamed miniuart-bt, keeping pi3-miniuart-bt asan alias for backwards compatibility.
Load:   <Deprecated>Name:   qca7000-uart0
Info:   in-tech's Evaluation Board for PLC Stamp micro (UART)This uses uart0/ttyAMA0 over GPIOs 14 & 15 to connect the QCA7000.But it requires disabling of onboard Bluetooth onPi 3B, 3B+, 3A+, 4B and Zero W.
Load:   dtoverlay=qca7000-uart0,<param>=<val>
Params: baudrate                Set the baudrate for the UART (default"115200")Name:   sc16is750-i2c
Info:   Overlay for the NXP SC16IS750 UART with I2C InterfaceEnables the chip on I2C1 at 0x48 (or the "addr" parameter value). Toselect another address, please refer to table 10 in reference manual.
Load:   dtoverlay=sc16is750-i2c,<param>=<val>
Params: int_pin                 GPIO used for IRQ (default 24)addr                    Address (default 0x48)xtal                    On-board crystal frequency (default 14745600)Name:   sc16is752-i2c
Info:   Overlay for the NXP SC16IS752 dual UART with I2C InterfaceEnables the chip on I2C1 at 0x48 (or the "addr" parameter value). Toselect another address, please refer to table 10 in reference manual.
Load:   dtoverlay=sc16is752-i2c,<param>=<val>
Params: int_pin                 GPIO used for IRQ (default 24)addr                    Address (default 0x48)xtal                    On-board crystal frequency (default 14745600)Name:   sc16is752-spi0
Info:   Overlay for the NXP SC16IS752 Dual UART with SPI InterfaceEnables the chip on SPI0.
Load:   dtoverlay=sc16is752-spi0,<param>=<val>
Params: int_pin                 GPIO used for IRQ (default 24)xtal                    On-board crystal frequency (default 14745600)Name:   sc16is752-spi1
Info:   Overlay for the NXP SC16IS752 Dual UART with SPI InterfaceEnables the chip on SPI1.N.B.: spi1 is only accessible on devices with a 40pin header, eg:A+, B+, Zero and PI2 B; as well as the Compute Module.Name:   uart0
Info:   Change the pin usage of uart0
Load:   dtoverlay=uart0,<param>=<val>
Params: txd0_pin                GPIO pin for TXD0 (14, 32 or 36 - default 14)rxd0_pin                GPIO pin for RXD0 (15, 33 or 37 - default 15)pin_func                Alternative pin function - 4(Alt0) for 14&15,7(Alt3) for 32&33, 6(Alt2) for 36&37Name:   uart1
Info:   Change the pin usage of uart1
Load:   dtoverlay=uart1,<param>=<val>
Params: txd1_pin                GPIO pin for TXD1 (14, 32 or 40 - default 14)rxd1_pin                GPIO pin for RXD1 (15, 33 or 41 - default 15)Name:   uart2
Info:   Enable uart 2 on GPIOs 0-3. BCM2711 only.
Load:   dtoverlay=uart2,<param>
Params: ctsrts                  Enable CTS/RTS on GPIOs 2-3 (default off)Name:   uart3
Info:   Enable uart 3 on GPIOs 4-7. BCM2711 only.
Load:   dtoverlay=uart3,<param>
Params: ctsrts                  Enable CTS/RTS on GPIOs 6-7 (default off)Name:   uart4
Info:   Enable uart 4 on GPIOs 8-11. BCM2711 only.
Load:   dtoverlay=uart4,<param>
Params: ctsrts                  Enable CTS/RTS on GPIOs 10-11 (default off)Name:   uart5
Info:   Enable uart 5 on GPIOs 12-15. BCM2711 only.
Load:   dtoverlay=uart5,<param>
Params: ctsrts                  Enable CTS/RTS on GPIOs 14-15 (default off)


2,以下为第二种方法:

sudo raspi-config

找到Interfacing选项,找到serial。
第一个问题是:would you like a login shell to be accessible  over serial? 选否。
第二个问题是would you like the serial port hardware to be enabled?选是。
我用的系统版本是2018-11-13-raspbian-stretch,你的问题顺序可能不一样,看清对应问题选择。

补充

/dev/serial0(对应的shell串口,有引脚引出) ->ttyS0(mini串口,用的是内核时钟不稳定,实际测试关闭shell之后并没有出现过异常情况)
/dev/serial1(对应的蓝牙,没有引脚引出) ->ttyAMA0(硬件串口稳定)

cd /boot/overlays/#提供了README

 

Name:   disable-bt
Info:   Disable onboard Bluetooth on Pi 3B, 3B+, 3A+, 4B and Zero W, restoringUART0/ttyAMA0 over GPIOs 14 & 15.N.B. To disable the systemd service that initialises the modem so itdoesn't use the UART, use 'sudo systemctl disable hciuart'.
Load:   dtoverlay=disable-bt
Params: <None>

README文件中说明了这个文件的功能是将树莓派3的蓝牙切换到mini串口(ttyS0),并且恢复硬件串口(ttyAMA0)到GPIO 14&15脚中。并且给出了载入的方法。

 

ls -l /dev

 此时GPIO映射的串口是默认的/dev/ttyS0这个mini串口。

使用下面这条指令编辑 /boot/config.txt 文件

sudo nano /boot/config.txt

在该文件中增加一行代码

dtoverlay=disable-bt

然后保存文件,重启树莓派使之生效。
再通过 ls -l /dev 命令查看修改后的映射关系

对比修改前的关系,可以看出serial0和serial1 与 ttyAMA0和ttyS0的映射关系对换完成了,也就是ttyAMA0映射到了引出的GPIO Tx Rx上。

禁用串口的控制台功能

前面的步骤已经交换了硬件串口与mini串口的映射关系,但现在想使用树莓派外接串口模块进行通信还不行,因为树莓派IO引出的串口默认是用来做控制台使用的,它的初衷是为了在没有网络接口时,通过串口对树莓派进行相关的配置。因此需要禁用这个默认功能,使得串口为我们自由使用。

树莓派命令窗口中分别通过如下两个命令停止和禁用串口的控制台功能
 

sudo systemctl stop serial-getty@ttyAMA0.service
sudo systemctl disable serial-getty@ttyAMA0.service

由于我们前面已经交换了串口的映射关系,因此这里注意是ttyAMA0。

然后通过下列指令编辑cmdline.txt文件
sudo nano /boot/cmdline.txt

 然后看到里面类似如下的内容

dwc_otg.lpm_enable=0 console=serial0,115200 console=tty1 root=/dev/mmcblk0p2 rootfstype=ext4 elevator=deadline fsck.repair=yes rootwait

把console=serial0,115200删掉 console=serial0,115200 ,剩下的内容类似如下

dwc_otg.lpm_enable=0 console=tty1 root=/dev/mmcblk0p2 rootfstype=ext4 elevator=deadline rootwait

 然后重新启动树莓派,使修改生效

编写一个简单的串口程序

新建一个test.py文件

# -*- coding: utf-8 -*
import serial
import time
# 打开串口
ser = serial.Serial("/dev/ttyAMA0", 9600)
def main():while True:# 获得接收缓冲区字符count = ser.inWaiting()if count != 0:# 读取内容并回显recv = ser.read(count)print(recv)ser.write(recv)# 清空接收缓冲区ser.flushInput()# 必要的软件延时time.sleep(0.1)if __name__ == '__main__':try:main()except KeyboardInterrupt:if ser != None:ser.close()

小插曲

#这里switch_function(reading)返回值是数字,但是ser.write()填入的值是数字则意思是发送几个字节
#所以我们把这里整成字符串,格式是utf-8.
count = str(switch_function(reading))if count != 0:recv = count.encode('utf-8')ser.write(recv)
#python中的多线程def Tmer_task(processed_frame):print("收到识别指令,正在处理...")image_base64 = image_to_base64(processed_frame)if image_base64:print("正在识别垃圾类型,请稍候...")reading = analyze_meter_image(image_base64)if reading:print("=" * 30)print(f"识别结果: {reading}")print("=" * 30)count = str(switch_function(reading))if count != 0:recv = count.encode('utf-8')ser.write(recv)print("已发送指令,请等待垃圾处理完成")else:print("识别失败,请重试")def periodic_task(cap, interval):while True:ret, frame = cap.read()if not ret:print("无法获取摄像头画面")break# 处理图像(现在返回彩色图像)processed_frame = img_p(frame)# 显示实时画面#cv2.imshow('Camera Feed', processed_frame)# 调用任务处理函数Tmer_task(processed_frame)# 等待间隔时间time.sleep(interval)ask_thread = threading.Thread(target=periodic_task, args=(cap, 10))task_thread.start()


http://www.ppmy.cn/ops/137540.html

相关文章

学习HTML第三十三天

学习文章目录 一.fieldset 与 legend 的使用&#xff08;了解&#xff09;二.表单总结三.框架标签 一.fieldset 与 legend 的使用&#xff08;了解&#xff09; fieldset 可以为表单控件分组、 legend 标签是分组的标题 二.表单总结 form表单&#xff1a; action 属性&#…

Python 网络爬虫进阶:动态网页爬取与反爬机制应对

在上一篇文章中&#xff0c;我们学习了如何使用 Python 构建一个基本的网络爬虫。然而&#xff0c;在实际应用中&#xff0c;许多网站使用动态内容加载或实现反爬机制来阻止未经授权的抓取。因此&#xff0c;本篇文章将深入探讨以下进阶主题&#xff1a; 如何处理动态加载的网…

【入门篇】小游戏——多语言求解版

题目链接 样例说明 样例中&#xff1a; 小明的第一次操作是查询&#xff0c;我们可以得到字符串 “010011” 中第一次出现 ‘1’ 的位置为 2 。 小明的第二次操作将下标为 2的字符反转&#xff0c;此时字符串变为 “000011” 。 小明的第三次操作是查询&#xff0c;我们可…

SSM--SpringMVC复习(一)

SpringMVC Spring MVC 是一个实现了MVC设计模式的请求驱动类型的轻量级Web框架&#xff0c; 本质上相当于 Servlet&#xff0c;以 DispatcherServlet 为核心&#xff0c;负责协调和组织不同组件以完成请求处理并返回响应。通过控制器&#xff08;C&#xff09;对模型数据&…

深入解析 Dockerfile:从入门到实践

一、Dockerfile 是什么&#xff1f; Dockerfile 是一个纯文本文件&#xff0c;其中包含了一系列指令&#xff0c;用于定义如何构建 Docker 镜像。通过 Dockerfile&#xff0c;开发者可以实现应用程序环境的一致性和自动化部署。 1.1 Dockerfile 的作用 描述镜像构建过程。自…

深度学习笔记之BERT(三)RoBERTa

深度学习笔记之RoBERTa 引言回顾&#xff1a;BERT的预训练策略RoBERTa训练过程分析静态掩码与动态掩码的比较模型输入模式与下一句预测使用大批量进行训练使用Byte-pair Encoding作为子词词元化算法更大的数据集和更多的训练步骤 RoBERTa配置 引言 本节将介绍一种基于 BERT \t…

后端接受大写参数(亲测能用)

重要点引入包别引用错了 import com.fasterxml.jackson.databind.annotation.JsonSerialize; import com.fasterxml.jackson.annotation.JsonProperty; import lombok.Data;JsonSerialize Data public class Item {JsonProperty(value "Token")private String token…

第144场双周赛题解:两个字符串得切换距离

两个字符串得切换距离 1、题目描述 给你两个长度相同的字符串 s 和 t &#xff0c;以及两个整数数组 nextCost 和 previousCost 。 一次操作中&#xff0c;你可以选择 s 中的一个下标 i &#xff0c;执行以下操作 之一 &#xff1a; 将 s[i] 切换为字母表中的下一个字母&am…