IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 嵌入式 -> 利用中断在 ESP32 MicroPython 程序中读取AM2302 -> 正文阅读

[嵌入式]利用中断在 ESP32 MicroPython 程序中读取AM2302

简 介: 由于使用MicroPython,如果使用指令查询方式读取AM2302的数据,无法控制到非常精确读取时间信息。利用ESP32 MicroPython中的IO中断,获取AM2302的上升沿时间,利用ticks_cpu() 对于上升沿进行标定。然后在通过比较两个上升沿时间的差异,获取输出0,1数据。

通过数据验证测试,可以验证这种方式可以比较方便完成AM2302的读取。

关键词 AM2302ESP32MicroPython

?

§01 湿度传感器


一、AM2302

??AM2302 是一款具有单总线接口的温湿度传感器。接口协议与 1-Wire 有区别。由于该款传感器比较简洁,它只发送转换后的数据。因此输出数据格式比较简单。

1、总线协议

(1) 硬件连接关系

??其中上拉电阻,在传输线的距离小于20米是选择 5k Ω; 大于20日是需要根据实际情况选择合适的上拉电阻。

▲ 图1.1.1 传感器与MCU之间的连接关系

▲ 图1.1.1 传感器与MCU之间的连接关系

(2) 总线数据格式

▲ 图1.1.2 总线数据格式

▲ 图1.1.2 总线数据格式

(3) 数据交互过程

??主机通过发送一次开始信号(超过 500us低电平),从机AM2302变开始发送转换数据了。

▲ 图1.1.3.1 数据发送格式

▲ 图1.1.3.1 数据发送格式

??总线空闲的时候为高电平,一次数据传输通讯是由主机(MCU)拉低总线 1 ~ 10ms 之后释放总线,延时 20 ~ 40us之后,主机便开始检测从机(AM2302)的输出响应信号。

??从机开始应答开始部分,发送出一个80us低电平。

▲ 图1.1.3.2 从机开始应答开始部分,发送出一个80us低电平

▲ 图1.1.3.2 从机开始应答开始部分,发送出一个80us低电平

??下面给出了 DHT发送数字1, 0 的波形。

▲ 图1.1.3.3 发送数字信号1的波形

▲ 图1.1.3.3 发送数字信号1的波形

▲ 图1.1.3.4 发送数字0的波形

▲ 图1.1.3.4 发送数字0的波形

二、读取方法

1、使用STM32查询方法

??由于单片机STM32运行速度比较快,因此可以通过软件延时的方式来获得AM2302的输出波数据。

2、使用端口中断

??也可以利用单片机的端口中断完成对于总线波形时刻的记录,然后在通过对于中断后时刻判断输出的数据位。

?

§02 MicroPython读取


??果在MicroPython环境下,MicroPython执行比较缓慢,因此采用查询方式获得数据会存在困难。而直接使用 ESP32 MicroPython 1-Wire 协议读取AM2302会存在错误。这在 ESP32 One-Wire驱动功能 测试中也反映直接使用ESP32 MicroPython 的1-wire 驱动读取AM2302存在错误。

一、中断读取AM2302数据

1、基本原理

??在 微型角度编码器 : KYTB-1503-1024 测试了 ESP32 MicroPython 的端口相应外部脉冲中断的能力。可以看到ESP32最大的脉冲响应速率为26kHz。比起AM2302的单个数据输出周期小,因此可以借助ESP32 MicroPython 端口中断读取定时器的时间来记录 AM2302输出脉冲的宽窄内容。

2、实验配置

??使用 ESP32-S模块转接板设计与实现 模块的输出 GPIO2作为触发AM2302总线的端口。

(1) 测试硬件平台

▲ 图2.1.1  测试平台

▲ 图2.1.1 测试平台

(2) 测试代码

from machine                import Pin,Timer
import time

io1 = Pin(2, Pin.OUT)
io1.on()

while True:
    io1.off()
    time.sleep_ms(5)
    io1.on()

    time.sleep_ms(1000)

(3) 测试波形

??测量对应的波形如下图所示。由于ESP32输出高平,使得AM2302相应信号变成了高电平。因此,需要将ESP32设置成高阻形式来读取AM2320响应数据。

▲ 图2.1.2  测量IO波形

▲ 图2.1.2 测量IO波形

??将ESP32的GPIO的模式修改成 OPEN_DRAIN,重新读取总线数据波形。

io1 = Pin(2, Pin.OPEN_DRAIN)
io1.on()

▲ 图2.1.3  将输出模式改成 OPEN_DRAIN 之后得到的波形

▲ 图2.1.3 将输出模式改成 OPEN_DRAIN 之后得到的波形

▲ 图2.1.4 展开IO数据波形

▲ 图2.1.4 展开IO数据波形

3、记录时间

??使用ESP32的 time.tick_us(), time_tick_cpu() 获得ESP32的高分辨率的时钟数据。

(1) 测试代码

from machine                import Pin,Timer
import time

io1 = Pin(2, Pin.OPEN_DRAIN)
io1.on()

bufLength = 128
timebuf = [0] * bufLength
timepoint = 0

def ISR_io(pin):
    global bufLength, timebuf, timepoint

    t = time.ticks_cpu()

    if timepoint < bufLength:
        timebuf[timepoint] = t
        timepoint += 1

io1.irq(trigger=Pin.IRQ_RISING, handler=ISR_io)

while True:
    io1.off()
    time.sleep_ms(5)
    io1.on()

    time.sleep_ms(5000)

    print(timebuf)
    timepoint = 0
[252517663, 252532392, 252556712, 252568518, 252583742, 252596740, 252609738, 252623106, 252638330, 252650136, 252672386, 252687610, 252707426, 252722650, 252734456, 252748226, 252763450, 252783266, 252798480, 252811478, 252824706, 252839930, 252851736, 252865506, 252880730, 252892536, 252906946, 252929358, 252950798, 252972238, 252993678, 253015118, 253036558, 253049556, 253064780, 253077778, 253091964, 253112064, 253134478, 253155918, 253177358, 253198798, 253220878, 253233876, 253247124, 253262102, 253275100, 253288098, 253301504, 253316728, 253329726, 253342944, 253358168, 253369974, 253383742, 253398966, 253410772, 253424542, 253439756, 253452754, 253465982, 253481206, 253493012, 253506782, 253522006, 253533812, 253546810, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0]

4、数据分析

(1) 时间取值

▲ 图2.1.4.1 读取的时间递增波形

▲ 图2.1.4.1 读取的时间递增波形

(2) 时间差值

from headm import *

t = tspload('time', 't')

t = [tt for tt in t if tt > 0]
td = [1 if x-y > 18000 else 0 for x,y in zip(t[1:], t[:-1])]

printf(td)

printf(len(td))

plt.plot(td)
plt.xlabel("Sample")
plt.ylabel("Time")
plt.grid(True)
plt.tight_layout()
plt.show()

▲ 图2.1.4.2 读取的时间递增波形

▲ 图2.1.4.2 读取的时间递增波形

(3) 译码0-1

▲ 图2.1.4.3 时间译码01 波形

▲ 图2.1.4.3 时间译码01 波形

(4) 确定起始数据位

??从前慢读取的数据来看,其中存在的脉冲个数为66,大约应该具有的40bit的数据,这说明其中存在一些无需要的 0。

??需要根据校验和,来确定起始的位数。

t = [tt for tt in t if tt > 0]
td = [1 if x-y > 18000 else 0 for x,y in zip(t[1:], t[:-1])]
#td = [x-y for x,y in zip(t[1:], t[:-1])]

#------------------------------------------------------------
def bits2byte(bits):
    return sum([b*(2**(len(bits)-e-1)) for b,e in zip(bits, list(range(len(bits))))])

def buf01_buf(buf, beginid=0):
    num = []
    for i in range(5):
        id = beginid + i*8
        nbits = buf[id:id+8]
        num.append(bits2byte(nbits))

    return num

for i in range(10):
    num = buf01_buf(td, i)
    printff(i, sum(num[0:4]) & 0xff, num[4])

??下面校验了从beginid cs 0 变化到9,可以看到从0,1,2,3 都可以完成校验。

0 16 16
1 32 33
2 66 66
3 133 132
4 13 8
5 27 16
6 55 32
7 111 64
8 224 128
9 193 0

??最后在根据输出的数字合理的数值范围,可以验证等beginid=2时,对应的数字帅帅正常的数值范围。

2 66 66
325
252

二、连续读取数据

1、完整的ESP32程序

from machine                import Pin,Timer
import time

io1 = Pin(2, Pin.OPEN_DRAIN)
io1.on()

bufLength = 128
timebuf = [0] * bufLength
timepoint = 0

def ISR_io(pin):
    global bufLength, timebuf, timepoint

    t = time.ticks_cpu()

    if timepoint < bufLength:
        timebuf[timepoint] = t
        timepoint += 1

io1.irq(trigger=Pin.IRQ_RISING, handler=ISR_io)

def bits2byte(bits):
    return sum([b*(2**(len(bits)-e-1)) for b,e in zip(bits, list(range(len(bits))))])

def buf01_buf(buf, beginid=0):
    num = []
    for i in range(5):
        id = beginid + i*8
        nbits = buf[id:id+8]
        num.append(bits2byte(nbits))

    return num

def timebuf2data():
    t = [tt for tt in timebuf if tt > 0]
    td = [1 if x-y > 18000 else 0 for x,y in zip(t[1:], t[:-1])]
    if len(td) < 40: return 0,0,1

    num = buf01_buf(td, 2)

    humidity = num[0]*256 + num[1]
    temperature = num[2]*256 + num[3]

    error = 0
    check = sum(num[0:4])&0xff
    if check != num[4]:
        error = 1

    return humidity, temperature, error

while True:
    io1.off()
    time.sleep_ms(5)
    io1.on()

    time.sleep_ms(500)

    h,t,e = timebuf2data()

    if e == 0:
        print((h,t))

    timepoint = 0
    timebuf = [0]*bufLength

2、读取数据曲线

??在下面数据中,首先使用手拿起AM2302传感器,然后通过往其中吹气增加其中的湿度以及温度。可以看到这个参量发生着改变。

▲ 图2.2.1  连续读取的数值

▲ 图2.2.1 连续读取的数值

?

验结论 ※


??然ESP32 的MicroPython环境具有1-wire驱动协议,但对于 AM2302的单线输出数据波形,直接使用ESP32并不能够读取AM2302的数据。

??由于使用MicroPython,如果使用指令查询方式读取AM2302的数据,无法控制到非常精确读取时间信息。利用ESP32 MicroPython中的IO中断,获取AM2302的上升沿时间,利用ticks_cpu() 对于上升沿进行标定。然后在通过比较两个上升沿时间的差异,获取输出0,1数据。

??通过数据验证测试,可以验证这种方式可以比较方便完成AM2302的读取。


■ 相关文献链接:

● 相关图表链接:

  嵌入式 最新文章
基于高精度单片机开发红外测温仪方案
89C51单片机与DAC0832
基于51单片机宠物自动投料喂食器控制系统仿
《痞子衡嵌入式半月刊》 第 68 期
多思计组实验实验七 简单模型机实验
CSC7720
启明智显分享| ESP32学习笔记参考--PWM(脉冲
STM32初探
STM32 总结
【STM32】CubeMX例程四---定时器中断(附工
上一篇文章      下一篇文章      查看所有文章
加:2021-09-23 11:37:35  更:2021-09-23 11:37:45 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/26 2:31:02-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码