如何访问UPnP设备的服务?

设备:Belkin Wemo Switch
开发环境:Windows 7上的MS VC ++ 2010

我想从Windows使用C ++枚举UPnP设备的服务。

我有IUPnPDevice指针,可以访问多个属性。
我有IUPnPServices指针,可以计算正确数量的服务(7)。
我使用QueryInterface()来获取IEnumVARIANT指针(它似乎成功)。
但是, Next()方法始终会失败, HRESULT0x80040500 – 转换为Windows error 1280 (0x500) - ERROR_ALREADY_FIBER
这个错误对我没有任何意义。

(我已经尝试使用IEnumVARIANTIEnumUnknown – 作为文档表明它可以是,但都产生相同的结果。)

我已经包含在完整的源文件下面,加上它产生的输出。
[注:这是硬编码使用我自己的设备的udn]

如果有人能够帮助,我会很感激,因为我目前被困住了。

最好的祝福,
戴夫

码:

 // UpnpTest1.cpp : Defines the entry point for the console application. // #include "stdafx.h" #include <windows.h> #include <upnp.h> static void DumpComError(const TCHAR *api, HRESULT hr); int _tmain(int argc, _TCHAR* argv[]) { int retcode=-1; // assume failure HRESULT hr = CoInitialize(0); if (hr==S_OK) { IUPnPDeviceFinder *deviceFinder=0; hr = CoCreateInstance(CLSID_UPnPDeviceFinder, 0, CLSCTX_INPROC_SERVER, IID_IUPnPDeviceFinder, (void**)&deviceFinder); if (hr==S_OK) { IUPnPDevice *device=0; hr = deviceFinder->FindByUDN(L"uuid:Socket-1_0-221239K11002F6", &device); if (hr==S_OK) { if (device) { TCHAR *manufacturer=0, *manufacturerUrl=0; TCHAR *description=0, *name=0, *modelUrl=0; TCHAR *serialNumber=0, *udn=0, *upc=0, *deviceType=0; TCHAR *presentationUrl=0; device->get_ManufacturerName(&manufacturer); device->get_ManufacturerURL(&manufacturerUrl); device->get_Description(&description); device->get_FriendlyName(&name); device->get_ModelURL(&modelUrl); device->get_SerialNumber(&serialNumber); device->get_UniqueDeviceName(&udn); device->get_UPC(&upc); device->get_Type(&deviceType); device->get_PresentationURL(&presentationUrl); _tprintf(_T("MANUFACTURER: %s [%s]\n"), manufacturer, manufacturerUrl); _tprintf(_T("MODEL: %s [%s]\n [%s]\n"), description, name, modelUrl); _tprintf(_T("DEVICE: serial=%s\n udn=%s\n upc=%s\n type=%s\n"), serialNumber, udn, upc, deviceType); _tprintf(_T("URL: %s\n"), presentationUrl); IUPnPServices *services=0; hr = device->get_Services(&services); if (hr==S_OK) { if (services) { long numberOfServices=0; services->get_Count(&numberOfServices); if (numberOfServices>0) { IUnknown *unknown=0; hr = services->get__NewEnum(&unknown); if (hr==S_OK) { if (unknown) { IEnumVARIANT *enumInterface=0; hr = unknown->QueryInterface(IID_IEnumVARIANT,(void**)&enumInterface); if (enumInterface) { VARIANT var; unsigned long fetched=0; hr = enumInterface->Next(1, &var, &fetched); if (hr==S_OK) { } else DumpComError(_T("IEnumVARIANT::Next"), hr); } else DumpComError(_T("IUnknown::QueryInterface"), hr); } else fprintf(stderr, "Failed to get enumeration interface.\n"); } else DumpComError(_T("IUPnPServices::get__NewEnum"), hr); } else fprintf(stderr, "No services available.\n"); } else fprintf(stderr, "Failed to get services collection.\n"); } else DumpComError(_T("IUPnPDevice::get_Services"), hr); } else fprintf(stderr, "Device not found.\n"); } else DumpComError(_T("IUPnPDeviceFinder::FindByUDN"), hr); } else DumpComError(_T("CoCreateIndex"), hr); } else DumpComError(_T("CoInitialize"), hr); return retcode; } static void AddBoolToString(const TCHAR *name, bool value, TCHAR *buf, int &i, int max) { if (name && *name && value && buf && i>=0) i += _snwprintf_s(&buf[i], max-i, (max-i-1)*sizeof(TCHAR), _T("%s%s=YES"), (i>0? _T("; "): _T("")), name); } static void AddIntToString(const TCHAR *name, int value, TCHAR *buf, int &i, int max) { if (name && *name && value && buf && i>=0) i += _snwprintf_s(&buf[i], max-i, (max-i-1)*sizeof(TCHAR), _T("%s%s=%d"), (i>0? _T("; "): _T("")), name, value); } static void DumpComError(const TCHAR *api, HRESULT hr) { bool failure = (hr&0x80000000? true: false); bool severe = (hr&0x40000000? true: false); bool microsoft = (hr&0x20000000? false: true); bool ntStatus = (hr&0x10000000? true: false); bool xBit = (hr&0x08000000? true: false); int facility = (hr&0x07FF0000)>>16; int code = (hr&0x0000FFFF); TCHAR buf[1024]={0}; int bufsize = sizeof(buf)/sizeof(TCHAR); int i=0; AddBoolToString(_T("failure"), failure, buf, i, bufsize); AddBoolToString(_T("severe"), severe, buf, i, bufsize); AddBoolToString(_T("microsoft"), microsoft, buf, i, bufsize); AddBoolToString(_T("ntStatus"), ntStatus, buf, i, bufsize); AddBoolToString(_T("xBit"), xBit, buf, i, bufsize); AddIntToString(_T("facility"), facility, buf, i, bufsize); AddIntToString(_T("code"), code, buf, i, bufsize); _ftprintf(stderr, _T("\n%s() failed, hr=0x%08x\n[%s]\n"), api, hr, buf); } 

输出:它产生以下输出:

 MANUFACTURER: Belkin International Inc. [http://www.belkin.com/] MODEL: Belkin Plugin Socket 1.0 [WeMo Switch] [http://www.belkin.com/plugin/] DEVICE: serial=221239K11002F6 udn=uuid:Socket-1_0-221239K11002F6 upc=123456789 type=urn:Belkin:device:controllee:1 URL: http://192.168.1.16:49153/pluginpres.html IEnumVARIANT::Next() failed, hr=0x80040500 [failure=YES; microsoft=YES; facility=4; code=1280] 

编辑:

在经历了许多死路线后,我设法通过手动构buildSOAP请求,并通过使用Windows套接字的TCP发送请求来实现这一目标。 棘手的一点是得到的语法恰到好处,因为我以前没有经验的SOAP。 [UPnP有助于识别IP地址和端口号,因为这些可以更改]。 一旦启动并运行 – 实际上比UPnP界面简单得多。 让我知道如果你有兴趣,我可以发布代码…它不直接回答我在这里提出的问题,所以用这个细节回答我的问题是没有意义的。

不过,如果您有兴趣,请告诉我,我可以发布代码。

干杯,
戴夫

0x80040500的HRESULT不是你想象的,而是UPNP_E_INVALID_DOCUMENT 。 为了解释这种歧义是如何可能的,请参阅另一个SO问题的答案 。

我的猜测是你的贝尔金设备给不符合设备的描述或服务描述的XML。 不符合不一定意味着格式不对,UPnP规范有大量的二级要求。 从英特尔开发人员工具(另一个答案的底部的链接)尝试设备间谍,如果设备弹出,然后从它的同一套件运行设备验证器。

我的经验是相似的,因为UPnPDeviceFinder根本不起作用。 它从不发送UPnP搜索数据包,所以设备不响应​​。 唯一能让它工作的方法是,如果你也使用windows media player或“Cast To Device”菜单(这是WMP)来启动搜索。 因为它是UPnPDeviceFinder将返回一些设备,只有当他们碰巧在那个时候广播,但即使找到一个XBox(另一个微软产品)不起作用的例子,没有其他活动进行。