c# - Hyper-V 机器上的套接字异常

标签 c# udp winsock multicast hyper-v

最近,我们的整个 QA 环境从 VMWare 迁移到了 Hyper-V 虚拟机。

我们的一个应用程序以每秒 20K 数据包的速率将 UDP 数据包发送到多播云。

虽然这在 VMWare 环境中完美运行,但 Hyper-V 使应用程序在工作几分钟后抛出以下异常:

System.Net.Sockets.SocketException (0x80004005): An invalid argument was supplied
   at System.Net.Sockets.Socket.Send(Byte[] buffer, Int32 offset, Int32 size, SocketFlags socketFlags)

在将套接字的发送缓冲区大小定义为 1,000,000 字节时,我还设法立即模拟了这个问题。

我该如何解决这个问题?

更新 1:这是异常发生后来自事件查看器的日志条目:

Faulting application name: Agent.exe, version: 1.0.12.7366, time stamp: 0x51389f69
Faulting module name: KERNELBASE.dll, version: 6.1.7601.18015, time stamp: 0x50b83c8a
Exception code: 0xe0434352
Fault offset: 0x0000c41f
Faulting process id: 0xaf0
Faulting application start time: 0x01ce1b4ce509dc7a
Faulting application path: C:\Users\DevUser\Desktop\QA\Agent.exe
Faulting module path: C:\Windows\syswow64\KERNELBASE.dll
Report Id: d2b45dce-8740-11e2-86f9-00155d022804

更新2:UDP数据包的大小为100-200字节。

更新 3:这是有问题的代码:

    m_socket = new Socket(AddressFamily.InterNetwork, SocketType.Dgram, ProtocolType.Udp);
    m_socket.Ttl = 1;

    if (GetRawParameter("send") != null)
    {
      Log("Starting sender...");

      StartSender();
    }

    ...snip...

private static void StartSender()
{
  m_lastPacketNumber = 0;

  m_socket.Connect(new IPEndPoint(m_ipAddress, m_port));

  if (m_bufferSize > 0)
    m_socket.SetSocketOption(
      SocketOptionLevel.Socket, SocketOptionName.SendBuffer, m_bufferSize);

  byte[] dataPad = null;

  if (m_packetSize > 8)
  {
    dataPad = new byte[m_packetSize - sizeof(long)];

    for (int i = 0; i < dataPad.Length; i++)
    {
      dataPad[i] = 0xFF;
    }
  }

  while (true)
  {
    Log("Sending data...");

    for (int i = 0; i < m_packetsPerSec; i++)
    {
      var data = BitConverter.GetBytes(m_lastPacketNumber.Value);

      if (dataPad != null)
        data = data.Concat(dataPad).ToArray();

      if (m_packetDump != null)
        m_packetDump.Add(m_lastPacketNumber.Value);

      m_socket.Send(data);

      if (m_usePerformanceCounters)
        IncreaseSendCounters(1);

      m_lastPacketNumber++;
    }

    Log(m_lastPacketNumber + " packets sent.");

    Thread.Sleep(1000);
  }
}

更新 4:当我尝试每秒发送 100K 数据包时,失败的 Send() 似乎发生在 #14156 或 #32485 或 #25412 数据包(不是第一个!)。

最佳答案

你能不能把数据包“卷起来”,这样你发送的数据包就少得多,但尺寸更大……然后在另一边展开它们? 20k 数据包/秒是很多 sendto 的。

关于c# - Hyper-V 机器上的套接字异常,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15276242/

相关文章:

c# - 如何将 foreach 循环转换为 lambda 表达式

c# - 转换对象导致异常

c++ - 如何使用 boost asio 通过 UDP 套接字发送 std::vector of unsigned char?

c++ - 如何通过 winsock 发送自定义数据(可能是来自指针的数据)

C++如何实现进程间socket通信?

c++ WINSOCK recvfrom 没有从网络获取数据

c# - 如何在 C# 中将固定字节/char[100] 转换为托管 char[]?

c# - ASP.NET MVC DropDownList 未设置选定值

udp - 无法显示通过 UDP 使用 GStreamer 流式传输的 h.264 网络摄像头图像

java - 套接字和数据报 channel