在Windows CE.NET的開發上,目前采用微軟公司2005年年底推出的Visual Studio 2005是最理想的選擇。用 Visual Studio 2005可以進行利用C#,或VB.NET等語言開發基于.NET Compact Framework 2.0或1.0的 托管碼WinCE程序,也可以利用C++語言來開發基于MFC,ATL或Win32API的本機碼WinCE程序。
因為在微軟最新的.NET Compact Framework 2.0的類庫中還未包含針對藍牙通訊模塊的類庫,而且目前關于在Windows CE中開發藍牙通訊模塊應用程序的介紹還很少,同時開發藍牙通訊技術的應用需要十分廣泛,所以本文將就此進行一些討論。
1 基于托管碼開發藍牙通訊模塊
基于托管碼的開發就是使用一套運行時環境(run-time environment)的應用程序接口來開發。
一般情況下,托管碼應用程序的開發會比較簡單和快速,并且可跨軟件平臺和處理器來運行,所以開發出的托管碼也能重新使用并有較高的可移植性。
另外,內存管理、資源管理、資源收集、安全性管理等瑣碎工作都由運行時環境來處理。應用程序開發工程師不必費心處理。托管應用程序在目標機器上運行,是通過目標機器端的實時編譯器來實時把托管碼編譯成目標機器碼后在目標機器上執行。
由于在.NET平臺下,采用CLR(公共語言運行時)可以用不同的語言來調用.NET Compact Framework來開發相同功能的應用程序,所以本文托管碼部分僅采用C#語言為例來介紹藍牙通訊模塊開發。
1.1 利用P/Invoke方法編寫藍牙通訊模塊
藍牙通訊模塊是一個涉及到驅動硬件的應用程序開發,而.NET Compact Framework并不是一個對Win32API進行了完整封裝的類庫。所以在基于托管碼開發藍牙通訊模塊中必須利用到托管代碼如何與非托管代碼交互技術。P/Invoke全稱為Platform Invoke,是.NET開發平臺下允許托管代碼調用DLL庫的本地代碼函數的服務,類似于JA-VA中的GNI的概念。圖1說明了P/Invoke方法的工作原理。首先用相應語言的編譯器將托管的源代碼編譯成Assembly的形式,其中包括元數據和中間語言代碼。而此時P/Invoke的聲明會以元數據的形式存在于 Assembly中,當Assembly被CLR調用的時候,CLR會根據元數據的聲明在對應的DLL函數中查找DLL的實現。如果找到,就將其加載到內存中,并定位此DLL函數的人口點。將托管的參數人棧,并將函數的人口點指向對應的native dll,從而完成了托管代碼調用非托管代碼的DLL。
利用P/Invoke方法編寫藍牙通訊模塊,DllI-port屬性非常有用。下面的代碼將用例子說明此通用方案,例中托管程序將調用MessageBox(位于User32.lib中):
using
using namespace System:: Runtime::InteropSer-vices;
namespace SysWin32
{
[DllImport ( "user32. dll", EntryPoint = "MessageBox", CharSet = Unicode)]
int MessageBox(void * hWnd, wchar_t * lpText,wchar_t * lpCaption, unsigned int uType);
}
int main()
SysWin32 :: MessageBox(0, L" Hello world ! ", L"Greetings", 0)
}
注意包含DllImport的代碼行。此代碼行根據參數值通知編譯器,使之聲明位于User32.dll中的函數,并將簽名中出現的所有字符串(如參數或返回值)視為Unicode字符串。如果缺少EntryPoint參數,則默認值為函數名。另外,由于CharSet參數指定Unicode,因此公共語言運行庫將首先查找稱為MessageBoxW的函數。如果運行庫未找到此函數,它將根據調用約定查找MessageBox以及相應的修飾名。
當調用用戶定義的DLL中所包含的函數時,有必要將extern"C"添加在DLL函數聲明之前,如下所示:
extern"C"SAMPLEDLL_API int fnSam-pleDLL(void);
在調用非本機碼時,需要注意的是要將非結構化參數由托管封送處理為本機碼形式。可以利用CharSet參數值的作用,將參數中字符串(string*類型)都自動轉換為wchar_t*。同樣,所有Int32參數類型轉換為非托管int,UInt32參數類型轉換為非托管unsignedint,而 Intl6參數類型轉換為了short int。char*用于[in]參數的為String*(CharSet=Ansi),用于[out]參數或返回值的為Text::StringBuilder*。wchar-t*用于[in]參數為String*(CharSet=Unicode),用于 [out]參數或返回值的為Text::StringBuilder*。需要注意的是函數指針必須具有_stdcall調用約定,這是因為這是 DllImport支持的唯一類型。對于數組來說數組(如wchar_t*[ ]),CharSet參數僅應用于函數參數的根類型。因此,無論 CharSet的值是什么,String*_ _gc[ ]將被封送處理為wchar_t*[]。除簡單類型外,運行庫還提供了一種機制,可以將簡單結構由托管上下文封送處理為非托管上下文。簡單結構不包含任何內部數據成員指針、結構化類型的成員或其他元素。
在做一個關于藍牙通訊程序前,還需要一些關于藍牙的基礎知識。一個藍牙模塊程序需要包含開啟藍牙,配對,連接,建立串行通道,然后開啟通訊過程,還需要在應用程序中設置串行端口。因為藍牙技術有安全方面的設置,所以需要對藍牙設備進行配對。藍牙的工作狀態總共有3種,分別為開啟、關閉、可發現。并且所有的通訊設備都必須有一個對應的DeviceID,藍牙也不例外,藍牙的DeviceID是一串以“:”分隔的16進制的數字。有了上述知識,就可以在托管碼中利用P/Invoke方法開始編寫藍牙通訊模塊了。
對應的每一步需要調用的基本函數如下:
?獲取本地設備的ID
[DllImport ( "Btdrt. dll", SetLastError = true) ]
public static extern int BthReadLocalAddr (byte[]PBa)
?獲取遠程設備的ID
[DllImport( "ws2. dll", EntryPoint = "WSALook-upServiceBegin", SetLastError= true)]
public static extern int CeLookupServiceBegin(byte[ ] pQuerySet, LookupFlags dwFlags, ref intlphLookup)
?監聽服務
[DllImport (" ws2. dll", EntryPoint = "WSASetSer-vice", SetLastError= true)]
public static extern int CeSetService
(byte[ ] pQuerySet, RNRSERVICE_REGISTER,LookupFlags dwFlags)
?連接
[DllImport ( "mscoree", EntryPoint = "@ 339" )]
public static extern int connect (int s, byte []name, int namelen)
?藍牙的安全設置
獲取配對碼請求
[DllImport("Btdrt. dll", SetLastError= true)]
public static extern int BthGetPINRequest(byte[]pba)
設置配對碼
[DllImport( "btdrt. dll", SetLastError= true)
public static extern int BthSetPIN(byte[] pba, intcPinLength, byte [] ppin)
創建ACL連接:
[DllImport("Btdrt. dll", SetLastError= true)
public static extern int BthCreateACLConnection (byte[] pbt, ref ushort phandle);
然后是配對碼驗證:
[DllImport("Btdrt. dll", SetLastError= true)]
public static extern int BthAuthenticate (byte []pbt);
然后一定要關閉連接:
[DllImport("Btdrt. dll", SetLastError= true)]
public static extern int BthCloseConnection(ushorthandle);
評論
查看更多