- 1. 初识ndn-lite
- 2. 前期准备:开发环境的搭建
- 3. 实战体验:一个使用ndn-lite的应用示例
- 4. nRF52上的BLE Mesh
- 5. ndn-lite学习与使用
NDN-Lite库实现了命名数据网络(NDN)栈。该库是用标准C编写的,需要最低版本的C11(ISO/IEC 9899:2011)。ndn-lite仓库地址:https://github.com/named-data-iot/ndn-lite
到目前为止,ndn-lite已经为POSIX平台(Linux,MacOS,Raspberry Pi),RIOT OS和Nordic NRF52840开发套件开发了基于ndn-lite的物联网软件包(准备好平台)。开发人员可以直接开发基于这些软件包的物联网应用程序,而无需担心适应性问题。
当前基于ndn-lite的包有如下几个(具体可以点击进行查看),未来可能添加更多的内容:
- NDN-Lite Unit Tests over RIOT OS
- NDN IoT Package for Nordic SDK using Segger IDE and Android Phone
- NDN IoT Package for Nordic SDK using GCC
- NDN IoT Package for POSIX using CMake
- NDN-Lite Doxygen Documentation
当前我们主要关注的就是第二个:NDN IoT Package for Nordic SDK using Segger IDE and Android Phone。
这部分内容主要是讲解一些有关开发环境的搭建,包括:JDK/JRE的安装、Android Studio的安装配置、Segger的安装配置等内容。开发环境的搭建都是**基于Windows10操作系统(64位)**上进行的。
这篇操作参考指南的目的就是让自己从零开始,按照操作指南一步一步的走下去以实现相同的效果。如果之前做过Java开发,想必JDK/JRE都已经在系统上安装配置好了,这一步骤就可以省略了。如果之前用Android Studio写过安卓App,那么JDK/JRE安装和Android Studio安装配置都可以跳过不用看了。
JDK版本为8u211,可以直接打开这个链接下载对应版本的JDK(如下图):https://www.oracle.com/technetwork/java/javase/downloads/jdk8-downloads-2133151.html
也可以通过百度云下载我已经上传的安装包:https://pan.baidu.com/s/14aMwA8NK3GDm854GIMPYOQ ,提取码:yt6t。
下载后直接双击选择安装目录即可。
- 新建环境变量,变量名称为
JAVA_HOME
,变量值就是你安装JDK的目录地址。如下图所示。
- 在Path中把jdk和jre中的/bin添加到环境变量中。双加打开Path然后点击新建,然后依次输入:
%JAVA_HOME%\bin
和%JAVA_HOME%\jre\bin
(这里的路径名称要和你安装路径下的文件名称一致)。如下图所示。
- 检测安装是否正确。打开终端,输入
javac
和java -version
看是否输出对应的提示内容和jdk版本信息,如果如下图所示了,说明jdk环境配置成功。如果不成功,可能最大的原因就是系统环境的路径没有设置正确;如果路径确认无误了还是不成功,建议重启或者注销下系统。
JDK环境配置好了之后,就可以进行Android Studio(后面简称AS)的安装了。因为后面要涉及到弄一个相对应的安卓手机的App,而AS又是Google官方推荐的IDE。
可以点击链接进行下载,https://developer.android.google.cn/studio/ 或者 http://www.android-studio.org/ (前面的链接因为是google的链接,不可访问可以点击后面的链接下载)。也可以通过百度云下载我上传的AS安装包:https://pan.baidu.com/s/1NDqBK523lE-V7I0uzbJ8qA 提取码:0142。下载完成后,直接双击安装包进行安装即可。至于安装过程如果不用修改目录的话直接Next即可。
第一次启动AS会出现一些设置项。
第一次启动AS后,如果弹出下面界面,这是让你导入配置文件,这是让之前做过安卓开发人员使用的,可以直接导入之前的配置。而这里是第一次使用,所以没有可用的配置文件可以导入,所以选择Do not import settings
,然后点击OK。
接下来如果弹出如下界面,是AS没有查选到可用的SDK目录,让我们设置代理,以便可以进行下载SDK等相关的文件。这里可以暂时不进行设置,直接选择Cancel。
接下来就会出现一个Welcome界面,直接点击Next。
然后就会出现一个安装类型设置,也就是选择自定义还是按照标准模式来设置和配置AS。这里我选择自定义Custom
,因为可以更好地去了解和学习相关的配置。然后点击Next。
如果选择了标准模式,也没有关系,这些配置都是可以在AS里面进行修改的。
如果选择自定义模式,下面首先就是要进行UI主题设置,这里我习惯了暗黑主题,所以选择了Darcula
(根据自己的习惯选择)。
接下来这张图片就是SDK的安装。可以修改一下SDK安装的位置,如果后面要安装手机模拟器的镜像这对空间占得还是比较大的。然后其他的默认选择,Android Virtual Device
就是模拟器镜像,可以选择也可以不选择,后面可以再去安装,我这里先不选择。然后Next。
接下来就是模拟器的设置,默认推荐就可以。
最后就直到AS完成这些操作。
在开发之前,必须安装一些所需的软件。这些软件包括:连接到开发板的工具(J-Link等)、用于开发应用程序的IDE(Segger Embedded Studio等),以及提供库和示例应用程序的nRF5 SDK。
Segger Embedded Studio(简称:SES)是Nordic公司推荐的IDE,全平台(Windows、Linux和MacOS)都支持。因为后面要涉及到nRF52840这块板子进行开发,所以SES安装也就必不可少。如果之前有用户习惯用Keil去开发嵌入式相关的内容,nRF52840当然也支持用Keil去开发,具体可以去参考这篇文件:nRF5 Series: Developing on Windows with ARM Keil MDK, 提取码:rq3p。
点击这个链接:https://www.segger.com/downloads/embedded-studio ,也可以直接通过百度云去下载我上传的资源:https://pan.baidu.com/s/16KimethzWKN2xYyu1IbnSw ,提取码:rmb9(官网上下载速度比较慢,建议通过百度云去下载)。
选择对应的nRF5_SDK版本进行下载下载链接为:https://developer.nordicsemi.com/nRF5_SDK/nRF5_SDK_v15.x.x/ ,也可以直接通过百度云去下载我上传的资源,
内容 | 地址 | 提取码 |
---|---|---|
15.2 SDK |
https://pan.baidu.com/s/15TbJwBNJS9M528NGeitDjQ | 49fp |
15.2 SDK offline文档 |
https://pan.baidu.com/s/1070Mta5dQWbsTMF6T4P7ig | gr5l |
15.3 SDK |
https://pan.baidu.com/s/1r9FYNFryrPZINsgzSWSHeg | 193o |
15.3 SDK offline文档 |
https://pan.baidu.com/s/1hiFg50Ml0urB6gPVpMjWaQ | c7kz |
官网下载比较慢,建议用百度云下载,这里建议把15.2
和15.3
两个版本都下载下来,以备后用。
nRF5命令行工具用于Nordic Semiconductor的nRF51和nRF52系列SoC的开发、编程和调试。
nRF5命令行工具包括以下组件:
- nrfjprog可执行文件:用于通过Segger J-LINK编程器和调试器进行编程的工具。
- mergehex可执行文件:可以将最多三个.HEX文件合并为一个文件。
- nrfjprog DLL:一个DLL,导出用于编程和控制nRF51和nRF52系列设备的函数,并允许开发人员使用DLL API创建自己的开发工具。
- SEGGER J-Link软件和文档包(仅包含在Windows安装程序中)。
下载链接为:https://www.nordicsemi.com/Software-and-Tools/Development-Tools/nRF5-Command-Line-Tools/Download#infotabs ,也可以通过百度云下载:https://pan.baidu.com/s/1l5dyAdRC3luBxSsU-zyUsA ,提取码:t93p。
安装期间会弹出如下对话框,这是因为SES安装的过程也安装了J-Link,所以会提示,直接点击Ok。
期间因为需要安装Microsoft Visual C++版本需要重启电脑,建议把一些你需要保存的东西先保存了,然后点击重启。
如果没有更换目录的要求,一直Next到成功安装即可。
通过上面的一些安装操作,接下来就把板子通过数据线与电脑USB口相连,看电脑是否提示J-Link连接了。
这里列出一点nRF52840板子的一些内容,毕竟要在这上面写代码,知道一点这块板子的硬件结构还是有点好处的,当然这部分是可以跳过不用看的。
nRF52840支持Bluetooth 5/Bluetooth mesh/Thread/Zigbee/802.15.4/ANT/2.4G,拥有一颗64MHz的Cortex M4F架构的CPU,搭载1MB Flash + 256Kb RAM。
nRF52840板子全貌如下图所示,板子左半部分为一个正版的J-Link OB,引出了烧录接口,可给其它设备烧录程序。板子右半部分就是nRF52840了。
这是一个应用示例,展示了使用ndn-lite在Android手机和nRF52840开发板之间进行ndn通信、安全登录和信任策略切换的基本功能。
具体来说,这个应用程序由两部分组成:Android手机中的用户应用程序和nRF52840开发板中的ndn-lite应用程序。用户应用程序是一个通用的Android应用程序,它在可用设备、基本设备信息和turst策略选项等方面提供用户界面。ndn-lite应用程序使用ndn-lite来提供基于ndn的通信、安全登录和信任策略切换功能等。
目前,该应用程序使用BLE作为面在Android手机和开发板之间传输数据包。下面开始这两方面的工作。
具体参考这个NDN-IoT-Android库。因为要的用到蓝牙相关的内容,模拟器是无法使用蓝牙功能的,所以建议使用真机来进行测试。
要求:
- 安卓手机(>= 6.0)
- 支持蓝牙5.0以上
下载NDN-IoT-Android到自己的电脑中,下载链接为:https://github.com/gujianxiao/NDN-IoT-Android.git 。可以通过百度云进行下载:https://pan.baidu.com/s/1Kx9c-xPQ5TTQccOz4DFzTQ ,提取码: fgsj。
打开git命令行(如果系统中没有安装git工具,可以去Git官网上去下载安装一下,下载地址为:https://git-scm.com/download/win ,也可以直接通过百度云链接下载:https://pan.baidu.com/s/1hkbYz7sJpxxTNbeEXPqOlQ ,提取码: vv7g),输入git cloen https://github.com/gujianxiao/NDN-IoT-Android.git
,
这里需要等待一下,因为AS会通过Gradle来构建这个Project。
在Gradle构建工程之后,如果出现AS提示下面这幅图片的的错误:
这个错误是提示当前我还没有下载对应版本的SDK,这里点击最下面的Install missing SDK package
进行安装即可,接下来就会出现SDK安装的界面,如下图所示,选择Accpet
之后,直接Next即可。
期间,如果遇到了AS提示下面这图片所示的,是要我们更新一下Gradle插件,直接Update更新就可以了。
经过上面的操作,应该是没有再报什么错误了,接下来就是编译运行这个App。
前期准备
- 上网查询一下如何让手机打开开发者模式,每个牌子的手机都是不一样的,这里以小米手机为例,
打开设置 -> 我的设备 -> 全部参数 -> 连续点击数次MIUI版本那栏
就可以打开开发者模式。 - 通过USB连接电脑,并将USB的用途选择
传输文件
。
- 在手机中找到开发者选项,打开USB调试。具体打开位置以小米手机为例:
打开设置 -> 更多设置 -> 开发者选项 -> USB调试
然后点击打开(如左图所示),之后就会出现一个密钥确认的界面(如右图所示),点击确定,电脑右下角状态栏就会有看到有手机连接了。
进过上面的前期准备,手机应该连接上了电脑了,下面开始正式通过AS编译安装App啦!
点击所下图所示的 Run app “斜三角”图标(图片所示的“锤子”图标是编译工程,但不会直接对手机安装app,点击“斜三角”图标就可以编译完了直接安装应用到手机上了),就会弹出一个选择安装app的手机列表,然后选择OK即可。
接下来AS就会对NDN-IoT-Android这个工程进行编译。编译完成之后,手机会弹出一个对话框让你进行选择是否安装这个App,直接点击继续安装进行安装。
手机自动安装完这个app之后默认会直接打开这个app,给与相应的权限之后,发现App会直接闪退。那是因为手机蓝牙没有开启。开启手机蓝牙之后,App运行的界面如下图
NDN-IoT-Android 这个App到这里也就安装完成了,从这个源代码可以看利用了ndn-lite这个库,如下图所示,在后面肯定会深入的学习得到,暂时知道这里已经使用了。接下来就是对nRF52840开发板进行安装操作了。
nRFProject工程链接地址为:链接: https://pan.baidu.com/s/1L9qydUhBlRB3ffUdknuFXw ,提取码: 3eqf。
前期准备: 进行这步操作之前,首先把ndn-lite和nRF52_SDK都下载下来。nRF52_SDK在上面的步骤已经提到过了,应该都下载下来了,直接解压到相应的目录下即可,这里使用的是nRF52_15.2版本。ndn-lite可以通过百度云盘下载:https://pan.baidu.com/s/1oyFMxZOIcBiuDoOedUSROQ ,提取码: 7ny9。
下面就正式开始修改SDK和ndn-lite路径了。
打开下载好的nRFProject,会看到里面有个ndn_lite_nRF52840_example.emProject
这个文件,然后用文本编辑器打开,我这里用nodepad++打开。
这里需要修改文件中指定的SDK和ndn-lite的路径,因为我把ndn-lite和nRF52_SDK都放在了nRFProject的上一级目录,所以我把原来文件像这这样的../../nRF5_SDK_15.2.0_9412b96/xx
和../../ndn-lite/xx
全都进行修改。
用notepad++可以很方便的进行修改,Ctrl + F就可以进行全局的替换。下面两幅图片就是notepad++修改sdk和ndn-lite路径的图片。
然后用SES打开这个nRFProject工程。Build编译这个工程,发现弹出密钥认证,如下图所示,因为这是第一次使用SES,之后就不弹出了,当然SES也是免费认证的,所以点击红色框框标出来的进行认证即可。
然后输入一些信息,邮箱地址请填写正确,之后SES会把密钥通过邮箱发送给你。
点击Request License之后,会收到一封邮件,邮件的内容就会包含密钥,然后把密钥复制填写到下图所示中。
点击SES的Build进行编译。下面说明我在编译的时候遇到的一些问题。
(1)问题描述:
SES输出的提示:
cannot find ../nRF5_SDK_15.2.0_9412b96/external/micro-ecc/nrf52hf_armgcc/armgcc/micro_ecc_lib_nrf52.a: No such file or directory
:也就是提示找不到micro_ecc_lib_nrf52.a这个文件。显示如下图所示。
(2)解决办法和步骤:
A. 在Windows下安装gcc,这里需要下载MinGW,下载地址为:https://mirrors.xtom.com.hk/osdn//mingw/68260/mingw-get-setup.exe 。百度云下载地址为:https://pan.baidu.com/s/1BrSoJ_-XgXHox3OK97S9Hw ,提取码: y4rq。
把有关gcc以及make相关的大部分都进行标记安装一下。注意要将MinGW加入环境变量,如下图所示。先创建一个MinGW的变量,然后在Path中进行添加。
这里在安装make的时候,发现MinGW中找不到对应的make包,所以,可以通过mingw-get install mingw32-make
来安装,如下图所示,这里也就是为什么要把WinGW添加到系统环境变量,如果不添加到系统环境变量之中,会提示该命令找不到。
B. 下载交叉编译工具,一开始系统是没有装这个交叉编译工具的,所以需要单独安装这个交叉编译工具。下载地址:https://developer.arm.com/open-source/gnu-toolchain/gnu-rm/downloads 。百度云下载的地址为:https://pan.baidu.com/s/1WDjKX8mX-sNRoZXSPtRJJw ,提取码: 4wxj。
选择Windows下的版本进行安装,建议安装到电脑的默认位置,最后完成的时候建议选择添加到环境变量。
C. 修改配置文件。找到nRF52840 SDK中的gcc配置文件,并进行修改。具体配置路径为:\nRF5_SDK_15.2.0_9412b96\components\toolchain\gcc
在该路径下,打开文件为:Makefile.windows
,修改内容如下图(具体的修改按照下载的交叉编译工具的具体版本和路径来修改)
D. 打开终端,找到nRFSDK的armgcc位置,具体位置如:xx\nRF5_SDK_15.2.0_9412b96\external\micro-ecc\nrf52hf_armgcc\armgcc
,然后mingw32-make
去make一下。
如果发现报错了,如下图,根据提示是\mirco-ecc
目录下缺少文件,把文件放到\micro-ecc
目录下,这个文件下载地址为:https://pan.baidu.com/s/1vj5EaRtg4X-qAsh4meuWdw ,提取码: wfcz。
下面这幅图片就是我添加文件之后的nRF52_SDK的文件夹:
当然如果没有报这个错误,那么就不用管了,应该就直接显示编译成功了,会生成SES提示缺失的那个micro_ecc_lib_nrf52.a
文件了。
E. 编译成功
对于演示效果,这里需要用到两块nRF52840板子。
先将nRFProject烧录第一块nRF52840板子中。
在SES软件中,先Build编译一下nRFProject(这一步骤在3.3.3节已经提过了)。编译完后,选择Target -> Connect J-Link
,如下图所示,这里是通过J-Link将板子与电脑相连。
然后把nRPject下载到板子中,选择Target -> Download xx
,如下图所示。
- 在SES中工程结构目录中,找到
hardcode-experimentation.h
文件,将#define BOARD_1
注释掉,然后将#define BOARD_2
取消注释。如下图所示。
然后Build -> PC连接第二块板子 -> J-Link Conncet -> 重新烧录到第二块板子,这些步骤都和上面提到过的烧录第一块板子一样的,不再赘述。
- 板子一上电,LED3就会闪烁3次,这表示的板子正在进行初始化相关的工作(观看效果请点击:https://pan.baidu.com/s/1jIuPspph3ZCYO6Yovnrb7Q ,提取码: a9ib,如果显示“正在转码。。。”,刷新一下网页就可以看了)。
- 第二次闪烁,表示的是设备在手机上已经完成了安全的sign-on(观看效果请点击:链接: https://pan.baidu.com/s/1BNagCaJn8NByoqaW_86GpA 提取码: 2t96)。
- 可以按按钮1去关闭LED1,或者去按按钮2去关闭LED1(观看效果请点击: https://pan.baidu.com/s/1eaRquGcrgWCtaNMREXk_qg ,提取码: 8bpv)。
- 如果有两块板子你可以按按钮3去发送命令兴趣包去打开另一块板子的LED1(观看效果请点击:https://pan.baidu.com/s/1AStjhpMK9p4QUcxqMXGQUQ ,提取码: 6n9p)。
因为后面会涉及到ble mesh与ndn-lite的结合,所以这里先看看在nRF52上运行的BLE Mesh相关的内容。如果没有用到可以省略不看。这里Nordic公司提供了nRF5 SDK for Mesh实现了BLE Mesh,下载地址为:https://www.nordicsemi.com/Software-and-Tools/Software/nRF5-SDK-for-Mesh/Download#infotabs 。
蓝牙mesh基于蓝牙4.0规范的低功耗部分,并与该协议共享最低层。在广播中,蓝牙mesh物理表示与现有的蓝牙低功耗设备兼容,因为mesh消息包含在蓝牙低功耗广告分组的有效载荷内。但是,蓝牙mesh指定了一个全新的Host层,虽然共享了一些概念,但蓝牙mesh与低功耗蓝牙的HOST层不兼容。具体内容请查看:https://infocenter.nordicsemi.com/index.jsp?topic=%2Fcom.nordic.infocenter.meshsdk.v3.1.0%2Fmd_doc_introduction_basic_concepts.html
- 应用领域
蓝牙mesh主要针对简单的控制和监控应用,如光控或传感器数据采集。数据包格式针对小型控制数据包进行了优化,发出单个命令或报告,不适用于数据流或其他高带宽应用程序。
使用蓝牙mesh会导致比传统蓝牙低功耗应用更高的功耗。这主要是因为需要保持接收不断运行。蓝牙mesh网络最多支持32767个设备,最大网络直径为126跳。
- 网络拓扑和中继
蓝牙网状网是一种基于广播的网络协议,其中网络中的每个设备都向无线电范围内的所有设备发送和接收所有消息。网状网络中没有连接的概念。网络中的任何设备可以中继来自任何其他设备的消息,这使得网状设备可以通过让一个或多个其他设备将消息中继到目的地来向无线电范围之外的设备发送消息。此属性还允许设备随时移动和进出网络。
- 中继
蓝牙mesh网通过中继消息扩展网络范围。任何mesh设备都可以配置为充当中继,并且不需要专用的中继设备来构建网络。中继一次则会减少生存时间(TTL)值,如果TTL为2或更高,则转发它们。这种无向中继被称为消息泛滥,并确保消息传递的高概率,而不需要任何关于网络拓扑的信息。网状配置文件规范不提供任何路由机制,所有消息都由所有中继转发,直到TTL值达到零。为了避免消息被反复转发,所有网状设备都维护消息缓存。此缓存用于过滤设备已处理的数据包。
基于洪泛的消息中继方法可能导致大量冗余流量,这可能会影响网络的吞吐量和可靠性。因此,强烈建议限制网络中的中继数量。
- GATT协议
为了让不支持接收mesh数据包的BLE支持mesh数据包,蓝牙网状网定义了一个单独的协议,用于通过BLE GATT协议隧道化mesh消息。为此,网状配置文件规范定义了GATT承载和相应的GATT代理协议。该协议允许传统BLE设备通过建立与启用代理功能的网状设备的GATT连接来参与mesh网络。
- 地址
将设备添加到网络时,会为其分配一系列代表它的单播地址。设备的单播地址无法更改,并且始终是顺序的。单播地址空间支持在单个网状网络中具有32767个单播地址。任何应用程序都可以使用单播地址直接向设备发送消息。
组地址作为网络配置过程的一部分进行分配和分配。组地址可以表示任意数量的设备,并且设备可以是任意数量的组的一部分。网状网络中最多可以有16127个通用组地址。
虚拟地址可以被认为是组地址的特殊形式,并且可以用于表示任意数量的设备。每个虚拟地址都是从文本标签生成的128位UUID。虚拟地址不必由网络配置设备跟踪,并且以这种方式,用户可以在部署之前生成虚拟地址,或者可以在网络中的设备之间临时生成地址。
- 安全
Bluetooth Mesh采用多种安全措施来防止第三方干扰和监控:
- 授权认证
- 消息加密
- 私有密钥
- Replay保护
下面这幅图片就是nRF52 for mesh的体系结构图:
本示例 light switch example 演示了mesh网络生态系统的主要部分。 它由三个部分组成:
- 灯开关服务器:实现Generic OnOff服务器模型,用于接收状态数据并控制板上LED 1的状态。
- 灯开关客户端:实现Generic OnOff客户端模型,当用户按下任何按钮时,OnOff Set消息将发送到配置的目标地址。
- Mesh Provisioner:一个简单的配置器实现,用于设置网络。该配置器能配置mesh网络中的所有节点。 此外,配置器还能在这些节点上配置模型实例的密钥绑定、发布和订阅设置,以使它们能够相互通信。
下图给出了将在此示例中设置的mesh网络的整体视图。 括号中的数字表示配置器分配给这些节点的地址。
需要至少2块nRF52840板子:
- 1块对应与客户端
- 1块或多块用于服务端(最多30块)
此外,您还需要以下其中一项:
- 如果使用静态配置程序示例,则还需要1块nRF52开发板。
- 如果使用App进行配置,则需要下载nRF Mesh App(iOS或Android)。Android版本下载地址为:https://github.com/NordicSemiconductor/Android-nRF-Mesh-Library/releases 。iOS源码下载地址为:https://github.com/NordicSemiconductor/IOS-nRF-Mesh-Library/releases 。
- nRF5 SDK for Mesh
- nrfjprog(安装过nRFx Command Line Tools for Windows就会包括)
这里选用App作为Mesh Provisioner,这里的App也就是nRF Mesh;选择一块nRF52板子作为客户端,也就是能够通过客户端板子上的按钮去操作自己或者服务端上LED1灯的开关;选择一块nRF52板子作为服务端,用于显示板子上LED1灯的ON/OFF状态。
如果不用App作为Mesh Provisioner,那么就还需要一块nRF52板子,然后把Provisioner程序下载进去来当做Provisioner,这里下载安装Provisioner程序的操作和4.2.3.2节 nRF52安装Client和Server应用
步骤一样的。如有机会后面会进行详细补充。
这里针对的是安卓系统上进行实例,对于nRF Mesh安卓应用程序的安装没有什么要讲的,直接下载apk文件,然后进行安装即可。把相应的权限打开就行。
Client和Server烧录到nRF52板子的操作是一样的,所以这里只用Client来说明。
找到nrf5_SDK_for_Mesh_v310_src
目录(这里名字可能不一样),这是官方提供的nRF5对mesh的SDK,然后找到该目录下\nrf5_SDK_for_Mesh_v310_src\examples\light_switch
目录,里面有client
、server
和provisioner
三个文件夹,分别对应switch
、light
和provisoner
。这里以client为例,所以继续打开client目录,用SES软件打开light_switch_client_nrf52840_xxAA_s140_6_1_0.emProject
这个Solution。
在Build
之前,请确保nRF5_SDK_15.2.0_9412b96
与nrf5_SDK_for_Mesh_v310_src
在同一级,如下图所示。这么做是因为nRF5_for_mesh是基于nRF5_15版本的SDK的,所以请保持它们在同一目录级,否则会出现找不到某些文件。
先点击Target -> Erase All
擦出板子上的程序,然后点击Build
去编译该Solution,编译完后选择Target -> Connect J-Link
,之后会弹出一个让你选择的是哪块板子的对话框,如下图,这里随便选一个,但是之后安装Server要选择另外一个。图片上所显示的SN号,就是对应着每块板子上J-link OB主芯片上的一串数字。
然后点击Target -> Download xx
把程序下载到开发板即可。Server和Provisioner烧录程序是一样的,不再重复说明。
- 添加节点。打开nRF Mesh App,点击由下角的+,如下图,之后就会弹出授予权限的要求,把对应的权限都选择允许。这里会要求把蓝牙打开。
- 扫描节点。接下来就会进行Scanner,就会出现如下图所示,一个是Switch对应的就是Client客户端,一个就是Light对应的就是Server服务端。
- Light加入Provisioner中。点击Mesh Light,就会跳转界面,然后点击IDENTIFY,然后在接下来的界面点击PROVISION,之后会跳出一个对话框,选择OOB类型,选择ok,接下来就会显示配置过程,直到出现配置完成对话框出现,点击ok。在App底部的Network标签下就会出现配置好的Mesh Light。整个过程如下五张图片所示。
- Switch加入Provisioner中。上面还只是把Light加入到Provisioner之中,还有switch没有加入。接下来继续点击右下角的+,在scanner界面就会显示switch,然后点击Mesh Switch,跳转到相应的界面,然后点击IDENTIFY,然后在接下来的界面点击PROVISION,之后会跳出一个对话框,选择OOB类型,选择ok,接下来就会显示配置过程,直到出现配置完成对话框出现,点击ok。在App底部的Network标签下就会出现配置好的Mesh Light和Mesh Switch。整个过程如下五张图片所示。
- 将Generic OnOff 客户端和服务端模型实例绑定相同的AppKey。选中App底部的Network标签,这里只以Light为例,点击列出来的齿轮状的图标,然后选择Element:0x0001,出现下图3所示;然后点击Generic On Off Server,出现下图4所示;点击BIND KEY,出现下图5所示;点击App key 0,也就绑定了App key。Switch的操作也是一样的,不再重复,在Swtich会有三个Element,可以发现后面两个Element(0x0003和0x0004)才是Generic On Off Client,选择其中一个绑定Appkey,我这里选择的是Element:0x0003。
- 对Client客户端设置Server服务器地址。首先查看Light服务器的地址,可以发现地址为0x0001,如下图1所示。点击Switch的齿轮图标,选择Element:0x0003的Generic On Off Client,如下图2所示。在Publish下选择SET PUBLICATION,如下图3所示。接下来选择Public Address,输入地址:0x0001(这里地址可能不一样,填写图1所示的地址),然后点击ok,如图5所示。接下来点击APPLY,如图6所示。最后完成如图7所示。
按Switch客户端的Button1可以同时打开客户端和服务端的LED1,按Switch客户端的Button2可以同时关闭客户端和服务端的LED1。
这部分将会深入去学习ndn-lite相关的内容,对使用ndn-lite打下基础。
ndn-lite的详细介绍请见:https://github.com/named-data-iot/ndn-lite/wiki 。ndn-lite库旨在提供多个核心NDN网络栈。该库允许应用程序直接有访问控制、服务发现、模式化信任等功能。
ndn-lite系统结构图如下:
打开SES软件,可以看到ndn-lite的代码结构如下图所示。
./encode
目录:NDN包的编码和解码。./forwarder
目录:NDN轻量级的转发实现和网络Face抽象。./face
目录:网络face和应用face的实现。每个face实例可能需要硬件/OS适配。./securtiy
目录:安全支持。./app-support
目录:访问控制、服务发现和其他可以促进应用程序开发的高级模块。./adaptation
目录:硬件/操作系统适配。当使用ndn-lite时,开发人员应该为他们的应用程序开发所使用的平台/OS选择一个或多个适配。
这个实战是基于第3部分的基础上来进行修改的,还是用nRF52840开发板上的实例程序,以及另外一个安卓App来实现这个操作。下面先看一下需要实现的需求和效果,来有个大致的印象。
这里用到的是两块nRF52840板子和一台安卓手机。
这里需要提前了解有两种信任策略,一种是Only controller,一种是All node。Only Controller的意思就是用户不能通过按下按钮1或者从另一个板发送命令来打开LED 1。All node的意思就是相当于一个普通的节点,无论是谁发过来的任何相应的开关灯的interest包,只要正确都会进行响应。
这个阶段两个nRF52840板子的LED3将会闪烁3次。这表示的板子正在进行初始化相关的工作。
除了上电LED3闪烁以外,以后LED3每次闪烁3次都代表的是板子正在进行sign on阶段。
在all node模式下,按下Button1,LED1会亮。在only controller模式下无反应。
在all node模式下,按下Buttion2,关闭LED1。
在all node模式下,按下其中一块板子的button3可以打开另外一块板子的LED1(LED1会闪烁3次)。如果对其中一块板子设置成了Only controller模式,另外一块板子为all node模式,那么设置成all node模式的那块板子按button3是对另外一块板子没有任何效果的,但是设置成only controller的那块板子按下button3对另外一块板子是有效果的(LED1会闪烁3次)。
打开App界面上的开关按钮,可以打开对应板子上的LED1,关闭App上界面的开关按钮,可以关闭LED1.
可以点击板子对应的图片上点击一下会弹出选项框,进行选择;也可以点击板子图片旁边的来选择其中一种策略,对应的板子LED3将会闪烁3次,之后LED4一直常亮,表示信任策略切换完毕。
这个App源代码下载地址为:https://github.com/ZoharAndroid/ndn-lite-ble.git 。该库中的/ndn-lite-ble-android
目录就是这个实例的android的源代码程序。
然后用Android Studio导入这个ndn-lite-ble-android工程,操作步骤为:File -> Open
。然后编译运行安装到手机上。如果遇到需要安装相应sdk版本,按照AS提示进行安装即可。具体操作在前面章节已经说过,可以参考前面的章节的内容。
当然,这个App还有未完成的功能,后续会继续添加。其中,如果遇到Bug,还请把情况反馈给我(Email:[email protected]或者github上的issues)。
这里也需要修改一下nRF52840源代码中的程序。
用SES软件打开3.3节提到的nRF5程序源代码,然后进行如下修改。
在main函数外,添加on_led(index)和off_led(index)代码,用来开关对应的LED[index]。
// on led1
static void on_led(int i){
const uint32_t pin = NRF_GPIO_PIN_MAP(0, 12 + i);
nrf_gpio_cfg_output(pin);
nrf_gpio_pin_write(pin, 0);
}
// off led1
static void off_led(int i){
const uint32_t pin = NRF_GPIO_PIN_MAP(0, 12 + i);
nrf_gpio_cfg_output(pin);
nrf_gpio_pin_write(pin, 1);
}
这里主要是添加对应的interst名称,然后对名字进行判定操作。
int on_CMDInterest(const uint8_t *interest, uint32_t interest_size) {
APP_LOG("Get into on_CMDInterest... Start to decode received Interest\n");
APP_LOG("interst -> %d, interest_size %d \n " , interest ,interest_size);
//initiate the name prefix of different interest here
ndn_name_t CMD_prefix;
ndn_name_t CMD_prefix_on;
ndn_name_t CMD_prefix_off;
#ifdef BOARD_1
char CMD_string[] = "/NDN-IoT/Board1/SD_LED/ON";
char CMD_string_on[] = "/NDN-IoT/Board1/LED/ON";
char CMD_string_off[] = "/NDN-IoT/Board1/LED/OFF"; // off led
#endif
#ifdef BOARD_2
char CMD_string[] = "/NDN-IoT/Board2/SD_LED/ON"; // 闪烁
char CMD_string_on[] = "/NDN-IoT/Board2/LED/ON"; // on led
char CMD_string_off[] = "/NDN-IoT/Board2/LED/OFF"; // off led
#endif
ndn_name_from_string(&CMD_prefix, CMD_string, sizeof(CMD_string));
ndn_name_from_string(&CMD_prefix_on, CMD_string_on, sizeof(CMD_string_on));
ndn_name_from_string(&CMD_prefix_off, CMD_string_off, sizeof(CMD_string_off));
ndn_interest_t check_interest;
int result = ndn_interest_from_block(&check_interest, interest, interest_size);
APP_LOG("compare results of SD_led ON: %d\n", ndn_name_compare(&CMD_prefix, &check_interest.name));
APP_LOG("compare results of led on: %d\n", ndn_name_compare(&CMD_prefix_on, &check_interest.name));
APP_LOG("compare results of led off: %d\n", ndn_name_compare(&CMD_prefix_off, &check_interest.name));
if (ndn_name_compare(&check_interest.name, &CMD_prefix) == 0) {
APP_LOG("Get into on_CMDtInterest... Received command to turn on LED\n");
if (schematrust_flag) {
blink_led(1);
APP_LOG("finish blink led 2");
}
}else if (ndn_name_compare(&check_interest.name, &CMD_prefix_on) == 0){
APP_LOG("Get cmdinterest to alwasy turn on LED\n");
on_led(1);
}else if(ndn_name_compare(&check_interest.name, &CMD_prefix_off) == 0){
APP_LOG("Get cmdinterest to alwasy turn off LED\n");
off_led(1);
}
}