当前位置: 首页 > article >正文

Nordic-RT-Thread5.1.0移植笔记

关键词:RT-Thread、v17.1.0、softdevie、BLE、HID、mouse、Nordic、52840

资源获取

  1. nRF5x SDK v17.1.0:https://github.com/cbraissant/nRF5_SDK_17.1.0_ddde560

  2. RT-Thread V5.1.0:https://github.com/RT-Thread/rt-thread

  3. ZJ-SDK-RT-Thread-nRF52840:https://github.com/ZJ-TEK/ZJ-SDK-RT-Thread-NORDIC

env的使用需要学习,必须要配置为右键时显示“ConEmu Here”这个选项。

env第一次启动后,不要移动目录,会造成env相关指令执行失败,删掉env文件夹,重新解压启动即可。

移植准备

  1. 解压 nRF5x SDK v17.1.0 备用,作为工程的主体

  2. 解压 RT-Thread V5.1.0

  • 提取根目录rt-thread-5.1.0\下的RT-Thread相关文件夹备用:

13d9afe3b6f6bee8b77174cc3dcb15fb.png

  • 提取 rt-thread-5.1.0\bsp\nrf5x\libraries\drivers 文件夹备用,按需添加到工程

8536398fb556d83b47ca121671aedf5b.png

9e2ce3863c25ce07ebdf31c7ea397026.png

  • 提取 rt-thread-5.1.0\bsp\nrf5x\nrf52840\board 文件夹备用;

  • 提取 rt-thread-5.1.0\bsp\nrf5x\nrf52840\applications 文件夹备用;

 3. 解压 ZJ-SDK-RT-Thread-nRF52840

提取 ZJ-SDK-RT-Thread-NORDIC-master\ZJ_Application_NRF52840\018.ble_nus\NORDIC_SDK\components\libraries\timer 路径下的两个文件备用:

dae5f3aec71e6d5131fb848d3a9081a4.png

由于当前RT-Thread版本不允许定时器在回调函数里面再次启动定时器,也就是不允许蛇头咬蛇尾。

因此,app_button.c 直接调用 app_timer_rtthread.c 会产生bug,需要通过修改 app_button.c 解决定时器套娃问题。

Keil MDK 移植

RT-Thread源码与相关库添加至nRF5X SDK工程

 1. 打开 ble_app_hids_mouse_pca10056_s140.uvprojx 这个工程,在项目资源管理器内创建四个文件夹,分别命名为:

dfab0813a86c335b2121e73edc823065.png

 2. 添加RT-Thread源文件,并添加头文件路径;

ea6f8e28614f1056ab54ad77e61e78d7.png

 3. RTT_Kernel 文件夹内的文件需要右键添加配置define __RT_KERNEL_SOURCE__,否则相关文件编译会报错;

6f61c2d2553ab8fe8b988318d58dcc07.png

 4. RTT_components 文件夹内的文件需要右键添加配置define __RT_IPC_SOURCE__,否则相关文件编译会报错;

5f9150551111efe9953e3423f9eb8463.png

 5. nRF_Libraries 文件夹添加 app_timer_rtthread.c,移除或者排除文件夹下的 app_timer2.c 与 drv_rtc.c;

fd504caa2127f1b0e68b145caae2a02c.png

必须覆盖掉app_timer.h,否则编译报错。

 6.  ALT + F7 打开工程配置,移除 APP_TIMER_V2 与 APP_TIMER_V2_RTC1_ENABLED 两项配置,并添加 __RTTHREAD__ 与 RTTHREAD_ENABLED ;

修改前:

2c76909153848d4e0697bbe25417411f.png

修改后:

000d79869bf80862d282c015c57391f2.png

 7. 需在 rtconfig.h 内添加宏定义 #define RT_TIMER_TICK_PER_SECOND RT_TICK_PER_SECOND,否则 app_timer_rtthread.c 将会出现报错;

 8. 在项目管理器创建 Drivers 文件夹,并添加源文件与头文件;

6a1d0b05be23751452c040a31ab0afa6.png

此处文件来自:

rt-thread-5.1.0\bsp\nrf5x\nrf52840\board

rt-thread-5.1.0\bsp\nrf5x\libraries\drivers

 9. Application 文件夹添加 application.c,移除或者排除文件夹下的 main.c ;

544d3c957fbd10dbd80ba6e091fb53b1.png

ble_app_hids_mouse.c 为修改后的main.c,在文件夹内复制修改后,添加到Application文件夹,需要进行如下的修改:

b53f80f7201d885875f07a11aac5cb74.png

将 mian.c 命名为 ble_app_hids_mouse.c,并将 mian() 修改为下面这段内容:

d766413ee0cbb8af45df79233a144a15.png

48500a3aa57f5587d65d2e4e685aadf7.png

 10. 头文件加载路径汇总:

ee14ba37f3a2a311bfc75f12a4bd231b.png

编译无报错,可尝试编译烧录,此时,RT-Thread是可以正常启动的,如果有打印输出,说明可以进入下一步了。

调试报错处理

 1. 不开启BLE,没有报错,LED闪;开启BLE之后,出现RAM报错。

24cc73efcc7d78cb4fd6976fa75322eb.png

打印 NRF_ERROR_NO_MEM,是内存相关问题,结合上下文,说明可能内存配小了,需要往后挪挪位置。

RAM修改:START 0x20002260 变更为 START 0x20002270,Size 倒是可以不用进行修改。

修改前:

ae6b929219e98cc3f5fd92511fb7a70e.png

修改后:

fbe4126b2fea28a288961ac4608c2a6a.png

 2. 启动BLE后,闪灯闪烁一段时间之后 或者 PC端进行连接, 然后LED灯就不闪了。

目测是某些机制没启动,导致跑死了,看起来像是EVENT管理,比如 NRF_SDH_DISPATCH_MODEL 之类的没有执行?

为什么会跑死?

  • NRF_SDH_DISPATCH_MODEL 没有配置好;

极大可能是这个问题,但是,由于是三选一,也不确定是哪个可行。

ab6804f48475e5dd062c7921342e75ab.png

  • 进入了休眠模式?睡死了?

蓝牙停止广播,理论上不应该让RTOS也睡死。

屏蔽掉进入休眠模式的代码:现象未解除,所以不是睡眠的问题;

通过观察时长,闪烁的时间是30秒左右,而且是打印 Fast advertising. 之后,就没有打印Slow advertising. 。

a1455b68190144dd436ddb21fbb223ea.png

根据代码推测,应该是服务切换失败了,而服务在Softdevice里面应该是Event相关的内容出了问题,而与Event相关的是 NRF_SDH_DISPATCH_MODEL 这个值,通过百度搜索可以获取到的可以参考的文章是FreeRTOS的相关移植文章:

52832带softdevice工程移植freertos

https://www.eemaker.com/52832-softdevice-freertos.html

“修改sdk_config.h文件中:NRF_SDH_DISPATCH_MODEL 2

该配置的意思是修改softdevice底层事件到应用层的方式,模式2代表是application主动获取。

在freertos的主动获取的实现就是在我们前面添加的nrf_sdh_freertos.c文件中如果用mode0 中断方式通知到应用层,就不需要添加nrf_sdh_freertos.c文件,但是我测试的时候发现会出现蓝牙断开的情况)”

为什么,FreeRTOS可以使用NRF_SDH_DISPATCH_MODEL

因为官方提供的移植API提供了相关的代码,使用的就是轮询方式。

创建一个Task,然后,在Task里面while()死循环获取事件,相关代码如下:

7373c45d9ec32f5a77f50350ba1a2af6.png

8f5d0bede85bd29f46f22d741ef4df5a.png

咨询群友,说是RTOS就应该配置为 2,但是,还是觉得不对劲,总觉得这个事件获取应该是和代码有关,而不是和系统的有无有关。

偶然之下,将 NRF_SDH_DISPATCH_MODEL 的值从 2 设置为 0,解决这个卡死问题,实锤了与系统的有无毫无关联:

0 是中断方式:RT-Thread

1 是app_scheduler:nRF5X SDK Demo

2 是轮询方式:FreeRTOS

中断模式 和 轮询模式 ,这两个哪个更好?不知道了。

Visual Studio + VisualGDB 导入MDK工程

最好是先KEIL MDK搭起来能用的工程,然后再导入到VisualGDB里面,是代码先跑,还是你人先跑,那就不好说了。

创建VisualGDB工程(ARMCC)

 1. 偷懒了,不想一个文件一个文件的添加,直接将Keil MDK的项目直接导入到VisualGDB里;

  • 单击【创建新项目】

fa95833c357e2f177952292e9e3d03b3.png

  • 选择【嵌入式工程向导】

a44fef9b532b223b3b18462105c4bd30.png

  • 填写【工程名】【方案名】【方案创建路径】

b0871a21782d44bf93ad22210455afec.png

方案(visualgdb) -> 工程(ble_app_hids_mouse_pca10056_s140)

  • 选择【工程类型】【编译器类型】【工程路径】

cee515a3847879658cb6b1af385c33eb.png

  • 选择【MCU】,但是这个界面已配置好,点选【Next】即可

b8b3feeea20346aa60e32ca3dba946a0.png

  • 配置 DEBUG的方式,这里选择【J-Llink】、【USB】、【SWD】、【Before programming】、【After programming】,点选【Next】进入下一页

898b5ee386585cb9be326327260cd261.png

  • 路径映射界面,看不懂,所以直接点击【Finish】结束配置;

e7e195c8476d7aed334aa49cddbf9519.png

完善工程配置(丢失的配置需要补全)

1. 打开左侧的【解决方案资源管理器】,找到筛选器【::Device】,添加图里面的这两个文件到筛选器;

Keil工程也有添加这两个文件,但是,导入工程时丢失了。

a2e8d2351421ec6849b7aa3e8a45b927.png

 2. 执行上述操作后,此时应该还有部分文件没有导入,如果右上角提示有文件没有include,直接点击确认包含即可;

 3. 找到 main.c ,app_timer2.c, drv_rt.c ,nrf_sdh_ble_rtt.c ,ble_conn_params_rtt.c 这几个文件:右键 - 属性- Keil Settings - Excluded From Build -> 【是】

 4.【右键】点击【ble_app_hids_mouse_pca10056_s140】弹出选单,选择【属性】进入工程配置界面:

4a663d1ca2de9c7560473e0d5ec62b11.png

 5. ble_app_hids_mouse_pca10056_s140 工程配置:

  • 配置应用程序二进制接口:Keil Settings -> Floating-point ABI : Hardware FP (-mfloat-abi=hard)

  • 配置浮点单元类型:Keil Settings -> Floating-point unit type : fpv4-sp-d16 (-mfpu=fpv4-sp-d16)

  • 配置ARM CPU类型:Keil Settings -> ARM CPU type : -mcpu=arm7m

  • 配置ARMCC CPU类型:Keil Settings -> CPU Type for ARMCC/ARMASM : Cortex-M4.fp.sp (--cpu=Cortex-M4.fp.sp)

  • 添加预编译器定义:C/C++ -> Preprocessor -> Preprocessor Definitions :

    __RTTHREAD__;RTTHREAD_ENABLED;BOARD_PCA10056;BOARD_PCA10059;NRF52840_XXAA;CONFIG_GPIO_AS_PINRESET;

    FLOAT_ABI_HARD;NRF_SD_BLE_API_VERSION=7;S140;SOFTDEVICE_PRESENT;__HEAP_SIZE=8192;__STACK_SIZE=8192;

    Keil MDK是用的”,“与” “进行宏的分隔,而VisualGDB只能用”;“去分隔,进入到编辑界面之后,回车换行即可,退出编辑之后,回车会自动转换为”;“

  • 配置C/C++语言标准:C/C++ -> Advanced -> Language Standard for C++ files : C99 (--c99)

  • 配置GNU拓展:C/C++ -> Advanced -> Enable GNU Language Extensions : 否

  • 配置ROM与RAM映射:Linker -> Memory Layout -> Scatter Files : ..\examples\ble_peripheral\ble_app_hids_mouse\pca10056\s140\visualgdb\ble_app_hids_mouse_pca10056_s140\nrf52840_xxaa.sct

    如果一开始就是直接作为GCC直接导入,可以从RT-Thread获取模板:

    rt-thread-5.1.0\bsp\nrf5x\libraries\templates\nrfx\board\linker_scripts

    link.sct简单修改:

d70ac4b11c34e7558a1078a653568cd4.png

  • link.lds简单修改:

663a11bface2ab6d0a60f9d1789b4b5c.png

  • 目前使用的是ARMCC,使用GCC时,内存相关的打印还要根据编译器进行启动地址的打印 :

565f789a932d49e89b8ce9d23f1237de.png

  • 时候,一般就编译成功了,有一堆警告,但是,没关系的。

    注意,导入成功了,不要随便切回Keil MDK进行编辑,否则再次启动VisualGDB触发改动检测的。


http://www.kler.cn/a/350682.html

相关文章:

  • 第十一章 图论
  • 【ESP32】ESP-IDF开发 | WiFi开发 | AP模式 + 基站连接例程
  • 数组常见解决方案
  • SimpleHelp远程管理软件存在任意文件读取漏洞(CVE-2024-57727)
  • 金融项目实战 06|Python实现接口自动化——日志、实名认证和开户接口
  • 【Linux系列】查看服务器是否使用了 SSD 的多种方法
  • 好玩的css按钮
  • 关于 GLFW、SDL2、Win32、GLUT 和 macOS(OSX)平台的介绍
  • UDP/TCP协议详解
  • Git 分支操作-开发规范
  • 条款2 理解auto类型的推导
  • 如何通过 Service Mesh 构建高效、安全的微服务系统
  • FFmpeg的简单使用【Windows】--- 视频倒叙播放
  • 一文详解线程池
  • R语言统计分析——折线图
  • 【uniapp】实现触底加载数据
  • Linux实验三
  • MySQL 数据库--查询
  • 《数据之海》
  • LeetCode 第419场周赛个人题解
  • Spring Boot在医疗病历B2B交互中的技术突破
  • 设计模式和软件框架的关系
  • 【优先算法】--双指针1
  • 【二刷hot-100】day1
  • 家政行业小程序需要哪些功能?
  • STM32—旋转编码器控制直流电机(标准库)