6

デバイス: Belkin Wemo Switch
開発環境: Windows7 上の MS VC++ 2010

Windows から C++ を使用して UPnP デバイスのサービスを列挙しようとしています。

IUPnPDeviceポインターを取得し、いくつかのプロパティにアクセスできます。
ポインターを取得しIUPnPServices、正しいサービス数 (7) を数えることができます。
私はポインターQueryInterface()を取得するために使用します(成功したようです)。 ただし、このメソッドは常にof で失敗します。これは として変換されます。 このエラーは私には意味がありません。IEnumVARIANT
Next()HRESULT0x80040500Windows error 1280 (0x500) - ERROR_ALREADY_FIBER

(私は両方IEnumVARIANTを使用してみましIEnumUnknownた - ドキュメントはどちらかである可能性があることを示していますが、両方とも同じ結果を生成します。)

完全なソース ファイルとそれが生成する出力を以下に示します。
[注: 自分のデバイスの 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]

編集:

多くの行き詰まりの後、SOAP 要求を手動で作成し、Windows ソケットを使用して TCP 経由で要求を送信することで、これを機能させることができました。これまでSOAPの経験がなかったので、構文を正しく理解するのは難しいことでした。[UPnP は、IP アドレスとポート番号を特定するのに役立ちました - これらは変更される可能性があります]。起動して実行すると、UPnP インターフェイスよりもはるかに簡単になります。興味があるかどうか教えてください。コードを投稿できます...ここで提起した質問に直接答えるものではないため、この詳細で私の質問に答えるのは意味がありません.

ただし、興味がある場合はお知らせください。コードを投稿できます。

乾杯、
デイブ

4

2 に答える 2

2

0x80040500 の HRESULT はあなたが考えているものではなく、UPNP_E_INVALID_DOCUMENTです。このようなあいまいさがどのように可能であるかの説明については、別の SO の質問で私の回答を参照してください。

私の推測では、あなたの Belkin デバイスは準拠していないデバイス記述またはサービス記述 XML を提供していると思われます。非準拠は必ずしも形式が壊れていることを意味するわけではなく、UPnP 仕様には大量の二次的な要件があります。デバイスがポップアップする場合は、Intel Developer Tools (他の回答の下部にあるリンク) から Device Spy を試してから、同じスイートから Device Validator を実行します。

于 2012-12-11T23:27:25.533 に答える
0

UPnPDeviceFinder が機能しないという点で、私の経験も同様です。UPnP 検索パケットを送信しないため、デバイスは応答しません。それを機能させる唯一の方法は、Windows Media Player または "Cast To Device" メニュー (WMP) を使用して検索を開始することです。UPnPDeviceFinder は、その時点でたまたまブロードキャストしている場合にのみ、いくつかのデバイスを返しますが、XBox (別の Microsoft 製品) を見つけることでさえ、他のアクティビティが進行していなければ、例では機能しません。

于 2016-12-31T19:21:22.040 に答える