如何访问UPnP设备的服务



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

我正在尝试从Windows中使用C++枚举UPnP设备的服务。

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

(我尝试过同时使用IEnumVARIANTIEnumUnknown——正如文档所示,两者都可以,但都会产生相同的结果。)

我在下面包含了完整的源文件,以及它产生的输出
[注意:使用我自己设备的udn是硬编码的]

如果有人能帮助我,我将不胜感激,因为我目前正处于困境。

尊敬的Dave

代码:

// 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=%sn              udn=%sn              upc=%sn              type=%sn"), serialNumber, udn, upc, deviceType);
_tprintf(_T("URL:          %sn"), 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%08xn[%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]

编辑:

在经历了很多死胡同之后,我通过手动构建SOAP请求,并使用Windows套接字通过TCP发送请求,成功地实现了这一点。技巧在于语法正确,因为我以前没有SOAP的经验。UPnP有助于识别IP地址和端口号,因为它们可能会更改。一旦启动并运行,它实际上比UPnP接口简单得多。如果你感兴趣,请告诉我,我可以发布代码。。。它没有直接回答我在这里提出的问题,所以用这个细节回答我的问题是没有意义的。

但是,如果你感兴趣,请告诉我,我可以发布代码。

干杯,
Dave

0x80040500的HRESULT不是您所想的,而是UPNP_E_INVALID_DOCUMENT。关于这种歧义的解释,请参阅我在另一个SO问题中的回答。

我的猜测是,您的Belkin设备给出了不一致的设备描述或服务描述XML。不符合并不一定意味着格式不完整,UPnP规范有很多次要要求。尝试"英特尔开发人员工具"中的"设备间谍"(另一个答案底部的链接),如果设备弹出,则从其上的同一套件运行"设备验证程序"。

我的经验类似,UPnPDeviceFinder根本不起作用。它从不发送UPnP搜索数据包,因此设备不会做出响应。让它工作的唯一方法是,如果你还使用windows媒体播放器或"播放到设备"菜单(即WMP)来启动搜索。事实上,UPnPDeviceFinder将返回一些设备,只有当它们碰巧在那一刻进行广播时,但如果没有其他活动,即使找到XBox(另一种微软产品)在示例中也不起作用。

最新更新