欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

NRF51822蓝牙服务(8)——动态密码配对

程序员文章站 2024-03-24 23:40:34
...

前言

上篇我们学习了使用静态密码的方式在连接时进行**的验证,那么我要使用动态密码,进一步提高安全性呢?这里我们就在上篇代码的基础上尝试使用动态密码的方式完成蓝牙配对。

实验分析

由于,静态密码的时候已经把基本步骤详细讲述,这里就不再过多讲述。

由于开发板没有显示屏,所以这里采用串口打印的方式获取动态密码。

static void gap_params_init(void)
{
    uint32_t                err_code;
    ble_gap_conn_params_t   gap_conn_params;
    ble_gap_conn_sec_mode_t sec_mode;

    BLE_GAP_CONN_SEC_MODE_SET_OPEN(&sec_mode);
    
    err_code = sd_ble_gap_device_name_set(&sec_mode,
                                          (const uint8_t *) DEVICE_NAME,
                                          strlen(DEVICE_NAME));
    APP_ERROR_CHECK(err_code);

    memset(&gap_conn_params, 0, sizeof(gap_conn_params));

    gap_conn_params.min_conn_interval = MIN_CONN_INTERVAL;
    gap_conn_params.max_conn_interval = MAX_CONN_INTERVAL;
    gap_conn_params.slave_latency     = SLAVE_LATENCY;
    gap_conn_params.conn_sup_timeout  = CONN_SUP_TIMEOUT;

    err_code = sd_ble_gap_ppcp_set(&gap_conn_params);
    APP_ERROR_CHECK(err_code);
							
//把这里注释掉得到的就是动态密码									  
//    uint8_t passkey[] = STATIC_PASSKEY; 
//    m_static_pin_option.gap_opt.passkey.p_passkey = passkey;

    err_code=sd_ble_opt_set(BLE_GAP_OPT_PASSKEY,&m_static_pin_option);
    APP_ERROR_CHECK(err_code);
}

这里需要对gap_params_init()函数进行修改。

tatic void on_ble_evt(ble_evt_t * p_ble_evt)
{
    uint32_t                         err_code;
    
    switch (p_ble_evt->header.evt_id)
    {
        case BLE_GAP_EVT_CONNECTED:
            err_code = bsp_indication_set(BSP_INDICATE_CONNECTED);
            APP_ERROR_CHECK(err_code);
            m_conn_handle = p_ble_evt->evt.gap_evt.conn_handle;
			//一建立连接就发送安全请求,从而促使手机发送配对请求过来
			ble_gap_sec_params_t params;
			params.bond = SEC_PARAM_BOND;
			params.mitm = SEC_PARAM_MITM;
			sd_ble_gap_authenticate(m_conn_handle, &params);
			//
            break;
            
        case BLE_GAP_EVT_DISCONNECTED:
            err_code = bsp_indication_set(BSP_INDICATE_IDLE);
            APP_ERROR_CHECK(err_code);
            m_conn_handle = BLE_CONN_HANDLE_INVALID;
            break;

        case BLE_GAP_EVT_SEC_PARAMS_REQUEST:
        
				   resp_pair_request();
			
            break;
		
		//将密码串口发送出去
		case BLE_GAP_EVT_PASSKEY_DISPLAY:
			printf("show passkey:");
			for(int i = 0; i < 6; i++)
			{
				printf("%c",p_ble_evt->evt.gap_evt.params.passkey_display.passkey[i]);
			}
			break;
		//
		
		//判断配对是否成功,如果不成功就断开连接,从而阻止他人任意连接
		case BLE_GAP_EVT_AUTH_STATUS:
			if(p_ble_evt->evt.gap_evt.params.auth_status.auth_status == BLE_GAP_SEC_STATUS_SUCCESS)
			{
				printf("Success!");
			}
			else
			{
				sd_ble_gap_disconnect(m_conn_handle, BLE_HCI_REMOTE_USER_TERMINATED_CONNECTION);
			}
			break;
		//

        case BLE_GATTS_EVT_SYS_ATTR_MISSING:
            // No system attributes have been stored.
            err_code = sd_ble_gatts_sys_attr_set(m_conn_handle, NULL, 0, 0);
            APP_ERROR_CHECK(err_code);
            break;

        default:
            // No implementation needed.
            break;
    }
}

这里将resp_pair_request()函数添加到BLE_GAP_EVT_SEC_PARAMS_REQUEST事件处理部分中去。这样在手机请求配对时,设备就可以将自己的配对信息发送给手机了。

信息交换完后,设备底层的协议栈就会自动产生6位随机的密码,并将密码通过事件BLE_GAP_EVT_PASSKEY_DISPLAY上抛给应用层,然后就可以在应用层中将密码通过串口打印出来。

结果验证

  • 开发板串口连接PC,手机连接蓝牙,我们可以发现弹出密码输出窗口
  • 根据串口调试助手打印的密码输入,即可完成蓝牙配对
  • 断开蓝牙重连,发现仍然需要重新输入密码

总结

通过这个实验,我们学会了如何使用动态密码连接蓝牙。

相关标签: BLE