开源一夏|在 STM32L051 上使用 RT-Thread (三、无线温湿度传感器 之 I2C 通讯)
前言
经过前面两篇文章的准备,我们终于可以开始写应用代码了,在添加我们自己的驱动文件之前,本文会说明一下 RT-Thread Studio 如何添加自己的.c 和 .h 文件。
然后在此基础上,完成 SHT21 温湿度传感器的 I2C 驱动移植。
本 RT-Thread 专栏记录的开发环境:
RT-Thread 记录(一、RT-Thread 版本、RT-Thread Studio 开发环境 及 配合 CubeMX 开发快速上手)
https://xie.infoq.cn/article/44be1057caace7a6a2c4c4b59
RT-Thread 记录(二、RT-Thread 内核启动流程 — 启动文件和源码分析)
https://xie.infoq.cn/article/44be1057caace7a6a2c4c4b59
RT-Thread 内核篇系列博文链接:
RT-Thread 记录(三、RT-Thread 线程操作函数及线程管理与 FreeRTOS 的比较)
https://xie.infoq.cn/article/1d2e8e030ae689d6b8ee44b05
RT-Thread 记录(四、RT-Thread 时钟节拍和软件定时器)
https://xie.infoq.cn/article/3198c9b741782036bfd6e54e9
RT-Thread 记录(五、RT-Thread 临界区保护
https://xie.infoq.cn/article/7a41020e03184950664df7391
RT-Thread 记录(六、IPC 机制之信号量、互斥量和事件集)
https://xie.infoq.cn/article/1f49bfd6c69377deb9eee838f
RT-Thread 记录(七、IPC 机制之邮箱、消息队列)
https://xie.infoq.cn/article/360b04e7bc6024917afecef1d
RT-Thread 记录(八、理解 RT-Thread 内存管理)
https://xie.infoq.cn/article/bd5d8f8a19fa11ea9feacf34d
RT-Thread 记录(九、RT-Thread 中断处理与阶段小结)
https://xie.infoq.cn/article/3da9530cfac06feece3523a0c
RT-Thread 应用篇系列博文连接:
在 STM32L051 上使用 RT-Thread (一、无线温湿度传感器 之 新建项目)
https://xie.infoq.cn/article/49028076c9f97e06b119e67d4
在 STM32L051 上使用 RT-Thread (二、无线温湿度传感器 之 CubeMX 配置)
一、RT-Thread Studio 添加自定义的.c .h 文件
使用 RT-Thread Studio 添加自定义的.c .h 文件有多种方式,本文根据博主的实际测试,分别说明一下。
1.1 在现有文件夹下添加
直接在工程现有的文件夹下面右击,选择新建头文件或者源文件,如下图:
添加完成以后就可以直接编译了,上面我在 cubemx 文件夹下的 Src 和 Inc 下分别添加了i2c.c
和两个头文件,直接编译会关联进去的:
但是这里得说明下,cubemx 是通过SConscript
受 SCons 构建工具管理的,所以直接这样添加的话,每次我们自己增加一个.c 文件,需要在 SConscript
文件中增加对应部分,要不然同步一下 scons,不相关的.c 文件就会被排除构建:
当然我们也可以在其他文件夹下添加,比如 drivers 文件夹,里面.c.h 文件是混在一起,我们也可以直接在这里面新建,如下图:
总结一下,在现有文件夹下面添加是比较方便的一种方式,因为项目整体的构建系统已经处理好了这些文件下的.c.h 文件的包含关系,我们不用再进行添加头文件路径这种操作。
但是这种方式会影响项目原始框架,小项目的文件少倒是无所谓,文件多了的话就很乱了。
1.2 新建文件夹添加
所以我们要讲一下如果新建文件夹放置我们自己的驱动程序应该怎么做。
首先,还是右击选择新建文件夹,然后出选择文件夹放置的位置,输入文件夹名字:
新建好以后,我们在左边资源管理器可以看到有文件夹了,我们可以通过上面 1.1 小节的内容添加文件,也可以直接打开文件夹所在的目录,直接把我们要的文件复制过来,如下图:
新建文件夹复制文件过来,不做任何操作直接编译,可以参与编译:
但是要备其他文件夹中的文件包含,需要添加头文件路径,如果不添加,会出现如下错误:
1.2.1 手动添加路径
在工程资源管理器界面右击点击构建配置,如下图:
然后按照下图所示步骤,把我们自己新建的头文件所在文件夹路径添加(和 keil 工具一样的添加)即可:
1.2.1 使用SConscript
脚本添加
在自己想要放驱动的文件夹下面新建一个SConscript
脚本,如下图:
然后把自己想添加的.c.h 文件放在对应文件夹下面,新建完成刷新工程,然后再右击,选择“同步 scons 配置至项目”即可,脚本会把该文件路径自动添加至编译的路径中。
脚本内容如下:
二、I2C 驱动移植
上面我们讲解了如果添加自己的驱动文件,那么我们就直接把以前驱动文件移植过来,因为 SHT21 的驱动文件,在我以前博文 STM32L051 和 nRF52832 专栏都说明过 SHT21 的程序(等本次应用篇结束我会把最终的源码上传的):
nRF52832学习记录(十一、TWI总线的应用 SHT21程序移植)
我们这里只是把裸机使用的文件改成在 RT-Thread 上使用的文件。
2.1 移植步骤
首先我们把驱动文件都拷贝过来(Datadef.h 是我个人习惯使用的一些数据类型宏定义):
编译过后,还是正常通过除了上面报的那个延时函数的警告:
这是因为delay_us()
这个函数在 HAL 库中是没有的额,以前使用裸机的时候是我自己实现的,这里我们既然使用了 RT-Thread ,在我们《RT-Thread 记录(四、RT-Thread 时钟节拍和软件定时器)》中正好介绍到过 RT-Thread 中有 us 延时函数,我们拿过来用用试试看。
我们把所有 delay_us
改成rt_hw_us_delay
,记得包含一下头文件:
然后把文件驱动文件中中所有的HAL_Delay
改成rt_thread_mdelay
,比如:
感觉已经好了……那么接下来看看效果。
2.2 使用测试
上面我们驱动移植好了,我们直接来测试一下,这里我们还是新建一个线程把,这个温湿度读取的线程我根据经验,设置为 192 字节,然后每隔 3s 读取一次温湿度打印:
程序编译正常没有问题,但是跑起来出问题了....
2.3 好多的问题= =!
按键没有修改,上篇文章一样预留的
虽然占用了 7000 多字节,但是应该还是有空间的,但是在程序运行直接报错:
key 线程昨天测试的时候完全没问题啊,这里居然出错了,我考虑到今天也就多加了一个 sht 的线程,于是我将温湿度读取线程中的逻辑去掉,又出另外的问题提示:
然后想着是不是刚启动的时候等一会读传感器,没有为什么,讲不出道理= =!于是:
在这个时候我忽然想到浮点数的特殊性,以前记得使用 RT-Thread 遇到过,然后网上查了下确实,rt_kprintf
是不能打印浮点数的,是不是浮点数的问题。
因为上面显示 sht21 线程栈溢出,说明线程栈给的不够,其实按照经验来说,是足够的,这里不管,那我给他增加线程栈看看结果,如下图:
好吧!还是有问题,没救了,先把浮点数的问题处理了吧。
改了一个不打印浮点数的,结果看图吧:
这么看来,或许应该是驱动修改的问题吧,我首先想到的就是 us 延时函数,那个延时函数有问题,于是我用了裸机中的干等函数……:
(后面证明不是这个原因,us 延时函数可用)
经验不管用,虽然在使用 FreeRTOS 的时候 192 字节有余:
但是在这里使用的时候,256 字节大小都出错了,直到我继续增大:
问题的解决
到头来,原来是这个驱动函数在 RT-Thread 线程中需要的线程栈大小相对 FreeRTOS 来说大,因为有些堆栈溢出的问题导致报错也不一样,而且根据以前的使用经验,所以开始并没有一味的想着增大线程栈空间。
后来考虑了一下,对于驱动操作来说,都是基于 STM32L051 芯片,这部分应该差距不大,还有一个最能的原因是打印函数,我们以前就说过打印函数很占用内存空间,rt_kprintf 是 RT-Thread 自己实现的,是不是因为这个打印函数占用空间大导致的呢? 这个目前只是猜想,后期有机会来研究一下!
回过头切换回 RT-Thread 中的 us 延时函数,也测试成功了。
2.4 RT-Thread nano 打印浮点数
1、虽然软件包中心,有一个名为rt_vsnprintf_full
的软件包,可以打印浮点数,但是我们使用的 nano 版本并不能安装软件包……
RT-Thread_rt_kprintf()打印浮点数(解决方法2:添加rt_vsnprintf_full)
2、同样 nano 版本无法使用标准 libc 库……,所以无法使用vsnprintf
替换rt_vsnprintf
的方式:
RT-thread rt_kprintf()函数格式化输出浮点数
3、如果做一些其他修改,使其可以支持标准 C 库调用,会额外占用 RAM 空间:
rt-thread printf打印信息(包含打印浮点型float)
综合来说看了一圈,好吧,认输了! 打印不出来,不玩了,本次测试不太顺利……
最后还是用土方法,简单应付一下,因为后期并不需要打印,打印只是为了测试:
三、时刻关注占 RAM 大小
本文就添加了一个 I2C 驱动, 但是上面我们经过多次测试修改,温湿度测试线程也用到了 384 个字节,如果不打印出来,应该会小很多,我们后面肯定会来调整工程的。
那么还是老样子,今天测试完成以后和以前占用空间的对比图上一下:
加了温湿度读取线程以后,程序运行时候需要占用 RAM 的大小: 7248 字节,我们的芯片 RAM:8192 字节。
结语
本文我们了解了 RT-Thread Studio 添加自定义的.c .h 文件的方法,然后成功的把 I2C 驱动移植过来。
虽然一个简单的移植,缺被一个基本的问题困扰了许久,再次强调一次,在线程中使用打印函数,是很占用 RAM 空间的行为,以前在使用操作系统的时候就知道这个问题。
本次测试,我推断在使用 RT-Thread 自己的rt_kprintf
函数会比 C 库 printf
占用更多内存空间(有错误请指出),也算是一个小收获。
然后就是使用 RT-Thread nano 打印浮点数,虽然我最终还是样子上打印出了浮点数= =!我没有研究出比较满意的方式,也算是一个小遗憾,不过话说话来,打印大多数是为了测试,正常的项目跑起来,也不需要打印这些。
没想到本次测试这么折腾,有点累= =! 还望小伙伴多多支持,多多指教!
好了,本文就到这,谢谢大家!
版权声明: 本文为 InfoQ 作者【矜辰所致】的原创文章。
原文链接:【http://xie.infoq.cn/article/87bbe7d4e46e34e7927e6a0e4】。文章转载请联系作者。
评论