为什么我不能使用 UPnP 单播 M-SEARCH 而不是多播 M-SEARCH?

Why can't I get UPnP unicast M-SEARCH to work instead of MultiCast M-SEARCH?

早上好,

我们决定尽可能使用 UPnP。我们在 239.255.255.250:1900 上使用 MultiCast 进行 M-SEARCH。

但是,我们正在研究如何处理客户在其网络上锁定多播的情况。查看 UPnP 1.1 规范,它讨论了使用带有 M-SEARCH 的单播。所以,如果我们已经知道我们想要与之交谈的各种设备的 IP 地址,并且它们正在侦听 0.0.0.0:1900,我们认为我们可以向 [=37= 上的每个设备发送单播 M-SEARCH ].

我一直在努力做到这一点,并且费了好大劲才让设备接收并响应单播 M-SEARCH 请求。

首先,是否允许您与以单播 M-SEARCH 开始的设备进行首次 UPnP 对话?

其次,监听 0.0.0.0:1900 是否有某些原因不接受发送到 deviceIP:1900 的消息?

当我在我的机器上执行 netstat 查看正在使用的 IP 和端口时,似乎 239.255.255.250:1900 不在列表中,或者显示为 0.0.0.0:1900。

因此,如果 0.0.0.0 是 (ANY_IP),那么让一个侦听器侦听 0.0.0.0:1900 足以接收到 239.255.255.250:1900 的任何多播消息以及直接通过单播到该机器的 ip:1900?

测试时,我总是能够收到多播,但我从未收到 M-SEARCH 的单播。在执行 GET 等操作时,我能够与其他端口上的设备进行通信,但似乎我无法让端口 1900 响应单播 M-SEARCH。

你真的可以在同一台机器上同时以多播方式侦听 239.255.255.250:1900 并以单播方式侦听 0.0.0.0:1900 而不会发生 udp 套接字冲突吗?

如有任何建议和指点,我们将不胜感激。

谢谢, 柯蒂斯

PS:我使用的代码如下。对于构造函数中的地址,我们传入 IPAddress.Any(即 0.0.0.0),Protocol.Port 为 1900。这是 windows 机器上的 运行 Windows8.1:

//
// SsdpSocket.cs
//
// Author:
//   Aaron Bockover <abockover@novell.com>
//
// Copyright (C) 2008 Novell, Inc.
//
// Permission is hereby granted, free of charge, to any person obtaining
// a copy of this software and associated documentation files (the
// "Software"), to deal in the Software without restriction, including
// without limitation the rights to use, copy, modify, merge, publish,
// distribute, sublicense, and/or sell copies of the Software, and to
// permit persons to whom the Software is furnished to do so, subject to
// the following conditions:
//
// The above copyright notice and this permission notice shall be
// included in all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
// LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
// OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
//

using System;
using System.Net;
using System.Net.Sockets;

namespace Mono.Ssdp.Mono.Ssdp.Internal
{
    class SsdpSocket : Socket
    {
        static readonly IPEndPoint ssdp_send_point = new IPEndPoint (Protocol.IPAddress, Protocol.Port);

        readonly IPEndPoint ssdp_receive_point;

        public SsdpSocket (IPAddress address)
            : base (AddressFamily.InterNetwork, SocketType.Dgram, ProtocolType.Udp)
        {
            ssdp_receive_point = new IPEndPoint (address, Protocol.Port);
            SetSocketOption (SocketOptionLevel.Socket, SocketOptionName.ReuseAddress, true);
        }

        public IAsyncResult BeginSendTo (byte [] data, AsyncCallback callback)
        {
            return BeginSendTo (data, callback, ssdp_send_point);
        }

        public IAsyncResult BeginSendTo (byte[] data, AsyncCallback callback, IPEndPoint endPoint)
        {
            return BeginSendTo (data, 0, data.Length, SocketFlags.None, endPoint, callback, this);
        }

        public IAsyncResult BeginReceiveFrom (AsyncReceiveBuffer buffer, AsyncCallback callback)
        {
            return base.BeginReceiveFrom (buffer.Buffer, 0, buffer.Buffer.Length, SocketFlags.None, 
                ref buffer.SenderEndPoint, callback, buffer);
        }

        public void Bind ()
        {
            Bind (ssdp_receive_point);
        }
    }
}

此问题的修复是在单播消息的验证中。

这里有两个消息示例。第一个是多播,第二个是单播:

M-SEARCH * HTTP/1.1
HOST: 239.255.255.250:1900 
MAN: "ssdp:discover" 
MX: seconds to delay response 
ST: search target 
USER-AGENT: OS/version UPnP/1.1 product/version



M-SEARCH * HTTP/1.1
HOST: hostname:portNumber
MAN: "ssdp:discover"
ST: search target
USER-AGENT: OS/version UPnP/1.1 product/version

请注意,第二个 M-SEARCH 是单播搜索,不需要包含 'MX:' 行。我使用的代码需要 MX: 行并使用它的值。如果没有 MX: 行,我们就会得到一个异常,它被悄悄地隐藏起来并被 catch(exception){}

吞噬

无论如何,UPnPServer 只监听 0.0.0.0:1900 并且会听到所有多播和单播的 msearches。