EFR32 ZDDC 设备开发
本文将介绍如何在 EFR32 上使用 ZDDC 协议和 ACOINFO ZigBee 通用设备应用规范开发一个能接入到 EdgerOS 的 ZigBee 智能设备。
硬件准备
EFR32 开发板
EFR32 是 Silicon lab 推出的同时集成 RF 射频收发器和 ARM Cortex M4 内核 MCU 的无线射频 SOC 解决方案。其主频为 40MHz,片上最高输出功率可达 20dBm(Sub1GHz:20dBm,2.4GHZ:19.5dBm)。内置 32KB RAM 和 256KB Flash。拥有丰富的外设资源,其射频部分可同时覆盖 2.4GHz 和 Sub1GHz 频段。
本实验使用的是一块画时科技的 EFR32 开发板,芯片型号为 EFR32MG21A020F768IM32 实物图如下所示:
data:image/s3,"s3://crabby-images/5f76a/5f76a5ec7228bb26434adf80b2038fabeb27b4ee" alt=""
软件准备
开发 EFR32 需要使用 Simplicity Studio IDE。
下载 IDE 软件包并安装
网盘链接:https://pan.baidu.com/s/1ZbmzPaex95f0cD4CHz4ztg 提取码:lvle
解压 SimplicityStudio-EmberZnet 6.7.8.0-WIN10.7z 文件到 D:\
,进入 D:\v4
目录双击 studio.exe 打开 IDE ,
出现创建 workspace 失败,不需理会:
data:image/s3,"s3://crabby-images/39f9b/39f9b1e05d3a521f373b247244b3423d65249c35" alt=""
点击两次 OK 后会进入选择 workspace 界面,选择或创建 workspace 如下所示:
data:image/s3,"s3://crabby-images/ef89e/ef89ea8f2441e3436adeff0b4106ac7d1fd7a690" alt=""
检查更新后主界面如下所示:
data:image/s3,"s3://crabby-images/66957/66957eaaf6038b44e59e09332ac5d628032172f1" alt=""
新建工程
点击 File > New > Project ,选择 Silicon Labs AppBuilder Project ,点击 Next 如下所示:
data:image/s3,"s3://crabby-images/7652b/7652b9b7ccf7ba70c60678f49c5767a0dbb6bc9d" alt=""
选择 Silicon Labs Zigbee 后点击 Next 如下所示:
data:image/s3,"s3://crabby-images/86161/86161b80ace58f007d4339472fbec06b7ab46c42" alt=""
选择 EmberZNet 6.7.8.0 GA SoC 6.7.8.0 后点击 Next 如下所示:
data:image/s3,"s3://crabby-images/70dbf/70dbf406652fddc1cb0dd366183ab8f36b0079aa" alt=""
选择 ZigbeeMinimal 后点击 Next 如下所示:
data:image/s3,"s3://crabby-images/367e9/367e93a51499b3b355c3e7120ae35bc601839cfa" alt=""
设置 Project name 后点击 Next 如下所示:
data:image/s3,"s3://crabby-images/85140/85140f6cecca3b823c99668698d3f71680d0b531" alt=""
选择 Part 为 EFR32MG21A020F768IM32
后点击 Finish 如下所示:
data:image/s3,"s3://crabby-images/32e04/32e04b48078a38397bde4a5ceb684ae0a3f53051" alt=""
至此工程创建完成:
data:image/s3,"s3://crabby-images/558d1/558d105cc6d2a7d7f0158fbc728ad5433964bdf8" alt=""
配置工程
选择 ZigbeeMinimalSoc.isc 文件中的 Zigbee Stack 选项,选择 Zigbee Device Type 为 End Device 如下所示:
data:image/s3,"s3://crabby-images/dba09/dba09b7305f1465ea455fe0ee5bf949849a98d85" alt=""
复制附录中的 acoinfo 自定义 Cluster 扩展规范文件 acoinfo.xml
,拷贝到 D:\v4\developer\sdks\gecko_sdk_suite\v2.7\app\zcl
目录下,Zigbee Stack 选项中点击 Add... ,添加 acoinfo.xml 到工程中,如下所示:
data:image/s3,"s3://crabby-images/c07c3/c07c380b9a7ab316055a1333c24a132d02946f52" alt=""
选择 ZCL Clusters 选项,按照下图所示选择 HA On/Off Light :
data:image/s3,"s3://crabby-images/9dbba/9dbba994b52155b52a01ed62dfa119f3cf905c07" alt=""
选择 ZCL Clusters 选项,点击下方 General 中的 Basic 在右侧添加 manufacturer name 为 acoinfo.light.led
,model id 为 acoinfo.zigbee.generic
如下所示:
data:image/s3,"s3://crabby-images/3d695/3d695a04c96105702f5bcbfd8a33018cf6953435" alt=""
选择 General 中的 On/Off ,勾选 DIO 通道 1,如下所示:
data:image/s3,"s3://crabby-images/80fdf/80fdf2ec8e1809664948e8e1b6b576be17f22f54" alt=""
选择 Lighting 中的 Color Control ,在右侧选择 acoinfo 自定义的六个 attribute ,修改 DIO 通道可用和可写以及 AIO 通道的可用的默认值为 0x00000001,如下所示:
data:image/s3,"s3://crabby-images/f7675/f767557f0e6184ca9c1c4e5ba1c9222715c17038" alt=""
勾选 ZLL 中的 ZLL Commissioning ,以及 acoinfo 中的 acoinfo_aio 中的 AIO 通道 1,如下所示:
data:image/s3,"s3://crabby-images/0a505/0a5056de8c897fa786495360e736b709c7930dc8" alt=""
选择 HAL 选项中的 Bootloader,设置为 None,如下所示:
data:image/s3,"s3://crabby-images/8feb0/8feb0e01521410c6106b4348aea3154c148ecbdc" alt=""
选择 Callbacks 选项,选中 Main Init 选项、Hal Button Isr 选项以及 Server Attribute Changed 选项,如下所示:
data:image/s3,"s3://crabby-images/17725/17725dbb9bc84425ee4760deb53d1e3346bfea5c" alt=""
data:image/s3,"s3://crabby-images/625e1/625e1ff9d46fb26ef5d1fdd2e7c5740f75bf9805" alt=""
选择 Includes 选项,点击 Add new 添加一个自定义事件,用于定期上报模拟量通道 1,Command 为 AcoinfoAioReportEventControl
,Callback 为 AcoinfoAioReportEventHandler
,如下所示:
data:image/s3,"s3://crabby-images/edd4a/edd4a1748098a71867dc1abe99a0fefd1f1df765" alt=""
保存后点击右上角 Generate 后会生成代码,如下所示:
data:image/s3,"s3://crabby-images/51a33/51a338166302983e9a6ba88ca7885bd426807e97" alt=""
双击 ZigbeeMinimalSoc_callbacks.c
文件,在文件末尾加入上面选中的 emberAfMainInitCallback
,并延时调用周期上报 AIO 事件,如下所示:
void emberAfMainInitCallback(void)
{
emberAfCorePrintln("---------------test---------------");
emberEventControlSetDelayMS(AcoinfoAioReportEventControl, 10000);
}
在上述文件中加入周期上报 AIO 处理函数 AcoinfoAioReportEventHandler
,如下所示:
EmberEventControl AcoinfoAioReportEventControl;
void AcoinfoAioReportEventHandler(void)
{
// 在下次使用之前禁用该事件
emberEventControlSetInactive(AcoinfoAioReportEventControl);
float data = 3.14;
uint8_t * p_data = (uint8_t *)&data;
uint8_t buf[7] = {0};
buf[0] = ZCL_ACOINFO_ZB_AIO_ATTR_1_ATTRIBUTE_ID && 0xFF;
buf[1] = ZCL_ACOINFO_ZB_AIO_ATTR_1_ATTRIBUTE_ID >> 8;
buf[2] = ZCL_FLOAT_SINGLE_ATTRIBUTE_TYPE;
for(int i=0;i<4;i++){
buf[3+i] = *p_data++;
}
emberAfFillCommandGlobalServerToClientReportAttributes(ZCL_ACOINFO_ZB_AIO_CLUSTER_ID,
(uint8_t *)buf, 7);
emberAfSetCommandEndpoints(1, 1);
emberAfSendCommandUnicast(EMBER_OUTGOING_DIRECT, 0x0000);
// 延迟 5 秒后重新触发事件
emberEventControlSetDelayMS(AcoinfoAioReportEventControl, 5000);
}
在上述文件中加入监听 DIO 值变化回调函数,并加入开关 LED 灯的功能, 如下所示:
void emberAfOnOffClusterServerAttributeChangedCallback(int8u endpoint,
EmberAfAttributeId attributeId)
{
EmberAfStatus status;
uint8_t data[1];
status = emAfReadAttribute(endpoint,
ZCL_ON_OFF_CLUSTER_ID,
attributeId,
0x40,
0x0000,
data,
1,
NULL);
if (status == EMBER_ZCL_STATUS_SUCCESS) {
if(attributeId == ZCL_ACOINFO_ZB_DIO_ATTR_1_ATTRIBUTE_ID){
if(data[0] == 0x01){
halSetLed(1); // 开启 LED 灯
emberAfCorePrintln("led on");
} else if(data[0] == 0x00) {
halClearLed(1); // 关闭 LED 灯
emberAfCorePrintln("led off");
} else {
emberAfCorePrintln("data error");
}
}
} else {
emberAfCorePrintln("read attribute failed... status = %x", status);
}
}
在上述文件中加入 BUTTON 监听回调函数,检测按下后开始配置入网,如下所示:
void emberAfHalButtonIsrCallback(uint8_t button, uint8_t state)
{
if (state == BUTTON_RELEASED) {
emberAfPluginNetworkSteeringStart();
}
}
双击左侧 custom_efr32mg21a020f768im32.hwconf 文件后点击下方 DefaultMode Peripherals 选项来配置管脚:
data:image/s3,"s3://crabby-images/a74cb/a74cb6171f0affefd6c920788723d71931d29a14" alt=""
根据开发板 Button 设置如下所示:
data:image/s3,"s3://crabby-images/b775e/b775ebadd77501653790f100c568f5a78e51f0a5" alt=""
根据开发板 LED 设置如下所示:
data:image/s3,"s3://crabby-images/0e794/0e7947aedee7edc3061e8d2bf5bc111694410023" alt=""
设置日志输出配置如下所示:
data:image/s3,"s3://crabby-images/76e66/76e66bb9dab3875cbb10125fba6cd37481e01a11" alt=""
取消勾选 Virtual COM Port :
data:image/s3,"s3://crabby-images/7339d/7339dd71df96ef2118d18f1b92d29b5df94862d3" alt=""
设置 USART0 如下所示:
data:image/s3,"s3://crabby-images/b6eb5/b6eb55c49a493e57b240344027eb5dce9c0ea280" alt=""
取消勾选 PTI:
data:image/s3,"s3://crabby-images/21348/21348e59aaf7981a7364141f261951a8707a0db0" alt=""
保存后会生成相关硬件配置代码,点击编译生成二进制文件:
data:image/s3,"s3://crabby-images/6ef42/6ef424549c639a14089a3cd56cfe5ba368e280ce" alt=""
烧写镜像
使用 JLink 把开发板连接到 PC,如下所示:
data:image/s3,"s3://crabby-images/9a9c7/9a9c7a5c79e34883f46dcb11ad6d99a1ce440c39" alt=""
点击右上角 Launcher 进入主页,再点击左侧工具图标,在弹出框中选择 Simplicity Commander 后点击 OK:
data:image/s3,"s3://crabby-images/ab775/ab7754dea18ac8ed22e80bc790cdca62dd6e2eb4" alt=""
选择左侧第三个选项 Flash,然后依次点击上方 Adapter 和 Target 右边的 Connect 按键,点击 Browse 选择对应路径下的 Z3SwitchSoc.hex 文件,点击 Erase chip 擦除,点击 Flash 进行烧写,如下所示:
data:image/s3,"s3://crabby-images/aa6eb/aa6eb1882fd21837fb82eaa7d4a3de7570b06bbe" alt=""
入网验证
验证前,请参考《 ZDDC 协议介绍》章节中的 ZigBee 设备入网流程,完成以下步骤:
进入扫描设备界面,如下图所示:
将 EFR32 开发板接入 PC,打开设备管理器,查看 PC 为 EFR32 开发板分配的 COM 号,如下图所示:
使用 PuTTY 作为串口调试工具,打开 PuTTY 进入 PuTTY Configuration 界面,选择 Session 类, Connection type 选择:Serial ,Serial line 为 EFR32 开发板接入 PC 后分配的 COM 号(如
COM23
),Speed 设置为115200
,最后点击 Open 按键进入 PuTTY 终端:开发板上电,按下 PD01 执行关联入网操作,串口打印结果如下图所示:
此时,EdgerOS 会发现新的设备,点击添加按钮完成新设备的添加:
附录
acoinfo.xml 文件
<configurator>
<domain name="acoinfo" />
<cluster>
<name>acoinfo_aio</name>
<domain>acoinfo</domain>
<description>Acoinfo custom cluster</description>
<code>0x000C</code>
<define>ACOINFO_ZB_AIO_CLUSTER</define>
<client init="false" tick="false">true</client>
<server init="false" tick="false">true</server>
<attribute side="server" code="0x2000" define="ACOINFO_ZB_AIO_ATTR_1"
type="INT32U" min="0x00000000" max="0xFFFFFFFF" writable="true"
default="0x00000000" optional="true">ACOINFO_ZB_AIO_ATTR_1</attribute>
<attribute side="server" code="0x2001" define="ACOINFO_ZB_AIO_ATTR_2"
type="INT32U" min="0x00000000" max="0xFFFFFFFF" writable="true"
default="0x00000000" optional="true">ACOINFO_ZB_AIO_ATTR_2</attribute>
<attribute side="server" code="0x2002" define="ACOINFO_ZB_AIO_ATTR_3"
type="INT32U" min="0x00000000" max="0xFFFFFFFF" writable="true"
default="0x00000000" optional="true">ACOINFO_ZB_AIO_ATTR_3</attribute>
<attribute side="server" code="0x2003" define="ACOINFO_ZB_AIO_ATTR_4"
type="INT32U" min="0x00000000" max="0xFFFFFFFF" writable="true"
default="0x00000000" optional="true">ACOINFO_ZB_AIO_ATTR_4</attribute>
<attribute side="server" code="0x2004" define="ACOINFO_ZB_AIO_ATTR_5"
type="INT32U" min="0x00000000" max="0xFFFFFFFF" writable="true"
default="0x00000000" optional="true">ACOINFO_ZB_AIO_ATTR_5</attribute>
<attribute side="server" code="0x2005" define="ACOINFO_ZB_AIO_ATTR_6"
type="INT32U" min="0x00000000" max="0xFFFFFFFF" writable="true"
default="0x00000000" optional="true">ACOINFO_ZB_AIO_ATTR_6</attribute>
<attribute side="server" code="0x2006" define="ACOINFO_ZB_AIO_ATTR_7"
type="INT32U" min="0x00000000" max="0xFFFFFFFF" writable="true"
default="0x00000000" optional="true">ACOINFO_ZB_AIO_ATTR_7</attribute>
<attribute side="server" code="0x2007" define="ACOINFO_ZB_AIO_ATTR_8"
type="INT32U" min="0x00000000" max="0xFFFFFFFF" writable="true"
default="0x00000000" optional="true">ACOINFO_ZB_AIO_ATTR_8</attribute>
</cluster>
<clusterExtension code="0x0006">
<attribute side="server" code="0x2000"
define="ACOINFO_ZB_DIO_ATTR_1"
type="BOOLEAN" min="0x00" max="0x01" writable="true"
default="0x00" optional="true">ACOINFO_ZB_DIO_ATTR_1</attribute>
<attribute side="server" code="0x2001"
define="ACOINFO_ZB_DIO_ATTR_2"
type="BOOLEAN" min="0x00" max="0x01" writable="true"
default="0x00" optional="true">ACOINFO_ZB_DIO_ATTR_2</attribute>
<attribute side="server" code="0x2002"
define="ACOINFO_ZB_DIO_ATTR_3"
type="BOOLEAN" min="0x00" max="0x01" writable="true"
default="0x00" optional="true">ACOINFO_ZB_DIO_ATTR_3</attribute>
<attribute side="server" code="0x2003"
define="ACOINFO_ZB_DIO_ATTR_4"
type="BOOLEAN" min="0x00" max="0x01" writable="true"
default="0x00" optional="true">ACOINFO_ZB_DIO_ATTR_4</attribute>
<attribute side="server" code="0x2004"
define="ACOINFO_ZB_DIO_ATTR_5"
type="BOOLEAN" min="0x00" max="0x01" writable="true"
default="0x00" optional="true">ACOINFO_ZB_DIO_ATTR_5</attribute>
<attribute side="server" code="0x2005"
define="ACOINFO_ZB_DIO_ATTR_6"
type="BOOLEAN" min="0x00" max="0x01" writable="true"
default="0x00" optional="true">ACOINFO_ZB_DIO_ATTR_6</attribute>
<attribute side="server" code="0x2006"
define="ACOINFO_ZB_DIO_ATTR_7"
type="BOOLEAN" min="0x00" max="0x01" writable="true"
default="0x00" optional="true">ACOINFO_ZB_DIO_ATTR_7</attribute>
<attribute side="server" code="0x2007"
define="ACOINFO_ZB_DIO_ATTR_8"
type="BOOLEAN" min="0x00" max="0x01" writable="true"
default="0x00" optional="true">ACOINFO_ZB_DIO_ATTR_8</attribute>
</clusterExtension>
<clusterExtension code="0x0300">
<attribute side="server" code="0x2000" define="ACOINFO_ZB_DIO_ACTIVE_T_ATTR"
type="INT32U" min="0x00000000" max="0xFFFFFFFF" writable="true"
default="0x00000000" optional="true">ACOINFO_ZB_DIO_ACTIVE_ATTR</attribute>
<attribute side="server" code="0x2001" define="ACOINFO_ZB_DIO_WRITE_T_ATTR"
type="INT32U" min="0x00000000" max="0xFFFFFFFF" writable="true"
default="0x00000000" optional="true">ACOINFO_ZB_DIO_WRITE_ATTR</attribute>
<attribute side="server" code="0x2002" define="ACOINFO_ZB_AIO_ACTIVE_T_ATTR"
type="INT32U" min="0x00000000" max="0xFFFFFFFF" writable="true"
default="0x00000000" optional="true">ACOINFO_ZB_AIO_ACTIVE_ATTR</attribute>
<attribute side="server" code="0x2003" define="ACOINFO_ZB_AIO_WRITE_T_ATTR"
type="INT32U" min="0x00000000" max="0xFFFFFFFF" writable="true"
default="0x00000000" optional="true">ACOINFO_ZB_AIO_WRITE_ATTR</attribute>
<attribute side="server" code="0x2004" define="ACOINFO_ZB_MEM_ACTIVE_T_ATTR"
type="INT32U" min="0x00000000" max="0xFFFFFFFF" writable="true"
default="0x00000000" optional="true">ACOINFO_ZB_MEM_ACTIVE_ATTR</attribute>
<attribute side="server" code="0x2005" define="ACOINFO_ZB_MEM_WRITE_T_ATTR"
type="INT32U" min="0x00000000" max="0xFFFFFFFF" writable="true"
default="0x00000000" optional="true">ACOINFO_ZB_MEM_WRITE_ATTR</attribute>
</clusterExtension>
<clusterExtension code="0x0500">
<attribute side="server" code="0x2000"
define="ACOINFO_ZB_MEM_ATTR_1"
type="INT32U" min="0x00000000" max="0xFFFFFFFF" writable="true"
default="0x00000000" optional="true">ACOINFO_ZB_MEM_ATTR_1</attribute>
<attribute side="server" code="0x2001"
define="ACOINFO_ZB_MEM_ATTR_2"
type="INT32U" min="0x00000000" max="0xFFFFFFFF" writable="true"
default="0x00000000" optional="true">ACOINFO_ZB_MEM_ATTR_2</attribute>
<attribute side="server" code="0x2002"
define="ACOINFO_ZB_MEM_ATTR_3"
type="INT32U" min="0x00000000" max="0xFFFFFFFF" writable="true"
default="0x00000000" optional="true">ACOINFO_ZB_MEM_ATTR_3</attribute>
<attribute side="server" code="0x2003"
define="ACOINFO_ZB_MEM_ATTR_4"
type="INT32U" min="0x00000000" max="0xFFFFFFFF" writable="true"
default="0x00000000" optional="true">ACOINFO_ZB_MEM_ATTR_4</attribute>
<attribute side="server" code="0x2004"
define="ACOINFO_ZB_MEM_ATTR_5"
type="INT32U" min="0x00000000" max="0xFFFFFFFF" writable="true"
default="0x00000000" optional="true">ACOINFO_ZB_MEM_ATTR_5</attribute>
<attribute side="server" code="0x2005"
define="ACOINFO_ZB_MEM_ATTR_6"
type="INT32U" min="0x00000000" max="0xFFFFFFFF" writable="true"
default="0x00000000" optional="true">ACOINFO_ZB_MEM_ATTR_6</attribute>
<attribute side="server" code="0x2006"
define="ACOINFO_ZB_MEM_ATTR_7"
type="INT32U" min="0x00000000" max="0xFFFFFFFF" writable="true"
default="0x00000000" optional="true">ACOINFO_ZB_MEM_ATTR_7</attribute>
<attribute side="server" code="0x2007"
define="ACOINFO_ZB_MEM_ATTR_8"
type="INT32U" min="0x00000000" max="0xFFFFFFFF" writable="true"
default="0x00000000" optional="true">ACOINFO_ZB_MEM_ATTR_8</attribute>
</clusterExtension>
</configurator>