RP2040制作水平仪-2022年寒假在家练RP2040游戏机项目分享(五)

电子森林 2022-03-15 21:56
RP2040GameKit也能制作水平仪,RP2040可太强大了,以后在家也不会无聊啦。

项目来自南京信息工程大学的【Godalin】同学的分享,并已开源在电子森林,大家可点击“阅读原文”获取,转载请注明出处。




 1

项目介绍

本项目使用硬禾学堂在“2022年寒假在家一起练”活动中提供的基于树莓派RP2040的嵌入式系统学习平台,使用 MicroPython平台进行编程,利用平台提供的姿态传感器以及LCD显示器实现一个简单的水平仪应用。水平仪应用通过一个滚动的小球或气泡,来显示当前板子的倾斜度,当板子处于水平位置的时候,小球停在屏幕的正中间,倾斜板子,小球偏移,并能够显示偏移的角度(二维信息)。


2

项目需求

  • 完成对MMA7660FC姿态传感器的数据采集。

  • 完成对MMA7660FC姿态传感器采集到的数据的分析与倾角的计算。

  • 实现对ST7789LCD屏幕的驱动。

  • 将每一时刻的倾角体现为“气泡”与“液体”相对位置,在屏幕上的具体位置(圆盘和导管)绘制气泡,并发送到屏幕上,并把倾角的角度值显示到屏幕上。


 3

设计思路

1. 程序流程图

2. 建立一个模块命名为GOS.py(Game kit Operating System),简单地封装对于硬件资源的直接调用。GOS.py模块下包含三个类的定义,分别为ControlDisplayGesture。其中Control类中包含了对于按钮和遥感状态的抽象,在水平仪的应用中用到得不多,故不在此具体描述。

3. Display类封装了初始化屏幕的代码以及一个名为UI的内部类。初始化代码包含了一个简单的开机动画(仅仅是显示了自己的id)。UI类定义了显示在屏幕上的部件的位置和尺寸信息,考虑到计算的复杂性,并未存储buffer信息。新的UI部件可以直接通过继承UI类来获取相应的位置和尺寸信息。

4. Gesture类封装了姿态传感器的初始化代码以及读取三轴倾角值的函数,函数的角度支持弧度制以及角度值。通过封装可以在调用的时候不过多地考虑底层实现的细节。

main.py模块包含对于水平仪应用(app)的导入,这么设计是为了考虑到将来对于应用的可拓展性(支持多个应用)。

5. 水平仪应用GOS_App_spirit_level.py是水平仪具体是实现的文件。其中包括:

对于硬件的初始化,包括屏幕,控制器(按钮和摇杆)以及姿态传感器。

6. 打开垃圾回收机制,防止内存不够用。
  • 定义一个容器类Container继承于Display.UI,包含了一块内存空间作为buffer,用于存储每一时刻的气泡和容器的位置,可以通过显示函数发送到屏幕。
  • 主循环中,每次迭代都计算了姿态传感器发送的姿态信息,转换为气泡在容器上的位置后写入容器的buffer,再发送到屏幕上。

 4

硬件介绍

  • RP2040是Raspberry Pi 的首款微控制器。它为微控制器领域带来了我们高性能、低成本和易用性的标志性价值。RP2040 提供了硬件SPI总线以及硬件 I2C 总线各两组,我们可以使用一组SPI总线控制LCD屏幕,使用一组I2C总线接收姿态传感器的返回数据。
  • MMA7660FC是具有数字输出的I2C、低功耗、紧凑型电容式微机械加速度传感器,提供低通滤波器、零重力加速度偏移和增益误差补偿,并可以转化为6位数字值,用户可配置输出数据的传输速率。该器件可通过中断引脚(INT)识别传感器的数据变化、产品的朝向和姿态等。MMA7660FC采用非常小的3mmx3mmx0.9mmDFN封装。
  • LCD显示器使用的是ST7789,240x240分辨率,使用一组SPI总线与微控制器通信。


 5

功能实现

1. 普通水平状态,气泡基本在正中间,略微有点偏移可能是因为桌面或者系统误差

2. 左上角显示的是三轴角度信息,只需要关注前两个数据

3. 任意角度可以正常偏移气泡

4. 极限位置,气泡在圆周内


 6

主要代码片段及说明

这里我将用代码与注释的方式进行说明

1. Display类的定义

class Display:     '''用于驱动屏幕以及提供组件的基类'''     def __init__(self):         '''初始化,定义SPI'''         spi = SPI(0, baudrate=40_000_000, polarity=1, phase=1,                   sck=Pin(2, Pin.OUT),                   mosi=Pin(3, Pin.OUT))         self.screen = st7789c.ST7789(             spi, 240, 240,             reset=Pin(0, Pin.OUT),             dc=Pin(1, Pin.OUT),             rotation=0)         self.screen.init()
def setup(self): '''显示开机图像''' self.screen.fill(st7789c.BLACK) self.screen.text(font2, "Powered By", 10, 10) self.screen.text(font2, "Godalin", 10, 50) utime.sleep(5)
class UI: '''组件的基类,提供位置与尺寸的信息''' def __init__(self, x, y, w, h): self.x = x self.y = y self.w = w self.h = h
@property def position(self): return (self.x, self.y, self.w, self.h)
@property def center(self): return (self.x + self.w // 2, self.y + self.h // 2)


2. Gesture类的定义

class Gesture():     '''用于包装姿态传感器'''
def __init__(self): self.sensor = mma7660fc.MMA7660FC()
def angles_h(self, mode="rad"): '''用于获取角度,可选角度数或者弧度数 将重力加速度在三轴的分量转化计算出三个轴的倾角 ''' Ax, Ay, Az = self.sensor.read_accl()
tx = math.atan2(Ax, math.sqrt(Ay * Ay + Az * Az)) ty = math.atan2(Ay, math.sqrt(Ax * Ax + Az * Az)) tz = math.atan2(Az, math.sqrt(Ax * Ax + Ay * Ay))
if mode == "rad": return tx, ty, tz
elif mode == "degree": tx = tx / math.pi * 180 ty = ty / math.pi * 180 tz = tz / math.pi * 180
return tx, ty, tz


3. App模块主循环

while True:     # 获取一次倾角测量     tx_t, ty_t, tz_t = gest.angles_h()
# 进行滑动平均 tx = avg(tx, tx_t, beta) ty = avg(ty, ty_t, beta) tz = avg(tz, tz_t, beta)
# 计算一些值,用于计算气泡的位置 pos_x = 1 - math.sin(ty) pos_y = 1 - math.sin(tx)
# 显示角度文字的更新 disp.text(font1, "X:{:+03d} deg".format(int(tx / math.pi * 180)), 10, 10, st7789.WHITE, st7789.RED) disp.text(font1, "Y:{:+03d} deg".format(int(ty / math.pi * 180)), 10, 20, st7789.WHITE, st7789.RED) disp.text(font1, "Z:{:+03d} deg".format(int(tz / math.pi * 180)), 10, 30, st7789.WHITE, st7789.RED)
# 先绘制气泡再进行刻度和文字的绘制,形成文字在玻璃上,气泡在下方的效果 # 气泡位置的更新 rail1.update(bubble, int(50 * pos_x), 0) rail2.update(bubble, 0, int(50 * pos_y)) plate.update(bubble, int(80 * pos_x), int(80 * pos_y))
# 绘制圆形刻度线以及水平垂直刻度线 rail1.buffer.fill_rect(29, 0, 2, 20, red) rail1.buffer.fill_rect(59, 0, 2, 20, red) rail1.buffer.fill_rect(89, 0, 2, 20, red) rail2.buffer.fill_rect(0, 29, 20, 2, red) rail2.buffer.fill_rect(0, 59, 20, 2, red) rail2.buffer.fill_rect(0, 89, 20, 2, red)
circle(plate.buffer, 1, 1, 178, red) circle(plate.buffer, 46, 46, 88, red) plate.buffer.fill_rect(89, 1, 2, 178, red) plate.buffer.fill_rect(1, 89, 178, 2, red)
# 书写文字刻度 for s, ly, lx in zip(scales, loc_y, loc_x): plate.buffer.text(s, 92, ly, red) plate.buffer.text(s, lx, 93, red)
# 将三部分含气泡的部分发送给屏幕 disp.blit_buffer(rail1.buffer, *rail1.position) disp.blit_buffer(rail2.buffer, *rail2.position) disp.blit_buffer(plate.buffer, *plate.position)
# 延时 utime.sleep_ms(10)


 7

遇到的主要难题及解决方法

1. MicroPython脚本版本的屏幕驱动库性能较低,无法实现流畅的屏幕刷新。

解决方法:使用交流群中老师提供的MicroPython固件,其中包含了用C语言写的st7789c拓展库,使用这个库可以实现高速刷屏,流畅程度很高,肉眼看不出刷屏的时间间隙。


2. 标准图形库framebuf使用color565编码的颜色(由LCD屏幕提供)无法正常显示,蓝色0x001f被显示为浅绿色,红色0xf800被显示为蓝色。

解决方法:查找st7789c源代码仓库,发现其中的一条issue:指出FrameBuffer类使用的数据编码是little endian,小端,而屏幕的blit_buffer方法使用的是big endian,大端,两种颜色的编码相反,所以显示异常。该issue提供了下面的函数以供进行大小端的转换:

# 颜色编码转换 def swap_rgb565(color):  color = int.from_bytes(color.to_bytes(2, 'little'), 'big', False) return color


3. 由于想做美观的水平仪UI,需要用到圆形的绘制,而st7789c库以及framebuf标准库中并未提供相应的方法进行圆形的绘制,故需要自己写一个快速的算法进行绘制圆形。

解决方法:通过网络查找,找到一个名为中点圆的算法很符合需求,而且很快速。大致的原理是,只需要计算八分之一的圆所需要的像素,利用对称性每次绘制八个点来画圆。原帖的算法指定了圆心和半径的大小,而我修改算法,改为指定直径以及圆外接正方形的左上顶点坐标来绘制圆形,以实现对于任意直径的支持。代码保存在cirle.py模块中,具体如下:

# 中点圆算法 def circle(display, x0, y0, d, color):     r = d >> 1     xc = x0 + r     yc = y0 + r     if d == r << 1:         r -= 1         d = 3 - (r << 1)
def circle_plot(x, y): display.pixel(xc + x, yc + y, color) display.pixel(xc + x, yc - y - 1, color) display.pixel(xc - x - 1, yc + y, color) display.pixel(xc - x - 1, yc - y - 1, color) display.pixel(xc + y, yc + x, color) display.pixel(xc + y, yc - x - 1, color) display.pixel(xc - y - 1, yc + x, color) display.pixel(xc - y - 1, yc - x - 1, color) else: def circle_plot(x, y): display.pixel(xc + x, yc + y, color) display.pixel(xc + x, yc - y, color) display.pixel(xc - x, yc + y, color) display.pixel(xc - x, yc - y, color) display.pixel(xc + y, yc + x, color) display.pixel(xc + y, yc - x, color) display.pixel(xc - y, yc + x, color) display.pixel(xc - y, yc - x, color)
xi = 0 yi = r while True: circle_plot(xi, yi) if d < 0: d = d + (xi << 2) + 6 else: d = d + (xi - yi << 2) + 10 yi -= 1 xi += 1
if xi > yi: break


4. 由于姿态传感器的噪声,并不能在摆弄的时候形成平稳的动画。

解决方案:采用的滑动平均值进行滤波:

# 滑动平均值函数def avg(avg, this, beta):     return avg * (1 - beta) + this * beta

其中avg是第上次迭代获取的平均值,返回值是新的平均值作为需要的输出,this是本次迭代的观测值。可以取avg=0作为初始值。程序中我采用beta=0.2,获得不错的效果,并且得益于取平均值的过程,模拟出了真实世界的惯性效果。


 8

未来的计划或建议

  • 水平仪由于自身工作需求,姿态传感器的模式始终保持active模式,因此可能具有不必要的性能损失。MMA7660FC有一个中断引脚,还没仔细研究,可以将其利用起来实现一些功能,如节约性能。
  • 水平仪由于硬件本身的工作原理或者制作工艺,存在一定的偏差,可以设计一种校准的功能,以某个姿态作为原点的基准,在此功能下可以实现校准。当然也可以用传感器的原始数据作为结果,这种功能仍会保留。
  • 将某些耗时的计算用C语言实现,用MicroPython提供的工具编译为mpy文件,即原生字节码以提高效率。目前这个技术我只了解了一点点,并没有很好地掌握。或者直接使用Pico C SDK进行开发,提高性能。目前Pico C SDK开发在朋友的帮助下克服了flash空间大小的限制,实现了Bad Apple!!的播放,后续可以上传到视频网站。
  • 在开发的过程中,虽然建立了较高层次的系统抽象以方便地调用硬件,但是并未把按钮和摇杆利用起来。后续如果做了多个APP,可以在系统层面写图形化见面或者shell,有效地把硬件资源利用起来。


END


硬禾学堂

硬禾团队一直致力于给电子工程师和相关专业的同学,带来规范的核心技能课程,帮助大家在学习和工作的各个阶段,都能有效地提升自己的职业能力。

硬禾学堂

我们一起在电子领域探索前进

关注硬禾公众号,随时直达课堂


点击阅读原文下载代码



电子森林 讲述电子工程师需要掌握的重要技能: PCB设计、FPGA应用、模拟信号链路、电源管理等等;不断刷新的行业新技术 - 树莓派、ESP32、Arduino等开源系统;随时代演进的热点应用 - 物联网、无人驾驶、人工智能....
评论
  • 数字隔离芯片是现代电气工程师在进行电路设计时所必须考虑的一种电子元件,主要用于保护低压控制电路中敏感电子设备的稳定运行与操作人员的人身安全。其不仅能隔离两个或多个高低压回路之间的电气联系,还能防止漏电流、共模噪声与浪涌等干扰信号的传播,有效增强电路间信号传输的抗干扰能力,同时提升电子系统的电磁兼容性与通信稳定性。容耦隔离芯片的典型应用原理图值得一提的是,在电子电路中引入隔离措施会带来传输延迟、功耗增加、成本增加与尺寸增加等问题,而数字隔离芯片的目标就是尽可能消除这些不利影响,同时满足安全法规的要
    华普微HOPERF 2025-01-15 09:48 83浏览
  • PNT、GNSS、GPS均是卫星定位和导航相关领域中的常见缩写词,他们经常会被用到,且在很多情况下会被等同使用或替换使用。我们会把定位导航功能测试叫做PNT性能测试,也会叫做GNSS性能测试。我们会把定位导航终端叫做GNSS模块,也会叫做GPS模块。但是实际上他们之间是有一些重要的区别。伴随着技术发展与越发深入,我们有必要对这三个词汇做以清晰的区分。一、什么是GPS?GPS是Global Positioning System(全球定位系统)的缩写,它是美国建立的全球卫星定位导航系统,是GNSS概
    德思特测试测量 2025-01-13 15:42 498浏览
  • 随着数字化的不断推进,LED显示屏行业对4K、8K等超高清画质的需求日益提升。与此同时,Mini及Micro LED技术的日益成熟,推动了间距小于1.2 Pitch的Mini、Micro LED显示屏的快速发展。这类显示屏不仅画质卓越,而且尺寸适中,通常在110至1000英寸之间,非常适合应用于电影院、监控中心、大型会议、以及电影拍摄等多种室内场景。鉴于室内LED显示屏与用户距离较近,因此对于噪音控制、体积小型化、冗余备份能力及电气安全性的要求尤为严格。为满足这一市场需求,开关电源技术推出了专为
    晶台光耦 2025-01-13 10:42 507浏览
  • 新年伊始,又到了对去年做总结,对今年做展望的时刻 不知道你在2024年初立的Flag都实现了吗? 2025年对自己又有什么新的期待呢? 2024年注定是不平凡的一年, 一年里我测评了50余块开发板, 写出了很多科普文章, 从一个小小的工作室成长为科工公司。 展望2025年, 中国香河英茂科工, 会继续深耕于,具身机器人、飞行器、物联网等方面的研发, 我觉得,要向未来学习未来, 未来是什么? 是掌握在孩子们生活中的发现,和精历, 把最好的技术带给孩子,
    丙丁先生 2025-01-11 11:35 463浏览
  • ARMv8-A是ARM公司为满足新需求而重新设计的一个架构,是近20年来ARM架构变动最大的一次。以下是对ARMv8-A的详细介绍: 1. 背景介绍    ARM公司最初并未涉足PC市场,其产品主要针对功耗敏感的移动设备。     随着技术的发展和市场需求的变化,ARM开始扩展到企业设备、服务器等领域,这要求其架构能够支持更大的内存和更复杂的计算任务。 2. 架构特点    ARMv8-A引入了Execution State(执行状
    丙丁先生 2025-01-12 10:30 471浏览
  • 流量传感器是实现对燃气、废气、生活用水、污水、冷却液、石油等各种流体流量精准计量的关键手段。但随着工业自动化、数字化、智能化与低碳化进程的不断加速,采用传统机械式检测方式的流量传感器已不能满足当代流体计量行业对于测量精度、测量范围、使用寿命与维护成本等方面的精细需求。流量传感器的应用场景(部分)超声波流量传感器,是一种利用超声波技术测量流体流量的新型传感器,其主要通过发射超声波信号并接收反射回来的信号,根据超声波在流体中传播的时间、幅度或相位变化等参数,间接计算流体的流量,具有非侵入式测量、高精
    华普微HOPERF 2025-01-13 14:18 491浏览
  • 在不断发展的电子元件领域,继电器——作为切换电路的关键设备,正在经历前所未有的技术变革。固态继电器(SSR)和机械继电器之间的争论由来已久。然而,从未来发展的角度来看,固态继电器正逐渐占据上风。本文将从耐用性、速度和能效三个方面,全面剖析固态继电器为何更具优势,并探讨其在行业中的应用与发展趋势。1. 耐用性:经久耐用的设计机械继电器:机械继电器依靠物理触点完成电路切换。然而,随着时间的推移,这些触点因电弧、氧化和材料老化而逐渐磨损,导致其使用寿命有限。因此,它们更适合低频或对切换耐久性要求不高的
    腾恩科技-彭工 2025-01-10 16:15 103浏览
  • 根据Global Info Research(环洋市场咨询)项目团队最新调研,预计2030年全球无人机电池和电源产值达到2834百万美元,2024-2030年期间年复合增长率CAGR为10.1%。 无人机电池是为无人机提供动力并使其飞行的关键。无人机使用的电池类型因无人机的大小和型号而异。一些常见的无人机电池类型包括锂聚合物(LiPo)电池、锂离子电池和镍氢(NiMH)电池。锂聚合物电池是最常用的无人机电池类型,因为其能量密度高、设计轻巧。这些电池以输出功率大、飞行时间长而著称。不过,它们需要
    GIRtina 2025-01-13 10:49 198浏览
  • 01. 什么是过程能力分析?过程能力研究利用生产过程中初始一批产品的数据,预测制造过程是否能够稳定地生产符合规格的产品。可以把它想象成一种预测。通过历史数据的分析,推断未来是否可以依赖该工艺持续生产高质量产品。客户可能会要求将过程能力研究作为生产件批准程序 (PPAP) 的一部分。这是为了确保制造过程能够持续稳定地生产合格的产品。02. 基本概念在定义制造过程时,目标是确保生产的零件符合上下规格限 (USL 和 LSL)。过程能力衡量制造过程能多大程度上稳定地生产符合规格的产品。核心概念很简单:
    优思学院 2025-01-12 15:43 529浏览
  • 食物浪费已成为全球亟待解决的严峻挑战,并对环境和经济造成了重大影响。最新统计数据显示,全球高达三分之一的粮食在生产过程中损失或被无谓浪费,这不仅导致了资源消耗,还加剧了温室气体排放,并带来了巨大经济损失。全球领先的光学解决方案供应商艾迈斯欧司朗(SIX:AMS)近日宣布,艾迈斯欧司朗基于AS7341多光谱传感器开发的创新应用来解决食物浪费这一全球性难题。其多光谱传感解决方案为农业与食品行业带来深远变革,该技术通过精确判定最佳收获时机,提升质量控制水平,并在整个供应链中有效减少浪费。 在2024
    艾迈斯欧司朗 2025-01-14 18:45 68浏览
  •   在信号处理过程中,由于信号的时域截断会导致频谱扩展泄露现象。那么导致频谱泄露发生的根本原因是什么?又该采取什么样的改善方法。本文以ADC性能指标的测试场景为例,探讨了对ADC的输出结果进行非周期截断所带来的影响及问题总结。 两个点   为了更好的分析或处理信号,实际应用时需要从频域而非时域的角度观察原信号。但物理意义上只能直接获取信号的时域信息,为了得到信号的频域信息需要利用傅里叶变换这个工具计算出原信号的频谱函数。但对于计算机来说实现这种计算需要面对两个问题: 1.
    TIAN301 2025-01-14 14:15 113浏览
  • 随着全球向绿色能源转型的加速,对高效、可靠和环保元件的需求从未如此强烈。在这种背景下,国产固态继电器(SSR)在实现太阳能逆变器、风力涡轮机和储能系统等关键技术方面发挥着关键作用。本文探讨了绿色能源系统背景下中国固态继电器行业的前景,并强调了2025年的前景。 1.对绿色能源解决方案日益增长的需求绿色能源系统依靠先进的电源管理技术来最大限度地提高效率并最大限度地减少损失。固态继电器以其耐用性、快速开关速度和抗机械磨损而闻名,正日益成为传统机电继电器的首选。可再生能源(尤其是太阳能和风能
    克里雅半导体科技 2025-01-10 16:18 328浏览
  • 随着通信技术的迅速发展,现代通信设备需要更高效、可靠且紧凑的解决方案来应对日益复杂的系统。中国自主研发和制造的国产接口芯片,正逐渐成为通信设备(从5G基站到工业通信模块)中的重要基石。这些芯片凭借卓越性能、成本效益及灵活性,满足了现代通信基础设施的多样化需求。 1. 接口芯片在通信设备中的关键作用接口芯片作为数据交互的桥梁,是通信设备中不可或缺的核心组件。它们在设备内的各种子系统之间实现无缝数据传输,支持高速数据交换、协议转换和信号调节等功能。无论是5G基站中的数据处理,还是物联网网关
    克里雅半导体科技 2025-01-10 16:20 449浏览
我要评论
0
点击右上角,分享到朋友圈 我知道啦
请使用浏览器分享功能 我知道啦