扫描回应scan response 当我们使用lightblue软件或者CC254X的Central程序来扫描从机时,如果从机正在广播,将被扫描到并且可以看到从机的设备名,发射功率等信息,而这些数据并没有出现在广播数据中,这是为什么呢?下图是lightblue扫描到的信息:
当从机接收到主机发来的扫描请求时,会有一个扫描回应scan response ,这个response中,携带了从机的设备名,发射功率等信息
1、定义扫描数据
static uint8 scanRspData[] =
{
// complete name
6, // 第一段长度指的是从机名的长度+1,这个1是GAP_ADTYPE_LOCAL_NAME_COMPLETE这个宏
GAP_ADTYPE_LOCAL_NAME_COMPLETE,
'a','b','c','d','e',
// 连接时间范围 Min 和 Max 值的连接时间间隔
//(2 个八位字节 Min,2 个八位字节最大) (0xFFFF 表示没有 conn 间隔 min 或 max)
0x05, // length of this data
GAP_ADTYPE_SLAVE_CONN_INTERVAL_RANGE,
LO_UINT16( DEFAULT_DESIRED_MIN_CONN_INTERVAL ), // 100ms
HI_UINT16( DEFAULT_DESIRED_MIN_CONN_INTERVAL ),
LO_UINT16( DEFAULT_DESIRED_MAX_CONN_INTERVAL ), // 1s
HI_UINT16( DEFAULT_DESIRED_MAX_CONN_INTERVAL ),
//TX 电源级别: 0xXX:-127 到 + 127 dBm 发射功率
0x02, // length of this data
GAP_ADTYPE_POWER_LEVEL,
0 // 0dBm
};
2、定义广告数据
static uint8 advertData[] =
{
// 这将设置要使用有限可发现模式 (广告 30 秒的时间) 的设备而不是一般可发现模式 (无限期地做广告)
0x02, // length of this data
GAP_ADTYPE_FLAGS,//发现模式
DEFAULT_DISCOVERABLE_MODE | GAP_ADTYPE_FLAGS_BREDR_NOT_SUPPORTED, //不支持BR/EDR,CC2540是单模芯片,不支持BR/EDR
//服务的 UUID,通知中央设备什么服务包括在此外围设备
0x03, // length of this data
GAP_ADTYPE_16BIT_MORE, // s服务: 更多的 16 位 可用 Uuid 但不是全部
LO_UINT16( SIMPLEPROFILE_SERV_UUID ), //0xFFF0
HI_UINT16( SIMPLEPROFILE_SERV_UUID ),
};
一般从机为service,因此从机设备会有一个 attDeviceName 和 主机扫描时发现设备的名称
主机扫面到的设备名称 我们在scanRspData中的前几位定义格式为 长度+标记(判定数组表达的数据是什么意思比如设备名 间隔时间 发射功率等等)+设备名称
attDeviceName 我们可以单独定义,例如 static uint8 attDeviceName[GAP_DEVICE_NAME_LEN] = "Osama";
然后用 GGS_SetParameter( GGS_DEVICE_NAME_ATT, GAP_DEVICE_NAME_LEN, attDeviceName );//GAP GATT服务器参数设置 这一句修改的是service的名称
将数据加载到GATT层
3、设置 GAP Peripheral Role Profile
{
#if defined( CC2540_MINIDK )
// For the CC2540DK-MINI keyfob, device doesn't start advertising until button is pressed
uint8 initial_advertising_enable = FALSE;
#else
// For other hardware platforms, device starts advertising upon initialization
uint8 initial_advertising_enable = TRUE;
#endif
// By setting this to zero, the device will go into the waiting state after
// being discoverable for 30.72 second, and will not being advertising again
// until the enabler is set back to TRUE
uint16 gapRole_AdvertOffTime = 0;
uint8 enable_update_request = DEFAULT_ENABLE_UPDATE_REQUEST;
uint16 desired_min_interval = DEFAULT_DESIRED_MIN_CONN_INTERVAL;
uint16 desired_max_interval = DEFAULT_DESIRED_MAX_CONN_INTERVAL;
uint16 desired_slave_latency = DEFAULT_DESIRED_SLAVE_LATENCY;
uint16 desired_conn_timeout = DEFAULT_DESIRED_CONN_TIMEOUT;
// Set the GAP Role Parameters
GAPRole_SetParameter( GAPROLE_ADVERT_ENABLED, sizeof( uint8 ), &initial_advertising_enable );//广播使能
GAPRole_SetParameter( GAPROLE_ADVERT_OFF_TIME, sizeof( uint16 ), &gapRole_AdvertOffTime );//表示外设关闭广播持续时间,
//该值为零表示无限期关闭广播直到下一次
//广播使能信号到来。
GAPRole_SetParameter( GAPROLE_SCAN_RSP_DATA, sizeof ( scanRspData ), scanRspData ); //外设用于回复主机扫描请求的信息
GAPRole_SetParameter( GAPROLE_ADVERT_DATA, sizeof( advertData ), advertData ); //包含在广播里的信息
GAPRole_SetParameter( GAPROLE_PARAM_UPDATE_ENABLE, sizeof( uint8 ), &enable_update_request );//使能自动更新连接参数,可以让外设连接失败时自动调整连接
//参数以便重新连接。
GAPRole_SetParameter( GAPROLE_MIN_CONN_INTERVAL, sizeof( uint16 ), &desired_min_interval );//设置最小连接间隙,缺省值为80个单位(每单位1.25ms)
GAPRole_SetParameter( GAPROLE_MAX_CONN_INTERVAL, sizeof( uint16 ), &desired_max_interval );//设置最大连接间隙,缺省值为3200个单位
GAPRole_SetParameter( GAPROLE_SLAVE_LATENCY, sizeof( uint16 ), &desired_slave_latency );//外设鄙视参数,缺省为零。
GAPRole_SetParameter( GAPROLE_TIMEOUT_MULTIPLIER, sizeof( uint16 ), &desired_conn_timeout );//最大耐心等待时间,缺省为1000个单位
}
4、设置广告时间
{
uint16 advInt = DEFAULT_ADVERTISING_INTERVAL;
GAP_SetParamValue( TGAP_LIM_DISC_ADV_INT_MIN, advInt ); //最小的广告时间间隔,在有限的可发现模式时
GAP_SetParamValue( TGAP_LIM_DISC_ADV_INT_MAX, advInt ); //最大的广告时间间隔,在有限的可发现模式时
GAP_SetParamValue( TGAP_GEN_DISC_ADV_INT_MIN, advInt );//最小的广告时间间隔,在一般可发现模式时
GAP_SetParamValue( TGAP_GEN_DISC_ADV_INT_MAX, advInt );//最大的广告时间间隔,在一般可发现模式时
}
5、// Setup the GAP Bond Manager //GAP 绑定管理器设置
{
uint32 passkey = 0; // passkey "000000" //密钥
uint8 pairMode = GAPBOND_PAIRING_MODE_WAIT_FOR_REQ;//配对模式,配置成等待主机的配对请求
uint8 mitm = TRUE;
uint8 ioCap = GAPBOND_IO_CAP_DISPLAY_ONLY;//只显示设备
uint8 bonding = TRUE;
GAPBondMgr_SetParameter( GAPBOND_DEFAULT_PASSCODE, sizeof ( uint32 ), &passkey ); //密钥,范围是 0-999999,默认值为 0
GAPBondMgr_SetParameter( GAPBOND_PAIRING_MODE, sizeof ( uint8 ), &pairMode );//告诉绑定管理器是否配对通过,不论它等待一个请求从控制设备或者是自己发起配对.默认的设置是等待一个请求从控制设备. 配对模式:配置成等待主机的配对请求
GAPBondMgr_SetParameter( GAPBOND_MITM_PROTECTION, sizeof ( uint8 ), &mitm );//设置中间人保护是否使能.如果使能了,配对请求将鉴定连接在从和主之间.profile默认的值为FALSE,即使应用设置它为TRUE在初始化的时候. 打开密钥保护的配对算法
GAPBondMgr_SetParameter( GAPBOND_IO_CAPABILITIES, sizeof ( uint8 ), &ioCap );//告诉绑定管理器设备的输入和输出的能力.为了判断设备是否有显示屏或者输入键盘这个参数是需要的.然而,默认的值为GAPBOND_IO_CAP_DISPLAY_ONLY,表明设备有一个显示屏但没有键盘.即使设备没有物理意义上的显示器,一个展示的钥匙(在用户指导中)被认为是一个显示器.默认的万能钥匙是一个六位数字字符串”000000”.
GAPBondMgr_SetParameter( GAPBOND_BONDING_ENABLED, sizeof ( uint8 ), &bonding );//使能绑定.profile默认的值为FALSE,即使SimpleBLEPeripheral应用设置它为TRUE在初始化时.
}
6、GATT Server的相关设置函数。
// Initialize GATT attributes
GGS_AddService( GATT_ALL_SERVICES ); // GAP Service
GATTServApp_AddService( GATT_ALL_SERVICES ); // GATT attributes
DevInfo_AddService(); // Device Information Service
SimpleProfile_AddService( GATT_ALL_SERVICES ); // Simple GATT Profile
通常一个GATT中GAP server和GATT server是必须强制存在的(Mandatory)以及自己设计的profile server.
作为GATT的server和client,主要通过Attribute来进行交互,当client请求server读取数据时,通过如下注册的回调函数来进行访问。
// Register callback with SimpleGATTprofile
VOID SimpleProfile_RegisterAppCBs( &simpleBLEPeripheral_SimpleProfileCBs );//给应用注册回调函数
在回调函数中对时间做出处理。