2008年1月7日星期一

蓝牙技术-设备查询及服务发现

正如标题所示,设备查询及服务发现是使用蓝牙技术的第一步工作,只有这步工作能够处理好,才可以接下来使用Bluetooth(蓝牙) for Windows DK API中其他的类,从而实现我们需要完成的功能,传输文件、语音发送、fax...

运行实现环境:Windows 2000 sp4 Bluetooth for Windows DK API Reference Guide(WIDCOMM) VC++6.0英文版

首先介绍开发蓝牙工程所在环境的一些必要设置:
确认安装了提供Bluetooth开发的相应SDK,本例是Bluetooth for Windows DK API Reference Guide,WIDCOMM公司提供的,版本1.4.2.10
(接下来的设置因sdk的不同而不同,本例设置可以参照上述dk的帮助说明有关Implementation部分)
接着介绍几个关键的类:
CBtIf 提供接口级函数,请求及服务发现使用该类
CSdpService管理sdp服务记录
CSdpDiscoveryRec获取sdp发现记录并提供查询方式

---------------------------------
由于蓝牙技术的实现类似socket,必须有服务器端及客户端部分。有所不同的是蓝牙的服务器一般来说可以支持多个客户端,而不像p2p中只能有一个。
服务器端和客户端都可以作为连接的发起者,因此作为设备查询和服务发现来说,服务器端和客户端的实现完全一致,所不同的是服务器端必须有相应的服务提供,而客户端则可以没有任何服务支持。因本文的重点放在如何实现设备查询及服务发现,因此暂不考虑如何在服务器端添加相应服务(具体实现是通过上述 CSdpService、CSdpDiscoveryRec)下面就来讲述sdp的具体实现。

具体来说实现主要都是通过类CBtIf完成的,下面有相应过程的简要分析:

首先,必须从CBtIf继承相应类的对象,这一步必须完成,否则无法使用sdk中的函数。
具体函数调用关系如下:
Client Server
CBtIf::StartInquiry()---------------------------------->
CBtIf::OnDeviceResponded()<-------------------------------
.
.
CBtIf::OnDeviceResponded()<-------------------------------
CBtIf::OnInquiryComplete()<--------------------------------- 设备查询阶段

CBtIf::StartDiscovery(address)---------------------->
CBtIf::OnDiscoveryComplete()<------------------------------ 服务发现阶段

CBtIf::ReadDiscoveryRecords()

上述的调用关系比较明确,有几点需要说明,首先提出任务可以从客户端发起,同样服务器端也可以发起,上图是客户端发起。
一方调用StartInquiry()函数后,一旦有设备回应,则OnDeviceResponded()函数就被唤起。值得注意的是这个回应可以是同一设备多次回应,也可以是不同设备的不同回应。(主要因为同一设备名字和其他属性都可以作为不同独立的回应)需要你在该函数中加入相应处理。
等OnInquiryComplete()函数被调用,表示设备查找完成,意味着可以进行服务发现。
然后可以调用StartDiscovery(),这个函数调用时必须附带一具体设备的地址,这个地址在上面的设备查询过程中可以得到。
另外最后一点就是当服务发现完成后,得到数据的一方是通过调用ReadDiscoveryRecords()函数来读取具体的服务内容。
(上述函数都有相应的参数,这里省略,具体可以查询相应sdk)
下面给一段具体代码,便于大家理解上述过程。
---------------------------------
void CBTDlg::OnSearch()
{
// TODO: Add your control notification handler code here
if (!StartInquiry()) //调用
SetDlgItemText(IDC_STATUS_TEXT,"ERROR--Unable to start device inquiry!");
}

void CBTDlg::OnDeviceResponded (BD_ADDR bda, DEV_CLASS devClass, BD_NAME bdName, BOOL bConnected) //对回应进行相应处理
{

// Add the device address and name (if any)
// to the Server List. It is OK to pass
// duplicates. This method will ignore them.
CString item_text;
int item_count = m_ServerList.GetItemCount(); //listview control

for(int x=0;x<item_count;x++)
{
CBdInfo* p_CmpInfo = (CBdInfo*)m_ServerList.GetItemData(x);
if (p_CmpInfo->isBdAddrEqual(bda))
{
if (p_CmpInfo->m_Name.GetLength() == 0)
{
p_CmpInfo->m_Name = bdName;
m_ServerList.SetItemText(x,0,p_CmpInfo->DeviceAsString());
}
return;
}
}
CBdInfo* p_Info = new CBdInfo(bda,bdName);

m_ServerList.InsertItem(item_count,p_Info->DeviceAsString());
m_ServerList.SetItemData(item_count,(LPARAM)p_Info);

}

void CBTDlg::OnDiscovery()
{
// TODO: Add your control notification handler code here
StopInquiry();
//通过新线程完成服务发现
m_pDiscoveryThread = AfxBeginThread(CBTDlg::DiscoverServices,this);
}

UINT CBTDlg::DiscoverServices(LPVOID pParam)
{
CBTDlg* p_Dlg = static_cast<CBTDlg*>(pParam);
CString strServiceName = "";
POSITION pos = p_Dlg->m_ServerList.GetFirstSelectedItemPosition();
if (pos!=NULL)
{
int nItem = p_Dlg->m_ServerList.GetNextSelectedItem(pos);
CBdInfo* p_Info = (CBdInfo*)p_Dlg->m_ServerList.GetItemData(nItem);

p_Dlg->SetDlgItemText(IDC_STATUS_TEXT,"Please Waiting for Services discovering...:"+p_Info->DeviceAsString());

if (!p_Dlg->StartDiscovery(p_Info->m_BdAddr,NULL)) //开始服务发现
{
p_Dlg->SetDlgItemText(IDC_STATUS_TEXT,"Discovery Failed!"+p_Info->DeviceAsString());
}
else
{
WaitForSingleObject(p_Dlg->m_hEventStopDiscoveryThread,30000);//等待完成到达

p_Dlg->SetDlgItemText(IDC_STATUS_TEXT,"Services on : "+p_Info->DeviceAsString());
p_Dlg->m_ServerList.DeleteAllItems();
p_Dlg->m_ServerList.DeleteColumn(0);
p_Dlg->m_ServerList.InsertColumn(0,"Service",LVCFMT_CENTER,180);

if (!p_Dlg->m_bDialogClosed)
{
CSdpDiscoveryRec sdp_Record[MAX_SERVICE];
int nServiceNumber = 0;
nServiceNumber = p_Dlg->ReadDiscoveryRecords(p_Info- >m_BdAddr,MAX_SERVICE,sdp_Record); //读取具体服务内容

if (nServiceNumber != 0)
{
for(int i=0;i<nServiceNumber;i++)
{
strServiceName = sdp_Record[i].m_service_name;
p_Dlg->m_ServerList.InsertItem(i,strServiceName);
}
}
else
p_Dlg->m_ServerList.InsertItem(0,"None!");

p_Dlg->m_ServerList.EnableWindow(FALSE);
}
else
SetEvent(p_Dlg->m_hEventKillDiscoveryThread);
}
}
p_Dlg->m_pDiscoveryThread = NULL;
return 0;
}

通过上面的介绍,只有掌握了蓝牙技术如何实现设备查询及服务发现后,才可以正确确认对方是否提供有相应的服务内容,从而为下一步使用其他传输或通讯技术打下基础。

没有评论: