应用篇-在STM32L051上使用RT-Thread 第三篇,学习RT-Thread Studio如何添加自定义的.c .h文件,
同时开始我们的程序移植和设计。
经过前面两篇文章的准备,我们终于可以开始写应用代码了,在添加我们自己的驱动文件之前,本文会说明一下 RT-Thread Studio 如何添加自己的.c 和 .h文件。
然后在此基础上,完成 SHT21 温湿度传感器的 I2C 驱动移植。
在STM32L051C8 上使用 RT-Thread 应用篇系列博文连接:
RT-Thread 应用篇 — 在STM32L051上使用 RT-Thread (一、无线温湿度传感器 之 新建项目)
RT-Thread 应用篇 — 在STM32L051上使用 RT-Thread (二、无线温湿度传感器 之 CubeMX配置)
使用RT-Thread Studio添加自定义的.c .h文件有多种方式,本文根据博主的实际测试,分别说明一下。
直接在工程现有的文件夹下面右击,选择新建头文件或者源文件,如下图:
添加完成以后就可以直接编译了,上面我在 cubemx 文件夹下的Src 和 Inc 下分别添加了i2c.c
和两个头文件,直接编译会关联进去的:
但是这里得说明下,cubemx 是通过SConscript
受SCons 构建工具管理的,所以直接这样添加的话,每次我们自己增加一个.c文件,需要在 SConscript
文件中增加对应部分,要不然同步一下scons,不相关的.c文件就会被排除构建:
当然我们也可以在其他文件夹下添加,比如 drivers 文件夹,里面.c.h文件是混在一起,我们也可以直接在这里面新建,如下图:
总结一下,在现有文件夹下面添加是比较方便的一种方式,因为项目整体的构建系统已经处理好了这些文件下的.c.h文件的包含关系,我们不用再进行添加头文件路径这种操作。
但是这种方式会影响项目原始框架,小项目的文件少倒是无所谓,文件多了的话就很乱了。
所以我们要讲一下如果新建文件夹放置我们自己的驱动程序应该怎么做。
首先,还是右击选择新建文件夹,然后出选择文件夹放置的位置,输入文件夹名字:
新建好以后,我们在左边资源管理器可以看到有文件夹了,我们可以通过上面 1.1小节的内容添加文件,也可以直接打开文件夹所在的目录,直接把我们要的文件复制过来,如下图:
新建文件夹复制文件过来,不做任何操作直接编译,可以参与编译:
但是要备其他文件夹中的文件包含,需要添加头文件路径,如果不添加,会出现如下错误:
然后按照下图所示步骤,把我们自己新建的头文件所在文件夹路径添加(和keil工具一样的添加)即可:
SConscript
脚本添加在自己想要放驱动的文件夹下面新建一个SConscript
脚本,如下图:
然后把自己想添加的.c.h文件放在对应文件夹下面,新建完成刷新工程,然后再右击,选择“同步scons配置至项目”即可,脚本会把该文件路径自动添加至编译的路径中。
脚本内容如下:
from building import *
cwd = GetCurrentDir()
src = Glob('*.c') + Glob('*.S')
libpath = [cwd]
CPPPATH = [os.path.join(cwd)]
group = DefineGroup('mydrivers', src, depend = [''], CPPPATH = CPPPATH,LIBPATH = libpath)
Return('group')
上面我们讲解了如果添加自己的驱动文件,那么我们就直接把以前驱动文件移植过来,因为 SHT21 的驱动文件,在我以前博文 STM32L051 和 nRF52832 专栏都说明过 SHT21的程序(等本次应用篇结束我会把最终的源码上传的):
STM32L051测试 (三、I2C协议设备的添加测试)
nRF52832学习记录(十一、TWI总线的应用 SHT21程序移植)
我们这里只是把裸机使用的文件改成在 RT-Thread 上使用的文件。
首先我们把驱动文件都拷贝过来(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
,比如:
感觉已经好了……那么接下来看看效果。
上面我们驱动移植好了,我们直接来测试一下,这里我们还是新建一个线程把,这个温湿度读取的线程我根据经验,设置为192字节,然后每隔3s读取一次温湿度打印:
虽然占用了7000多字节,但是应该还是有空间的,但是在程序运行直接报错:
key线程昨天测试的时候完全没问题啊,这里居然出错了,我考虑到今天也就多加了一个sht的线程,于是我将温湿度读取线程中的逻辑去掉,又出另外的问题提示:
然后想着是不是刚启动的时候等一会读传感器,没有为什么,讲不出道理= =!于是:
在这个时候我忽然想到浮点数的特殊性,以前记得使用 RT-Thread 遇到过,然后网上查了下确实,rt_kprintf
是不能打印浮点数的,是不是浮点数的问题。
因为上面显示 sht21 线程栈溢出,说明线程栈给的不够,其实按照经验来说,是足够的,这里不管,那我给他增加线程栈看看结果,如下图:
好吧!还是有问题,没救了,先把浮点数的问题处理了吧。
这么看来,或许应该是驱动修改的问题吧,我首先想到的就是 us延时函数,那个延时函数有问题,于是我用了裸机中的干等函数……:
经验不管用,虽然在使用 FreeRTOS 的时候 192字节有余:
但是在这里使用的时候,256字节大小都出错了,直到我继续增大:
到头来,原来是这个驱动函数在 RT-Thread 线程中需要的线程栈大小相对 FreeRTOS 来说大,因为有些堆栈溢出的问题导致报错也不一样,而且根据以前的使用经验,所以开始并没有一味的想着增大线程栈空间。
后来考虑了一下,对于驱动操作来说,都是基于STM32L051芯片,这部分应该差距不大,还有一个最能的原因是打印函数,我们以前就说过打印函数很占用内存空间,rt_kprintf 是 RT-Thread 自己实现的,是不是因为这个打印函数占用空间大导致的呢? 这个目前只是猜想,后期有机会来研究一下!
回过头切换回 RT-Thread 中的 us 延时函数,也测试成功了。
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)
综合来说看了一圈,好吧,认输了! 打印不出来,不玩了,本次测试不太顺利……
最后还是用土方法,简单应付一下,因为后期并不需要打印,打印只是为了测试:
本文就添加了一个 I2C驱动, 但是上面我们经过多次测试修改,温湿度测试线程也用到了384个字节,如果不打印出来,应该会小很多,我们后面肯定会来调整工程的。
那么还是老样子,今天测试完成以后和以前占用空间的对比图上一下:
加了温湿度读取线程以后,程序运行时候需要占用 RAM的大小: 7248 字节,我们的芯片 RAM:8192字节。
本文我们了解了 RT-Thread Studio添加自定义的.c .h文件的方法,然后成功的把 I2C驱动移植过来。
虽然一个简单的移植,缺被一个基本的问题困扰了许久,再次强调一次,在线程中使用打印函数,是很占用RAM空间的行为,以前在使用操作系统的时候就知道这个问题。
本次测试,我推断在使用RT-Thread自己的rt_kprintf
函数会比 C库 printf
占用更多内存空间(有错误请指出),也算是一个小收获。
然后就是使用 RT-Thread nano打印浮点数,虽然我最终还是样子上打印出了浮点数= =!我没有研究出比较满意的方式,也算是一个小遗憾,不过话说话来,打印大多数是为了测试,正常的项目跑起来,也不需要打印这些。
没想到本次测试这么折腾,有点累= =! 还望小伙伴多多支持,多多指教!
好了,本文就到这,谢谢大家!