基于nRF7002-DK的NFC功能切换系统(nRF Connect SDK+NFC)

👉 【Funpack2-6】基于nRF7002-DK的NFC功能切换系统
👉 Github: EmbeddedCamerata/nRF7002-DK-nfc-function-switching

项目介绍

本项目基于nRF7002-DK,使用nRF Connect SDK v2.4.2 开发,使用NFC外设,实现NFC记录英文文本信息、中文文本信息与打开安卓应用三个功能,并可通过按键切换,通过手机NFC触碰即可触发。

👉 官网:nRF Connect SDK

硬件介绍

nRF7002-DK是用于nRF7002 Wi-Fi 6协同IC的开发套件,该开发套件采用nRF5340多协议片上系统 (SoC) 作为nRF7002的主处理器,在单一的电路板上包含了开发工作所需的一切,可让开发人员轻松开启基于nRF7002 的物联网项目。该 DK 包括 Arduino 连接器、两个可编程按钮、一个 Wi-Fi 双频段天线和一个低功耗蓝牙天线,以及电流测量引脚。

这款DK支持低功耗 Wi-Fi 应用开发,并实现了多项 Wi-Fi 6 功能,比如 OFDMA、波束成型和 TWT。nRF7002 Wi-Fi 6配套IC为另一个主机添加了低功耗Wi-Fi 6功能,提供无缝连接和基于Wi-Fi的定位(本地Wi-Fi集线器的SSID嗅探)功能。该IC设计用于搭配Nordic现有的nRF52®和nRF53®系列多协议片上系统 (SoC) 和nRF91®系列蜂窝物联网系统级封装 (SiP) 使用。nRF7002 IC 还可与非nordic主机器件搭配使用。通过SPI或QSPI与主机通信,并带有额外的共存功能,可与其他协议如蓝牙、Thread或Zigbee无缝共存。

nRF7002在Nordic的nRF Connect SDK中提供集成和支持。

板卡特性:

  • 用于nRF7002双频带Wi-Fi 6配套IC的开发套件
  • nRF5340 SoC主机器件
  • Wi-Fi 6 (IEEE 802.11 a/b/g/n/ac/ax)、蓝牙低功耗(LE)、蓝牙网状网络、802.15.4、Thread、Zigbee®、ANT、2.4GHz专有和NFC无线协议支持2.4GHz、5GHz芯片和NFC天线
  • SWF射频连接器
  • SEGGER J-Link板载编程器/调试器
  • 用户可编程LED (2x) 和按钮 (2x)
  • 用于测量功耗的引脚
  • 来自USB、外部或锂聚合物电池的2.9V至5.0V电源
  • Arduino连接器

nRF7002结构框图

项目设计

开发环境及工程目录

根据 官网文档 手动安装SDK等依赖,nRF Connect SDK 版本 v2.4.2 。由于笔者是Linux环境,且有一定的洁癖,因此在安装 west 的时候单独为其使用Poetry,创建了一个Python虚拟环境。并在此基础上,安装额外的Python依赖。每次使用 west 先激活Python虚拟环境即可。

👉 Poetry

工程目录可根据 ncs/nrf/samples 中的sample复制一份,并在此基础上添加自己的代码或配置。例如:

├ build(编译时所产生的构建文件)
├ src
│ ├ main.c
│ └ ...(如有需要,其他的源代码)
├ CMakeLists.txt
├ prj.conf
├ sample.yaml
├ README.md
└ ...

prj.conf 中配置需要启用的库,从而在编译程序的时候编译对应的头文件。sample.yaml 仅做描述示例程序之用。此外,Kconfig 是为了用户可在menuconfig内手动配置某些选项(主要是可选功能),在本工程中省略。

CMakeLists.txt 内,额外添加 set(BOARD nrf7002dk_nrf5340_cpuapp) ,从而在 west build 时可省略 -b nrf7002dk_nrf5340_cpuapp) 编译选项。

cmake_minimum_required(VERSION 3.20.0)
set(BOARD nrf7002dk_nrf5340_cpuapp)
find_package(Zephyr REQUIRED HINTS $ENV{ZEPHYR_BASE})
project(nfc_function_switching)
FILE(GLOB app_sources src/*.c)
# NORDIC SDK APP START
target_sources(app PRIVATE ${app_sources})
# NORDIC SDK APP END

总体流程图

工程流程图

硬件初始化

在此初始化板卡上的LED与按键。并将LED2打开以表示板卡正常工作。

dk_buttons_init(NULL);
dk_leds_init();
dk_set_led_on(SYSTEM_ON_LED);

NFC功能实现

首先通过 nfc_t2t_setup() 注册NFC事件的回调函数,其回调函数内主要响应NFC标签检测到外部NFC场时与移除时的事件。当检测到外部NFC场时,LED1亮起,移除时灭。

nfc_t2t_setup(nfc_callback, NULL)
static void nfc_callback(void *context,
						 nfc_t2t_event_t event,
						 const uint8_t *data,
						 size_t data_length)
{
	ARG_UNUSED(context);
	ARG_UNUSED(data);
	ARG_UNUSED(data_length);

	switch (event)
	{
	case NFC_T2T_EVENT_FIELD_ON:
		dk_set_led_on(NFC_FIELD_LED);
		break;
	case NFC_T2T_EVENT_FIELD_OFF:
		dk_set_led_off(NFC_FIELD_LED);
		break;
	default:
		break;
	}
}

文本记录

主要使用nRF提供的nfc库。先生成NDEF文本记录描述符,再将这个记录加入NDEF消息中,之后 nfc_ndef_msg_encode 编码,并存入 buffer 内,长度为 len

static int nfc_text_encode(uint8_t *buffer, uint32_t *len)
{
	NFC_NDEF_TEXT_RECORD_DESC_DEF(nfc_text_rec,
								  UTF_8,
								  en_code,
								  sizeof(en_code),
								  en_payload,
								  sizeof(en_payload));

	NFC_NDEF_MSG_DEF(nfc_text_msg, MAX_REC_COUNT);

	/* Add record */
	if (nfc_ndef_msg_record_add(&NFC_NDEF_MSG(nfc_text_msg),
								&NFC_NDEF_TEXT_RECORD_DESC(nfc_text_rec)) < 0)
	{
		printk("Cannot add record!\n");
		return -1;
	}

	/* Encode text message */
	if (nfc_ndef_msg_encode(&NFC_NDEF_MSG(nfc_text_msg), buffer, len) < 0)
	{
		printk("Cannot encode message!\n");
		return -1;
	}

	return 0;
}

需要注意的是如何编码中英文文本信息至NDEF。对于英文使用UTF-8编码,直接编码ASCII字符,例如“Hello, World!”:

static const uint8_t en_payload[] = {
	'H', 'e', 'l', 'l', 'o', ' ', 'W', 'o', 'r', 'l', 'd', '!'};
static const uint8_t en_code[] = {'e', 'n'};

NFC_NDEF_TEXT_RECORD_DESC_DEF(nfc_text_rec,
						      UTF_8,
							  en_code,
							  sizeof(en_code),
							  en_payload,
							  sizeof(en_payload));

而对于中文信息,每个中文字符需要UTF-16编码,但是 NFC_NDEF_TEXT_RECORD_DESC_DEF 宏内传入的数据的类型却是 uint8_t const* ,因此需要将“遥遥领先!”对应十六进制“\u6590\u6590\u8698\u4851\u01ff”拆分,先写低位再高位,使用UTF-16编码:

static const uint8_t zh_payload[] = {
	'\x90', '\x65', '\x90', '\x65', '\x98', '\x86', '\x51', '\x48', '\xff', '\x01'};
static const uint8_t zh_code[] = {'z', 'h'};

NFC_NDEF_TEXT_RECORD_DESC_DEF(nfc_text_rec,
							  UTF_16,
							  zh_code,
							  sizeof(zh_code),
							  zh_payload,
							  sizeof(zh_payload));

安卓应用打开

所需要的是安卓应用的名称 android_pkg_name 。在此,将打开明日方舟 com.hypergryph.arknights 。并通过 nfc_launchapp_msg_encode 完成该类型消息的创建与编码,并存到 buffer,长度为 len

/* Package: com.hypergryph.arknights */
static const uint8_t android_pkg_name[] = {
	'c', 'o', 'm', '.',
	'h', 'y', 'p', 'e', 'r', 'g', 'r', 'y', 'p', 'h', '.',
	'a', 'r', 'k', 'n', 'i', 'g', 'h', 't', 's'};
static int nfc_launchapp_encode(uint8_t *buffer, uint32_t *len)
{
	/* Encode launch app data  */
	if (nfc_launchapp_msg_encode(android_pkg_name,
								 sizeof(android_pkg_name), NULL, 0, buffer, len) < 0)
	{
		printk("Cannot encode message!\n");
		return -1;
	}

	return 0;
}

按键切换NFC功能

实现该功能,需要一个 enum 类型变量 nfc_app_t 存储当前功能的类别。并在初始化NFC功能后,在 while (true) 循环内扫描按键,判断 DK_BTN1 是否按下,按下则切换功能类别,同时根据 nfc_app 重新生成NFC payload并设置。

typedef enum
{
	NFC_APP_TEXT = 0U,
	NFC_APP_LAUNCHAPP
} nfc_app_t;

// In main() function
while (true)
{
	dk_read_buttons(&button_state, NULL);

	if (button_state & DK_BTN1_MSK)
	{
		nfc_app = 1 - nfc_app;
		/* Stop sensing NFC field */
		if (nfc_t2t_emulation_stop() < 0)
		{
			printk("Cannot stop emulation!\n");
			return -1;
		}
		if (nfc_payload_set(nfc_app) < 0)
		{
			printk("NFC payload set failed!\n");
			goto fail;
		}
	}
	k_sleep(K_MSEC(200));
}

需要注意的是,两种功能的payload,长度、内容不同,如果它们共用同一个buffer存储NDEF消息,例如 static uint8_t buffer[256] ,实测会出错。因此,单独为两个功能各自设置一个buffer用于存储NDEF消息。那么,根据 nfc_app 的值,编码与设置不同类别的NDEF消息:

/* Buffer used to hold an NFC NDEF message. */
#define NDEF_MSG_BUF_SIZE 256
static uint8_t text_msg_buf[NDEF_MSG_BUF_SIZE];
static uint8_t launch_app_msg_buf[NDEF_MSG_BUF_SIZE];

// When nfc_app_t == NFC_APP_TEXT
len = sizeof(text_msg_buf);
nfc_text_encode(text_msg_buf, &len);
/* Set created message as the NFC payload */
nfc_t2t_payload_set(text_msg_buf, len);

// When nfc_app_t == NFC_APP_LAUNCHAPP
len = sizeof(launch_app_msg_buf);
nfc_launchapp_encode(launch_app_msg_buf, &len);
/* Set created message as the NFC payload */
nfc_t2t_payload_set(launch_app_msg_buf, len);

功能展示

板卡:

板卡全貌
通过按键切换NFC功能,串口打印提示语句:

串口打印
当功能为记录英文文本信息时,手机NFC触碰后,可读取UTF-8编码的文本信息“Hello, World!”。

NFC读取英文文本信息
当功能为记录中文文本信息时,手机NFC触碰后,可读取UTF-16编码的文本信息“遥遥领先!”。

NFC读取中文文本信息
当功能为记录安卓应用信息时,手机NFC触碰后,可读取应用信息“com.hypergryph.arknights”,即明日方舟包名。此外,当不用NFC标签助手等APP读取NFC信息,而直接触碰,则会唤起应用。

NFC读取安卓应用信息
在手机NFC触碰后,LED2将亮起,手机移开后,NFC连接移除,LED2灭。

👉 详细展示参见:B站:基于nRF7002-DK的NFC功能切换系统

项目总结

本次项目使用nRF7002-DK开发板,使我接触了Nordic家产品的开发环境,N家的开发环境如果使用VSCode插件全套的话,体验还算不错。但只是在笔者Linux端上VSCode插件识别不到板子,因此只能手动安装SDK、Zypher等工具,这个流程上倒是不难,只不过west工具还需要依赖Python,但隔离得不如esp-idf优雅。在编译流程上,有esp-idf的经验那么对于N家的流程就差不多。

工程上,借助NFC实现了多个功能切换,这让我学习了有关NFC的概念,对NFC功能了解更为深入,同时还让我接触到安卓设备投屏至Linux端的软件(推荐scrcpy)。希望日后再有机会用N家的开发板,能更深入地学习设备树、Kconfig、prj.conf、Zypher 等知识。

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: nRF52840-DK开发板是一款由Nordic Semiconductor推出的功能强大的开发板,适用于无线应用的开发和原型设计。该开发板基于nRF52840芯片,具有蓝牙5、蓝牙Mesh、Thread、Zigbee等多种无线通信协议的支持,是开发智能穿戴设备、物联网设备以及传感器网络等应用的理想选择。 nRF52840-DK开发板具有丰富的硬件接口,包括USB、UART、SPI、I2C、GPIO等,使得开发者可以方便地连接、控制各种外部设备。同时,开发板上还集成了按钮、LED指示灯、天线等常用的外设,便于用户进行基础的应用开发和调试。 nRF52840-DK开发板不仅提供了开发所需的硬件资源,还配备了丰富的软件开发工具和库函数。开发者可以使用Nordic Semiconductor提供的nRF5 SDK进行应用开发,还可以通过Keil、IAR和GCC等常用的集成开发环境进行编程。 除此之外,nRF52840-DK开发板还支持在线固件更新(OTA),提供了方便快捷的开发和调试环境。开发者可以使用Segger J-link等调试工具进行调试和烧录。 总之,nRF52840-DK开发板是一款强大而灵活的开发工具,适用于广泛的无线应用开发。无论是初学者还是有经验的开发者,都能够利用该开发板快速构建各种创新的无线应用,并实现功能丰富、稳定的产品。 ### 回答2: nrf52840-dk是一款常用的开发板,广泛用于物联网(IoT)和蓝牙应用的开发。以下是关于该开发板的一些重要信息: 1. 芯片:nrf52840是Nordic Semiconductor公司推出的一款高性能低功耗蓝牙LE SoC(System-on-a-Chip)芯片。这款芯片集成了一个Arm Cortex-M4F处理器和2.4GHz无线收发器,可支持蓝牙5.0和其他无线通信协议。 2. 开发板:nrf52840-dk是基于nrf52840芯片的开发板,具有丰富的外围设备和接口,方便开发者进行软件编程和硬件原型设计。该开发板包含了带有按钮、指示灯和天线的nRF52840 SoC模块,以及一系列GPIO引脚、USB接口、电源管理电路等。 3. 软件开发工具:使用nrf52840-dk进行开发时,可以使用Nordic Semiconductor提供的nRF5软件开发工具套件。这个工具套件包含了一系列用于编写和调试nRF52840软件的开发工具、库文件和示例代码。开发者可以使用该工具套件进行代码编辑、编译、下载和调试等操作。 4. 功能和应用:nrf52840-dk开发板支持蓝牙LE和其他无线通信协议,可以广泛应用于物联网设备、可穿戴设备、智能家居、健康监测、无线传感器网络等领域的开发。开发者可以利用该开发板进行传感器数据采集、通信协议开发和物联网应用的原型验证。 总之,nrf52840-dk开发板是一款功能强大、易于使用的开发工具,适用于物联网和蓝牙应用的开发。它提供了丰富的外围设备和接口,配合Nordic Semiconductor的软件开发工具套件,开发者可以快速进行硬件原型设计和软件编程。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值