富芮坤FR801xH蓝牙协议栈启动流程和notify实现温度数据主动上传

本文详细解析了基于FR801xH芯片的SDK中蓝牙协议栈的启动流程,包括设置设备名、安全参数、回调函数等,并介绍了如何实现蓝牙notify功能,用于数据的实时上传。在蓝牙设备与手机连接成功后,通过gatt_notification函数发送数据。最后,文章还进行了蓝牙调试工具的设置及温度上传的测试,验证了notify功能的正确性。
摘要由CSDN通过智能技术生成


  基于sdk开发的好处是方便项目的快速落地,现在用的是SDK里面的源文件,目前只是学习FR801xH芯片的特性,裁剪这个SDK需要点时间,在此之前,就用SDK来解释一下SDK的蓝牙协议栈的启动流程和notify实现方法。可以先复习一下 GATT的内容。

一、蓝牙协议栈启动流程

1.1 初始化代码

void simple_peripheral_init(void)
{
    // set local device name
	uint8_t local_name[] = "Simple Peripheral";
	gap_set_dev_name(local_name, sizeof(local_name));

	// Initialize security related settings.
	gap_security_param_t param =
	{
	    .mitm = false,
	    .ble_secure_conn = false,
	    .io_cap = GAP_IO_CAP_NO_INPUT_NO_OUTPUT,
	    .pair_init_mode = GAP_PAIRING_MODE_WAIT_FOR_REQ,
	    .bond_auth = true,
	    .password = 0,
	};

	gap_security_param_init(&param);

	gap_set_cb_func(app_gap_evt_cb);

	gap_bond_manager_init(BLE_BONDING_INFO_SAVE_ADDR, BLE_REMOTE_SERVICE_SAVE_ADDR, 8, true);
	gap_bond_manager_delete_all();

	mac_addr_t addr;
	gap_address_get(&addr);
	co_printf("Local BDADDR: 0x%2X%2X%2X%2X%2X%2X\r\n", addr.addr[0], addr.addr[1], addr.addr[2], addr.addr[3], addr.addr[4], addr.addr[5]);

	// Adding services to database
    sp_gatt_add_service();
	speaker_gatt_add_service();				    //创建Speaker profile,
    
	//按键初始化 PD6 PC5
	pmu_set_pin_pull(GPIO_PORT_D, (1<<GPIO_BIT_6), true);
	pmu_set_pin_pull(GPIO_PORT_C, (1<<GPIO_BIT_5), true);
	pmu_port_wakeup_func_set(GPIO_PD6|GPIO_PC5);
	button_init(GPIO_PD6|GPIO_PC5);

	demo_LCD_APP();							            //显示屏
	demo_CAPB18_APP();						            //气压计
	demo_SHT3x_APP();						            //温湿度
	gyro_dev_init();						            //加速度传感器
	
	//OS Timer
	os_timer_init(&timer_refresh,timer_refresh_fun,NULL);//创建一个周期性1s定时的系统定时器
	os_timer_start(&timer_refresh,1000,1);

}

1.2 初始化流程

  由上面的初始化函数可以看到如下的启动流程

  • gap_set_dev_name设置蓝牙设备的名称
  • gap_security_param_init设置蓝牙安全连接的参数
  • gap_set_cb_func注册GAP层的回调函数,GAP层产生的诸如连接成功、连接断开等事件都会进入这个回调函数里处理。
  • gap_bond_manager_init绑定初始化,绑定的意思是把配对产生的信息存储在flash里。
  • gap_address_get获取本地的mac地址
  • sp_gatt_add_service添加sp服务
  • speaker_gatt_add_service添加speaker服务

1.3 回调函数里的初始化

上面是明面上的初始化过程,下面要介绍的是通过回调实现的看不到的流程,大部分任务都由api完成,完成后会产生一个事件,根据事件执行不同的功能。

void app_gap_evt_cb(gap_event_t *p_event)
{
    switch(p_event->type)
    {
        case GAP_EVT_ADV_END://广播结束。示例:adv_end,status:0x00
        {
            co_printf("adv_end,status:0x%02x\r\n",p_event->param.adv_end.status);
            //gap_start_advertising(0);
        }
        break;
        
        case GAP_EVT_ALL_SVC_ADDED://所有的 service 都添加完毕。
        {
            co_printf("All service added\r\n");
            sp_start_adv();
#ifdef USER_MEM_API_ENABLE
            //show_mem_list();
            //show_msg_list();
            //show_ke_malloc();
#endif
        }
        break;

        case GAP_EVT_SLAVE_CONNECT://做为 slave 链接建立。示例:slave[0],connect. link_num:1
        {
					sp_conidx = p_event->param.slave_connect.conidx;
            co_printf("slave[%d],connect. link_num:%d\r\n",p_event->param.slave_connect.conidx,gap_get_connect_num());
			gatt_mtu_exchange_req(p_event->param.slave_connect.conidx);
            gap_conn_param_update(p_event->param.slave_connect.conidx, 6, 6, 0, 500);
        }
        break;

        case GAP_EVT_DISCONNECT://链接断开, 可能是 master 或 slave。示例:Link[0] disconnect,reason:0x13
        {
            co_printf("Link[%d] disconnect,reason:0x%02X\r\n",p_event->param.disconnect.conidx
                      ,p_event->param.disconnect.reason);
            sp_start_adv();
#ifdef USER_MEM_API_ENABLE
            show_mem_list();
            //show_msg_list();
            show_ke_malloc();
#endif
        }
        break;

        case GAP_EVT_LINK_PARAM_REJECT://链接参数更新被拒绝 示例:
            co_printf("Link[%d]param reject,status:0x%02x\r\n"
                      ,p_event->param.link_reject.conidx,p_event->param.link_reject.status);
            break;

        case GAP_EVT_LINK_PARAM_UPDATE://链接参数更新成功。 示例:Link[0]param update,interval:6,latency:0,timeout:500
            co_printf("Link[%d]param update,interval:%d,latency:%d,timeout:%d\r\n",p_event->param.link_update.conidx
                      ,p_event->param.link_update.con_interval,p_event->param.link_update.con_latency,p_event->param.link_update.sup_to);
            break;

        case GAP_EVT_PEER_FEATURE://收到对端的 feature 特性回复 示例:
            co_printf("peer[%d] feats ind\r\n",p_event->param.peer_feature.conidx);
            show_reg((uint8_t *)&(p_event->param.peer_feature.features),8,1);
            break;

        case GAP_EVT_MTU:
            co_printf("mtu update,conidx=%d,mtu=%d\r\n"
                      ,p_event->param.mtu.conidx,p_event->param.mtu.value);
            break;
        
        case GAP_EVT_LINK_RSSI:
            co_printf("link rssi %d\r\n",p_event->param.link_rssi);
            break;
                
        case GAP_SEC_EVT_SLAVE_ENCRYPT:
            co_printf("slave[%d]_encrypted\r\n",p_event->param.slave_encrypt_conidx);
            break;

        default:
            break;
    }
}

  服务添加完成后产生了一个GAP_EVT_ALL_SVC_ADDED事件,这个事件的条件下,执行sp_start_adv广播函数,连在一起就是在家添加完服务后开始蓝牙广播。
  到此,FR801xH的蓝牙协议栈的启动部分解释完毕,你可以在回调函数里去执行对应蓝牙事件下的操作,比如蓝牙连接成功、蓝牙断开等。

二、notify实现

2.1 notify介绍

  如果设备主动给手机发信息,则可以通过notification的方式,这种方式不用手机去轮询地读设备上的数据。notify的应用场景就是数据的实时上传。

2.2 notify实现

  实现notify需要的api函数是gatt_notification,使用方法

2.3.1 定义数据

uint8_t sp_conidx;//蓝牙连接号conidx
extern uint8_t sp_svc_id;//服务号

2.3.2 获取连接号conidx

在这里插入图片描述
  在蓝牙设备与手机连接成功后,在回调函数里获得连接号

		case GAP_EVT_SLAVE_CONNECT://做为 slave 链接建立。示例:slave[0],connect. link_num:1
        {
			sp_conidx = p_event->param.slave_connect.conidx;
            co_printf("slave[%d],connect. link_num:%d\r\n",p_event->param.slave_connect.conidx,gap_get_connect_num());
			gatt_mtu_exchange_req(p_event->param.slave_connect.conidx);
            gap_conn_param_update(p_event->param.slave_connect.conidx, 6, 6, 0, 500);
        }
        break;

  GAP_EVT_SLAVE_CONNECT:从机连接成功事件。

2.3.3 gatt_notification函数的使用

  在任意一个位置将入以下代码即可向指定的属性发送notify。这里是在获得温度后将发送温度。

	gatt_ntf_t ntf_att;
	ntf_att.att_idx = SP_IDX_CHAR4_VALUE;
	ntf_att.conidx = sp_conidx;
	ntf_att.svc_id = sp_svc_id;
	uint8_t show_string[30]={0};
	sprintf((char *)show_string,"temperature: %f",(float)temperature/1000);
	ntf_att.p_data = (uint8_t *)show_string;
	ntf_att.data_len = sizeof(show_string);
	gatt_notification(ntf_att);

  SP_IDX_CHAR4_VALUE:UUID为FFF4,接收notify的通道,在手机设置蓝牙接收用UUID时需要用到。

三、notify温度上传测试

3.1 蓝牙调试工具

本次使用的蓝牙调试工具是蓝牙调试器。
在这里插入图片描述

3.2 蓝牙调试器设置

  TX特征UUID与前面的SP_IDX_CHAR4_VALUE一致。
在这里插入图片描述

3.3 测试结果

  串口调试助手显示启动流程,与前面分析的启动流程一致。
在这里插入图片描述
在这里插入图片描述
  温度数据实时上传测试成功。

四、参考链接

BLE安全机制

  • 7
    点赞
  • 42
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
以下是一个简单的C# WinForms程序,实现BLE设备的搜索和数据读写功能。需要使用NuGet包管理器安装`Windows.Devices.Bluetooth`和`Windows.Devices.Bluetooth.GenericAttributeProfile`两个库。 ``` using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; using System.Drawing; using System.Linq; using System.Runtime.InteropServices.WindowsRuntime; using System.Text; using System.Threading.Tasks; using System.Windows.Forms; using Windows.Devices.Bluetooth; using Windows.Devices.Bluetooth.Advertisement; using Windows.Devices.Bluetooth.GenericAttributeProfile; namespace BLETest { public partial class Form1 : Form { private BluetoothLEAdvertisementWatcher _watcher; private GattCharacteristic _characteristic; public Form1() { InitializeComponent(); // 初始化BLE搜索器 _watcher = new BluetoothLEAdvertisementWatcher(); _watcher.Received += OnAdvertisementReceived; } private async void OnAdvertisementReceived(BluetoothLEAdvertisementWatcher sender, BluetoothLEAdvertisementReceivedEventArgs args) { // 获取广告中的设备名称 string deviceName = args.Advertisement.LocalName; // 如果设备名称包含“BLE设备”字样,则连接该设备 if (deviceName.Contains("BLE设备")) { // 停止搜索 _watcher.Stop(); // 连接设备 BluetoothLEDevice device = await BluetoothLEDevice.FromBluetoothAddressAsync(args.BluetoothAddress); // 获取设备的服务和特征 GattDeviceServicesResult result = await device.GetGattServicesAsync(); GattDeviceService service = result.Services.FirstOrDefault(s => s.Uuid == new Guid("0000FFF0-0000-1000-8000-00805F9B34FB")); GattCharacteristicsResult charsResult = await service.GetCharacteristicsAsync(); _characteristic = charsResult.Characteristics.FirstOrDefault(c => c.Uuid == new Guid("0000FFF1-0000-1000-8000-00805F9B34FB")); // 订阅特征的通知 GattCommunicationStatus status = await _characteristic.WriteClientCharacteristicConfigurationDescriptorAsync(GattClientCharacteristicConfigurationDescriptorValue.Notify); if (status == GattCommunicationStatus.Success) { _characteristic.ValueChanged += OnCharacteristicValueChanged; // 显示连接成功的提示信息 MessageBox.Show("连接成功!"); } else { // 显示连接失败的提示信息 MessageBox.Show("连接失败!"); } } } private async void OnCharacteristicValueChanged(GattCharacteristic sender, GattValueChangedEventArgs args) { // 读取设备发送的数据 byte[] buffer = args.CharacteristicValue.ToArray(); string message = Encoding.ASCII.GetString(buffer); // 将数据显示在文本框中 textBox1.Text = message; } private void btnSearch_Click(object sender, EventArgs e) { // 开始搜索BLE设备 _watcher.Start(); } private async void btnConnect_Click(object sender, EventArgs e) { // 发送数据到设备 byte[] buffer = Encoding.ASCII.GetBytes(textBox2.Text); GattCommunicationStatus status = await _characteristic.WriteValueAsync(buffer.AsBuffer()); if (status == GattCommunicationStatus.Success) { // 显示发送成功的提示信息 MessageBox.Show("发送成功!"); } else { // 显示发送失败的提示信息 MessageBox.Show("发送失败!"); } } } } ``` 在上面的代码中,我们使用`BluetoothLEAdvertisementWatcher`类进行BLE设备的搜索,使用`BluetoothLEDevice`类连接设备,使用`GattDeviceService`类获取设备的服务和特征,使用`GattCharacteristic`类订阅特征的通知,并使用`WriteValueAsync`方法向设备发送数据。当设备发送数据时,我们通过`ValueChanged`事件读取设备的数据,并将数据显示在文本框中。
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值