2008年1月11日星期五

GCC

 使用语法:
  gcc [ option filename ]...
    g++ [ option filename ]...
  
  其中 option 为 gcc 使用时的选项(后面会再详述),而 filename 为欲以 gcc 处理的文件
  
  说明:
   这 C 与 C++ 的 compiler 已将产生新程序的相关程序整合起来。产生一个新的程序需要经过四个阶段:预处理、编译、汇编、连结,而这两个编译器都能将输入的文件做不同阶段的处理。虽 然原始程序的扩展名可用来分辨编写原始程序码所用的语言,但不同的compiler,其预设的处理程序却各不相同:
  
  gcc  预设经由预处理过(扩展名为.i)的文件为 C 语言,并於程式连结阶段以 C 的连结方式处理。
  
  g++  预设经由预处理过(扩展名为.i)的文件为 C++ 语言,并於程序连结阶段以 C++ 的连结方式处理。
  
  原始程序码的扩展名指出所用编写程序所用的语言,以及相对应的处理方法:
  
    .c  C 原始程序         ; 预处理、编译、汇编
    .C  C++ 原始程序        ; 预处理、编译、汇编
    .cc C++ 原始程序        ; 预处理、编译、汇编
    .cxx C++ 原始程序        ; 预处理、编译、汇编
    .m  Objective-C 原始程序    ; 预处理、编译、汇编
    .i  已经过预处理之 C 原始程序  ; 编译、汇编
    .ii 已经过预处理之 C++ 原始程序 ; 编译、汇编
    .s  组合语言原始程序      ; 汇编
    .S  组合语言原始程序      ; 预处理、汇编
    .h  预处理文件(标头文件)    ; (不常出现在指令行)
  
  其他扩展名的文件是由连结程序来处理,通常有:
    .o  Object file
    .a  Archive file
  
   除非编译过程出现错误,否则 "连结" 一定是产生一个新程序的最後阶段。然而你也可以以 -c、-s 或 -E 等选项,将整个过程自四个阶段中的其中一个停止。在连结阶段,所有与原始码相对应的.o 文件、程序库、和其他无法自文件名辨明属性的文件(包括不以 .o为扩展名的 object file 以及扩展名为 .a 的 archive file)都会交由连结程序来处理(在指令行将那些文件当作连结程序的参数传给连结程序)。
  
  选项:
  
  不同的选项必须分开来下:例如 `-dr' 这个选项就与 `-d -r' 大不相同。
  
  绝大部份的 `-f' 及 `-W' 选项都有正反两种形式:-fname 及-fno-name (或 -Wname 及 -Wno-name)。以下只列出非预设的那个形式。
  
  以下是所有选项的摘要。以形式来分类。选项的意义将另辟小节说明。
  
  一般性(概略、常用的)选项
         -c -S -E -o file -pipe -v -x language
  
    程序语言选项
         -ansi -fall-virtual -fcond-mismatch
         -fdollars-in-identifiers -fenum-int-equiv
         -fexternal-templates -fno-asm -fno-builtin
         -fno-strict-prototype -fsigned-bitfields
         -fsigned-char -fthis-is-variable
         -funsigned-bitfields -funsigned-char
         -fwritable-strings -traditional -traditional-cpp
         -trigraphs
  
    编译时的警告选项
         -fsyntax-only -pedantic -pedantic-errors -w -W
         -Wall -Waggregate-return -Wcast-align -Wcast-qual
         -Wchar-subscript -Wcomment -Wconversion
         -Wenum-clash -Werror -Wformat -Wid-clash-len
         -Wimplicit -Winline -Wmissing-prototypes
         -Wmissing-declarations -Wnested-externs -Wno-import
         -Wparentheses -Wpointer-arith -Wredundant-decls
         -Wreturn-type -Wshadow -Wstrict-prototypes -Wswitch
         -Wtemplate-debugging -Wtraditional -Wtrigraphs
         -Wuninitialized -Wunused -Wwrite-strings
  
    除错选项
         -a -dletters -fpretend-float -g -glevel -gcoff
         -gxcoff -gxcoff+ -gdwarf -gdwarf+ -gstabs -gstabs+
         -ggdb -p -pg -save-temps -print-file-name=library
         -print-libgcc-file-name -print-prog-name=program
  
   最佳化选项
         -fcaller-saves -fcse-follow-jumps -fcse-skip-blocks
         -fdelayed-branch -felide-constrUCtors
         -feXPensive-optimizations -ffast-math -ffloat-store
         -fforce-addr -fforce-mem -finline-functions
         -fkeep-inline-functions -fmemoize-lookups
         -fno-default-inline -fno-defer-pop
         -fno-function-cse -fno-inline -fno-peephole
         -fomit-frame-pointer -frerun-cse-after-loop
         -fschedule-insns -fschedule-insns2
         -fstrength-reduce -fthread-jumps -funroll-all-loops
         -funroll-loops -O -O2
  
    预处理选项
         -Aassertion -C -dD -dM -dN -Dmacro[=defn] -E -H
         -idirafter dir -include file -imacros file -iprefix
         file -iwithprefix dir -M -MD -MM -MMD -nostdinc -P
         -Umacro -undef
  
    汇编程序选项
         -Wa,option
    连结程序选项
         -llibrary -nostartfiles -nostdlib -static -shared
         -symbolic -Xlinker option -Wl,option -u symbol
  
    目录选项
         -Bprefix -Idir -I- -Ldir
  
   Target Options
         -b machine -V version
  
   与机器(平台)相关的选项
         M680x0 Options
         -m68000 -m68020 -m68020-40 -m68030 -m68040 -m68881
         -mbitfield -mc68000 -mc68020 -mfpa -mnobitfield
         -mrtd -mshort -msoft-float
  
  VAX Options
         -mg -mgnu -munix
  
  SPARC Options
         -mepilogue -mfpu -mhard-float -mno-fpu
         -mno-epilogue -msoft-float -msparclite -mv8
         -msupersparc -mcypress
  
  Convex Options
         -margcount -mc1 -mc2 -mnoargcount
  
  AMD29K Options
         -m29000 -m29050 -mbw -mdw -mkernel-registers
         -mlarge -mnbw -mnodw -msmall -mstack-check
         -muser-registers
  
  M88K Options
         -m88000 -m88100 -m88110 -mbig-pic
         -mcheck-zero-division -mhandle-large-shift
         -midentify-revision -mno-check-zero-division
         -mno-ocs-debug-info -mno-ocs-frame-position
         -mno-optimize-arg-area -mno-serialize-volatile
         -mno-underscores -mocs-debug-info
         -mocs-frame-position -moptimize-arg-area
         -mserialize-volatile -mshort-data-num -msvr3 -msvr4
         -mtrap-large-shift -muse-div-instruction
         -mversion-03.00 -mwarn-passed-structs
  
    RS6000 Options
         -mfp-in-toc -mno-fop-in-toc
  
  RT Options
         -mcall-lib-mul -mfp-arg-in-fpregs -mfp-arg-in-gregs
         -mfull-fp-blocks -mhc-struct-return -min-line-mul
         -mminimum-fp-blocks -mnohc-struct-return
  
  MIPS Options
         -mcpu=cpu type -mips2 -mips3 -mint64 -mlong64
         -mlonglong128 -mmips-as -mgas -mrnames -mno-rnames
         -mgpopt -mno-gpopt -mstats -mno-stats -mmemcpy
         -mno-memcpy -mno-mips-tfile -mmips-tfile
         -msoft-float -mhard-float -mabicalls -mno-abicalls
         -mhalf-pic -mno-half-pic -G num -nocpp
  
  i386 Options
         -m486 -mno-486 -msoft-float -mno-fp-ret-in-387
  
  HPPA Options
         -mpa-risc-1-0 -mpa-risc-1-1 -mkernel -mshared-libs
         -mno-shared-libs -mlong-calls -mdisable-fpregs
         -mdisable-indexing -mtrailing-colon
  
    i960 Options
         -mcpu-type -mnumerics -msoft-float
         -mleaf-procedures -mno-leaf-procedures -mtail-call
         -mno-tail-call -mcomplex-a

http://blog.csdn.net/visioncat/archive/2006/05/07/711693.aspx

http://blog.vckbase.com/bastet/archive/2005/04/26/4985.aspx
http://www.mingw.org/download.shtml
Best Regards
Miao Jia Hong

2008年1月8日星期二

使用WinSock搜索蓝牙设备

可以利用下面这些变成元素创建一个查询来搜索一定范围内的远程蓝牙设备:

* WSAQUERYSET 结构体
* WSALookupServiceBegin 函数
* WSALookupServiceNext函数
* WSALookupServiceEnd函数

注意 为了使清晰起见,文中忽略了错误处理
搜索并返回远程设备的地址

1. 提供Winsock的版本和实现细节的数据来初始化caller application。可以通过调用WSAStartup函数来获得这个数据。

WSADATA wsd;
WSAStartup (MAKEWORD(1,0), &wsd);

2. 创建并初始化一个WSAQUERYSET变量用于指定搜索参数,设置dwNameSpace成员为NS_BTH限制为查询蓝牙设备。

WSAQUERYSET wsaq;
ZeroMemory(&wsaq, sizeof(wsaq));
wsaq.dwSize = sizeof(wsaq);
wsaq.dwNameSpace = NS_BTH;
wsaq.lpcsaBuffer = NULL;

3. 调用WSALookupServiceBegin函数来执行一个查询。

int iRet = WSALookupServiceBegin (&wsaq, LUP_CONTAINERS, &hLookup);
将LUP_CONTAINERS赋给dwFlags参数,启动SDP来搜索蓝牙设备。
注意 将dwFlags参数设为零将执行一个服务搜索。
WSALookupServiceBegin函数返回一个句柄到hLookup参数中。

4. 要枚举在上一步中调用WSALookupServiceBegin所找到的设备,就要使用WSALookupServiceNex函数。函数返回一个指向存有查询结果的WSAQUERYSET及构体。
1. 设置一个WSAQUERYSET结构体来储存WSALookupServiceNext函数返回的设备数据。

CHAR buf[4096];
LPWSAQUERYSET pwsaResults = (LPWSAQUERYSET) buf;
ZeroMemory(pwsaResults, sizeof(WSAQUERYSET));
pwsaResults->dwSize = sizeof(WSAQUERYSET);
pwsaResults->dwNameSpace = NS_BTH;
pwsaResults->lpBlob = NULL;

2. 调用WSALookupServiceNext并将WSALookupServiceBegin返回的hLookUp参数传递给它。为了提高性能,对WSALookupServiceBegin 的调用只返回设备的地址并存储在内存中。要返回设备的名称和地址,就要将LUP_RETURN_NAME | LUP_RETURN_ADDR赋给dwFlags参数。

DWORD dwSize = sizeof(buf);
int iRet = WSALookupServiceNext (hLookup, LUP_RETURN_NAME | LUP_RETURN_ADDR, &dwSize, pwsaResults)
可以通过循环调用WSALookupServiceNext来枚举所有的设备。

5. 调用WSALookupServiceEnd函数来结束设备搜索。这个函数将释放由WSALookupServiceBegin创建的lookup句柄。

WSALookupServiceEnd(hLookup);

6. 要结束对Winsock服务的使用,调用WSACleanup函数。在程序中对每个成功调用的WSAStartup都必须对应地调用WSACleanup。

error LNK2001错误

在我的程序中,碰到如下错误:
nafxcwd.lib(thrdcore.obj) : error LNK2001: unresolved external symbol __endthreadex
nafxcwd.lib(thrdcore.obj) : error LNK2001: unresolved external symbol __beginthreadex
我的解决方法如下:
点击"Project|settings",选择C/C++标签,在CODE GENERATION分类中选择"Debug MultiThreaded"
再重新rebuild all后,问题可以得到解决.
可能原因是我在程序中用到
HINSTANCE hInstance = GetModuleHandle(NULL);
DialogBoxParam(hInstance, (LPCTSTR)IDD_QQTDlg, 0, (DLGPROC)QQDlg, 0);
这个是需要多线程的,因此会出现上面的问题。
经网上查阅资料,得到此类问题的常用解决办法。如下:
在创建MFC项目时, 不使用MFC AppWizard向导, 如果没有设置好项目参数, 就会在编译时产生很多连接错误, 如error LNK2001错误, 典型的错误提示有:
libcmtd.lib(crt0.obj) : error LNK2001: unresolved external symbol _main
LIBCD.lib(wincrt0.obj) : error LNK2001: unresolved external symbol _WinMain@16
msvcrtd.lib(crtexew.obj) : error LNK2001: unresolved external symbol _WinMain@16
nafxcwd.lib(thrdcore.obj) : error LNK2001: unresolved external symbol __beginthreadex
nafxcwd.lib(thrdcore.obj) : error LNK2001: unresolved external symbol __endthreadex
下面介绍解决的方法:
1. Windows子系统设置错误, 提示:
libcmtd.lib(crt0.obj) : error LNK2001: unresolved external symbol _main
Windows项目要使用Windows子系统, 而不是Console, 可以这样设置:
[Project] --> [Settings] --> 选择"Link"属性页,
在Project Options中将/subsystem:console改成/subsystem:windows
2. Console子系统设置错误, 提示:
LIBCD.lib(wincrt0.obj) : error LNK2001: unresolved external symbol _WinMain@16控制台项目要使用Console子系统, 而不是Windows, 设置:
[Project] --> [Settings] --> 选择"Link"属性页,
在Project Options中将/subsystem:windows改成/subsystem:console
3. 程序入口设置错误, 提示:
msvcrtd.lib(crtexew.obj) : error LNK2001: unresolved external symbol _WinMain@16
通常, MFC项目的程序入口函数是WinMain, 如果编译项目的Unicode版本, 程序入口必须改为wWinMainCRTStartup, 所以需要重新设置程序入口:
[Project] --> [Settings] --> 选择"C/C++"属性页,
在Category中选择Output,
再在Entry-point symbol中填入wWinMainCRTStartup, 即可
4. 线程运行时库设置错误, 提示:
nafxcwd.lib(thrdcore.obj) : error LNK2001: unresolved external symbol __beginthreadex
nafxcwd.lib(thrdcore.obj) : error LNK2001: unresolved external symbol __endthreadex
这是因为MFC要使用多线程时库, 需要更改设置:
[Project] --> [Settings] --> 选择"C/C++"属性页,
在Category中选择Code Generation,
再在Use run-time library中选择Debug Multithreaded或者multithreaded
其中,
Single-Threaded 单线程静态链接库(release版本)
Multithreaded 多线程静态链接库(release版本)
multithreaded DLL 多线程动态链接库(release版本)
Debug Single-Threaded 单线程静态链接库(debug版本)
Debug Multithreaded 多线程静态链接库(debug版本)
Debug Multithreaded DLL 多线程动态链接库(debug版本)
单线程: 不需要多线程调用时, 多用在DOS环境下
多线程: 可以并发运行
静态库: 直接将库与程序Link, 可以脱离MFC库运行
动态库: 需要相应的DLL动态库, 程序才能运行
release版本: 正式发布时使用
debug版本: 调试阶段使用

2008年1月7日星期一

手工编辑.inf文件 破解 WIDCOMM 蓝牙驱动的方法

对于使用 WidcommLicensePatcher 1.8 破解license 文件仍然无法找到蓝牙适配器的可以用以下方法完成破解 (转自魔蝎,并进行了局部修改)

widcomm 经过修改安装inf文件和lisence文件,达到了破解的目的,支持32位及64位操作系统,支持多种usb蓝牙设备,如下:
%ABOCOMDFU.DeviceDesc%=DFUBTUSB, USBVID_07B8&PID_B02D
%ABOCOMDFU.DeviceDesc%=DFUBTUSB, USBVID_07B8&PID_B02E
%ABOCOM.DeviceDesc%=SUSPENDUSB, USBVID_07B8&PID_B02C
%BRCMANYCOMUHE.DeviceDesc%=BRSMARTUSB, USBVID_0A5C&PID_2111
%BRCMANYCOM.DeviceDesc%=SUSPENDUSB, USBVID_0A5C&PID_2102
%ABOCOM.DeviceDesc%=SUSPENDUSB, USBVID_07B8&PID_B02B
%FORMOSA4.DeviceDesc%=SUSPENDUSB, USBVID_2001&PID_F118
%FORMOSA3.DeviceDesc%=SUSPENDUSB, USBVID_2001&PID_F117
%FORMOSA2.DeviceDesc%=SUSPENDUSB, USBVID_2001&PID_F116
%FORMOSA1.DeviceDesc%=SUSPENDUSB, USBVID_2001&PID_F111
%KENSINGTONEDR.DeviceDesc%=BRSMARTUSB, USBVID_047D&PID_105E
%BARCODA.DeviceDesc%=SUSPENDUSB, USBVID_174B&PID_0001
%SITECOM.DeviceDesc%=SUSPENDUSB, USBVID_3574&PID_21F6
%SITECOM.DeviceDesc%=SUSPENDUSB, USBVID_3574&PID_21F4
%LINKSYS.DeviceDesc%=SUSPENDUSB, USBVID_13B1&PID_0019
%ALPS.DeviceDesc%=BTWUSB, USBVID_044e&PID_3001 ; ALPS based VID&PID
%COMPAQ.DeviceDesc%=BTWUSB, USBVID_049F&PID_0027 ; COMPAQ CSR based VID&PID
%IBM.DeviceDesc%=BTWUSB, USBVID_04BF&PID_030A ; IBM CSR based VID&PID
%TDK.DeviceDesc%=BTWUSB, USBVID_04BF&PID_0309 ; TDK CSR based VID&PID
%IBMBDC.DeviceDesc%=BTWUSB, USBVID_04BF&PID_0310 ; IBM BDC CSR based VID&PID
%TDK.DeviceDesc%=BTWUSB, USBVID_04BF&PID_0319 ; TDK USB ADAPTER
%TDK.DeviceDesc%=BTWUSB, USBVID_04BF&PID_0320 ; TDK CSR BC2 based VID&PID
%IBM.DeviceDesc%=BTWUSB, USBVID_04BF&PID_0317 ; ULTRAPORT
%IBMBDC.DeviceDesc%=BTWUSB, USBVID_04BF&PID_0318 ; BDC
%ERICMN.DeviceDesc%=BTWUSB, USBVID_0BDB&PID_1000 ; Ericsson Business Mobile Networks BV VID&PID
%ERICMP.DeviceDesc%=BTWUSB, USBVID_0BDB&PID_1002 ; Ericsson Multipoint VID&PID
%CSR.DeviceDesc%=BTWUSB, USBVID_0A12&PID_0001 ; CSR VID&PID
%ERICSSONUSB.DeviceDesc%=BTWUSB, USBVID_08EA&PID_ABBA ; Ericsson VID&PID
%BROADCOM.DeviceDesc%=SUSPENDUSB, USBVID_0A5C&PID_200A ; Broadcom VID&PID
%BROADCOM.DeviceDesc%=BRSMARTUSB, USBVID_0A5C&PID_2009 ; Broadcom VID&PID Smart dongle
%BROADIBM.DeviceDesc%=SUSPENDUSB, USBVID_0A5C&PID_201E ; Broadcom-IBM VID&PID Smart dongle
%BROADCOM.DeviceDesc%=BRSMARTUSB, USBVID_0A5C&PID_2020 ; Broadcom VID&PID Unified UHE
%BROADCOM.DeviceDesc%=SUSPENDUSB, USBVID_0A5C&PID_2045 ; Broadcom 2045 VID&PID
%BROADIBMEDR.DeviceDesc%=SUSPENDUSB, USBVID_0A5C&PID_2110 ; Broadcom IBM VID&PID
%BRCM2045UHE.DeviceDesc%=BRSMARTUSB, USBVID_0A5C&PID_2100 ; Broadcom 2045 VID&PID, UHE
%BRCM2045.DeviceDesc%=SUSPENDUSB, USBVID_0A5C&PID_2101 ; Broadcom 2045 VID&PID, non UHE
%SAMSUNG.DeviceDesc%=BTWUSB, USBVID_055D&PID_0BB1 ; SAMSUNG BC02 external
%SIWAVE.DeviceDesc%=BTWUSB, USBVID_0C10&PID_0000 ; Silicon Wave VID&PID
%SIWAVE.DeviceDesc%=BTWUSB, USBVID_05B1&PID_1389 ; Silicon Wave VID&PID
%ZEEVO.DeviceDesc%=BTWUSB, USBVID_0B7A&PID_07D0 ; Zeevo VID&PID
%DELL.DeviceDesc%=SUSPENDUSB, USBVID_413C&PID_8000 ; Dell BC02 VID&PID
%IBMBM.DeviceDesc%=BTWUSB, USBVID_1668&PID_0441 ; IBM integrated BT Modem VID&PID
%IBMBM3.DeviceDesc%=SUSPENDUSB, USBVID_1668&PID_2441 ; IBM Integrated Bluetooth III VID&PID
%MICROTUNE.DeviceDesc%=BTWUSB, USBVID_0F4D&PID_1000 ; Microtune USB Bluetooth Device VID&PID
%MSI.DeviceDesc%=BTWUSB, USBVID_0DB0&PID_1967 ; MSI USB Device VID&PID
%MSI.DeviceDesc%=BTWUSB, USBVID_0DB0&PID_6970 ; MSI USB Device VID&PID
%MSI.DeviceDesc%=BTWUSB, USBVID_0DB0&PID_697A ; MSI USB Device VID&PID
%MSI.DeviceDesc%=BTWUSB, USBVID_0DB0&PID_6855 ; MSI USB Device VID&PID
%MSFT.DeviceDesc%=BTWUSB, USBVID_045E&PID_007E ; Microsoft
%NSCEVAL.DeviceDesc%=BTWUSB, USBVID_0400&PID_0807 ; National Semi Bluetooth board
%STM.DeviceDesc%=WDSMARTUSB, USBVID_0483&PID_5000 ; STM VID&PID
%TI.DeviceDesc%=WDSMARTUSB, USBVID_0451&PID_1234 ; TI VID&PID
%TI.DeviceDesc%=BTWUSB, USBVID_0451&PID_ffff ; TI VID&PID
%SONY.DeviceDesc%=BTWUSB, USBVID_044e&PID_2014 ; SONY VID&PID
%FORMOSA.DeviceDesc%=BTWUSB, USBVID_0D9A&PID_0001 ; FORMOSA VID&PID
%FORMOSA.DeviceDesc%=BTWUSB, USBVID_4851&PID_1103 ; FORMOSA VID&PID
%BLUETHUMB.DeviceDesc%=SUSP1USB, USBVID_049F&PID_0086 ; BLUETHUMB USB Device VID&PID
%LOGITECH.DeviceDesc%=BTWUSB, USBVID_046D&PID_C707 ; LOGITECH VID&PID
%LOGITECH.DeviceDesc%=BTWUSB, USBVID_046D&PID_C708 ; LOGITECH VID&PID
%LOGITECH.DeviceDesc%=BTWUSB, USBVID_046D&PID_C709 ; LOGITECH VID&PID
%LOGITECH.DeviceDesc%=BTWUSB, USBVID_046D&PID_C70D ; LOGITECH VID&PID
%GIANT.DeviceDesc%=BTWUSB, USBVID_0FD1&PID_0001 ; Giant VID&PID
%BELKIN.DeviceDesc%=SUSPENDUSB, USBVID_050D&PID_0083
%PANASONIC.DeviceDesc%=BTWUSB, USBVID_04AD&PID_2501 ; Panasoic VId&PID
%ICSI.DeviceDesc%=BTWUSB, USBVID_0547&PID_0001 ; ICSI USB Device VID&PID
%PHILIPS.DeviceDesc%=BTWUSB, USBVID_0471&PID_0809 ; Philips BluePearl VID&PID
%ITRONIX.DeviceDesc%=BTWUSB, USBVID_11D9&PID_2600 ; Itronix Bluetooth Device VId&PID
%DELLELWOOD.DeviceDesc%=BRSMARTUSB, USBVID_413C&PID_8501 ; Dell Elwood VId&PID
%BROADCOMHP.DeviceDesc%=SUSP1USB, USBVID_03F0&PID_011D ; BROADCOM VId&PID for Hp
%BROADCOM2045HP.DeviceDesc%=SUSP1USB, USBVID_03F0&PID_171D ; BROADCOM VId&PID for Hp
%BRCMHPPRINT.DeviceDesc%=SUSPENDUSB, USBVID_03F0&PID_D104 ; BROADCOM VId&PID for Hp
%BCHPTransceiver.DeviceDesc%=BRSMARTUSB, USBVID_03F0&PID_0624 ; BROADCOM VId&PID for Hp
%ISSC.DeviceDesc%=BTWUSB, USBVID_1131&PID_1001 ; ISSC USB Device Vid&PID
%AIR2U.DeviceDesc%=BTWUSB, USBVID_1310&PID_0001 ; Air2U Device Vid&Pid
%AIR2UBRCM.DeviceDesc%=SUSPENDUSB, USBVID_1310&PID_0002 ; Air2U(BCM) Device Vid&Pid
%FCS.DeviceDesc%=BTWUSB, USBVID_0BF8&PID_1003 ; Fujitsu Siemens Vid&Pid
%ACTIONTEC.DeviceDesc%=BTWUSB, USBVID_1668&PID_0500 ; Actiontec USB Bluetooth Device BTM200
%TAIYOYUDEN.DeviceDesc%=BTWUSB, USBVID_0C24&PID_0001 ; TAIYO YUDEN USB Device VID&PID
%TAIYOYUDEN2.DeviceDesc%=BTWUSB, USBVID_0C24&PID_0002 ; TAIYO YUDEN USB Device VID&PID
%SAMSUNGTY.DeviceDesc%=BTWUSB, USBVID_04E8&PID_7021 ; Samsung TY USB Device VID&PID
%MICROLINK.DeviceDesc%=BTWUSB, USBVID_148D&PID_2430 ; MICROLINK USB Device VID&PID
%DELLU2.DeviceDesc%=SUSPENDUSB, USBVID_413C&PID_8103 ; DELLCSR's U2 module VID&PID
%DELLPHISH.DeviceDesc%=SUSPENDUSB, USBVID_413C&PID_8110 ; DELL PHISH module VID&PID
%TECOM.DeviceDesc%=SUSPENDUSB, USBVID_041E&PID_5015 ; TECOM USB PID&VID
%KENSINGTON.DeviceDesc%=BRSMARTUSB, USBVID_047D&PID_105D ; Kensington USB PID&VID
%TECOM.DeviceDesc%=BRSMARTUSB, USBVID_0D62&PID_2026 ; TECOM Darfon USB PID&VID
%BRCMMOTO.DeviceDesc%=SUSPENDUSB, USBVID_22B8&PID_0850

在这儿顺便普及一下windows对usb设备的识别,windows将通过VID及PID来识别usb设备。
VID指Vendor ID(厂家ID);PID指Product ID(产品ID);
因此只要在安装文件btwusb.inf里面列明所有的VID和PID,理论上就支持全部的蓝牙适配器,上面就是这个版本的安装脚本,你可以找你的蓝牙适配器是否在里面,比方说我的是ISSC的芯片,就在:
%ISSC.DeviceDesc%=BTWUSB, USBVID_1131&PID_1001 ; ISSC USB Device Vid&PID
这儿指明了。

如果此安装脚本里面没有你的蓝牙适配器,你仅仅需要在btwusb.inf文件里面加入你的VID和PID就可以了,照着上面的格式。

下一个问题,怎么知道我的usb dongle的VID和PID呢?很简单,你还没装驱动时:
在桌面上右键单击我的电脑->属性->硬件->设备管理器->蓝牙设备 下面会有一个未知设备->双击查看属性->详细信息->在下拉框选 设备范例ID->下面会显示大概如下:
USBVID_1131&PID_1001&2593D763&0&1
的字串,其中的VID_1131&PID_1001就是你usb dongle的VID和PID了。

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

正如标题所示,设备查询及服务发现是使用蓝牙技术的第一步工作,只有这步工作能够处理好,才可以接下来使用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;
}

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