对于研发和测试BLE来说,经常看到同名的设备,是极为不方便的,一大堆设备同时上电会让同事不知道哪一个设备才是自己真正想操作的目标。再说一下小米手环,家中有三支小米手环,打开设备搜索全是“MI”,都不知道连接哪一个,所以在开始使用的时候才要求用户去敲手环几下,当然这个体验也还算不错。但不管怎样,作为开发者,面对Office里一大堆的BLE设备,能够方便区分还是不错的。因此萌生让设备名称包含一个唯一的标识——MAC地址。
第一步先说如何更改设备名称
BLE中要更改名称是极为方便的,只需要更改扫描应答数据和GAP GATT NAME属性。以TI BLE STACK 1.4.0中的simpleBLEPeripheral工程为例来说,
修改的地方是:
simpleBLEPeripheral.c
static uint8 scanRspData[];
static uint8 advertData[];
第二步再说如何生成一个包含设备MAC地址的名称
每颗BLE 的芯片都会有一个6字节的MAC地址,读到它并转化成ASCII字符,比如0x12 34 56 78 9a bc转换成“123456789ABC”,如果MAC地址全部写入设备名称,则显得太长,取后2字节地址,即字串“ABCD”即可。将设备名称的前辍连同地址字串写入扫描应答数据和GAP GATT NAME属性,设备运行时,我们再扫描就可以看到它的名称已经变了。
第三步来说实现
程序可以根据MAC地址生成一个自已唯一的名称,并在利用初始程序来修改步骤一提到的两处即可。下文的代码是生成动态名称的参考。在simpleBLEPeripheral.c文档的SimpleBLEPeripheral_Init()函数调用bleTask_BlePara()函数,同时注释掉该函数中以下调用。
//GAPRole_SetParameter( GAPROLE_SCAN_RSP_DATA, sizeof ( scanRspData ), scanRspData );
//GAPRole_SetParameter( GAPROLE_ADVERT_DATA, sizeof( advertData ), advertData );
// Set the GAP Characteristics
//GGS_SetParameter( GGS_DEVICE_NAME_ATT, GAP_DEVICE_NAME_LEN, attDeviceName );
参考代码:
/**************************************************************************************************
Filename: simpleBLEOTAPatch.c
Editor: Tome @ Newbit Studio
Revised: $Date: 2015/8/11 11:20:02 +0800 $
Revision: $Revision: 00001 $
Description:
History:
Notes:
**************************************************************************************************/
/**************************************************************************************************
// INCLUDES
**************************************************************************************************/
#include "bcomdef.h"
#include "OSAL.h"
#include "gatt.h"
#include "gatt_profile_uuid.h"
#include "hci.h"
#include "gap.h"
#include "gapgattserver.h"
#include "gattservapp.h"
#include "peripheral.h"
#include "osal_snv.h"
#include "simpleBLEOTAPatch.h"
/**************************************************************************************************
// TYPEDEF
**************************************************************************************************/
/**************************************************************************************************
// CONSTANTS
**************************************************************************************************/
#define DEVICE_NAME "NB BOOT V3 ----"
#define DEVICE_NAME_WITH_HEADER "xNB BOOT V3 ----*----*----*---"
#define DEVICE_NAME_PLACEHOLDER_IDX 0 //x is placeholder
/**************************************************************************************************
// GLOBAL VARIABLES
**************************************************************************************************/
/**************************************************************************************************
// LOCAL VERIABLE
**************************************************************************************************/
// GAP - SCAN RSP data (max size = 31 bytes)
static uint8 scanRspData[31];
// GAP - Advertisement data (max size = 31 bytes, though this is
// best kept short to conserve power while advertisting)
static uint8 advertData[31];
/**************************************************************************************************
// FUNCTIONS DECLERATION
**************************************************************************************************/
/**************************************************************************************************
// FUNCTIONS
**************************************************************************************************/
/**************************************************************************************************
* @fn BleTask_ScanRspDataInit
*
* @brief
*
* @param uint8 *dat
*
* @return int
**************************************************************************************************/
int BleTask_ScanRspDataInit(uint8 *scanrsp)
{
uint8 dat[5] = {
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 ),
};
int len =
utilSerialization(scanrsp, 0, 5, dat);
dat[0] = GAP_ADTYPE_POWER_LEVEL;
dat[1] = 0;
len = utilSerialization(scanrsp, len, 2, dat);
return len;
}
/**************************************************************************************************
* @fn bleTask_BlePara
*
* @brief
*
* @param void
*
* @return void
**************************************************************************************************/
void bleTask_BlePara(void)
{
int len = 0;
len = BleTask_ScanRspDataInit(scanRspData);
uint8 nvName[20];
uint8 rearNameLen = 4;
uint8 *pName;
if ( SUCCESS == osal_snv_read(NVID_USER_REARNAME, len, nvName) )
{
pName = nvName;
rearNameLen = osal_strlen((char *)nvName);
}
else
{
uint8 mac[6] = {1,2,3,4,5,6};
mac[5]=*(unsigned char *)(0x780E);
mac[4]=*(unsigned char *)(0x780F);
mac[3]=*(unsigned char *)(0x7810);
mac[2]=*(unsigned char *)(0x7811);
mac[1]=*(unsigned char *)(0x7812);
mac[0]=*(unsigned char *)(0x7813);
utilPrintMacAddress(mac, (char*)nvName);
pName = nvName+8;
rearNameLen = 4;
}
uint8 name[] = DEVICE_NAME_WITH_HEADER;
uint8* pFill = utilSearchChar2( name, ‘-‘);
uint8* pEnd = osal_memcpy(pFill, pName, rearNameLen);
uint8 length = pEnd-name;
// uint8 length = sizeof(name);
// name[length-5] = macString[8];
// name[length-4] = macString[9];
// name[length-3] = macString[10];
// name[length-2] = macString[11];
name[DEVICE_NAME_PLACEHOLDER_IDX] = GAP_ADTYPE_LOCAL_NAME_COMPLETE;
len = utilSerialization(scanRspData, len, length, name);
// should remove the placeholder! and decrease the length
GGS_SetParameter( GGS_DEVICE_NAME_ATT, length-1, name+1 );
uint8 dat[10];
// advertise data
dat[0] = GAP_ADTYPE_FLAGS;
dat[1] = DEFAULT_DISCOVERABLE_MODE | GAP_ADTYPE_FLAGS_BREDR_NOT_SUPPORTED;
len = utilSerialization(advertData, 0, 2, dat);
dat[0] = GAP_ADTYPE_16BIT_MORE;// some of the UUID‘s, but not all
dat[1] = LO_UINT16( DEVINFO_SERV_UUID );
dat[2] = HI_UINT16( DEVINFO_SERV_UUID );
dat[3] = LO_UINT16( 0xFEF9 );
dat[4] = HI_UINT16( 0xFEF9 );
len = utilSerialization(advertData, len , 5, dat);
GAPRole_SetParameter( GAPROLE_SCAN_RSP_DATA, sizeof ( scanRspData ), scanRspData );
GAPRole_SetParameter( GAPROLE_ADVERT_DATA, sizeof( advertData ), advertData );
}
/**************************************************************************************************
Copyright 2015 Newbit Studio All rights reserved.
**************************************************************************************************/