WebAssembly中的UDP套接字

12

我正在尝试将使用C和C++编写的桌面应用程序移植到WebAssembly平台,并正在调查其是否可能。该应用程序重要的功能之一是通过发送和接收UDP消息进行通信。我已经实现了一个最小的UDP客户端,它只是创建UDP套接字并将数据包发送到服务器(该服务器是原生构建的,并作为独立可执行文件在同一台机器上运行)。Socket、bind和sendto API没有返回任何错误,并且一切看起来都正常,但是服务器端没有接收到任何消息,wireshark上也没有显示任何活动。

WebAssembly libc端口中的UDP套接字是被stubbed掉了吗?还是它是在某个Web标准连接(例如WebRTC)之上实现的?

客户端代码如下。我已检查过本地版本的工作情况。

#include <stdlib.h>
#include <string.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <unistd.h>

#define BUFLEN 512
#define NPACK 100
#define PORT 9930

void diep(char *s)
{
  perror(s);
  exit(1);
}

#define SRV_IP "127.0.0.1"


int main(void)
{
  struct sockaddr_in si_other;
  int s, i, slen=sizeof(si_other);
  char buf[BUFLEN];

  if ((s=socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP))==-1)
    diep("socket");

  memset((char *) &si_other, 0, sizeof(si_other));
  si_other.sin_family = AF_INET;
  si_other.sin_port = htons(PORT);
  if (inet_aton(SRV_IP, &si_other.sin_addr)==0) {
    fprintf(stderr, "inet_aton() failed\n");
    exit(1);
  }

  for (i=0; i<NPACK; i++) {
    printf("Sending packet %d\n", i);
    sprintf(buf, "This is packet %d\n", i);
    if (sendto(s, buf, BUFLEN, 0, (struct sockaddr*)&si_other, slen)==-1)
      diep("sendto()");
  }

  close(s);
  return 0;
}

我按照http://webassembly.org/getting-started/developers-guide/的说明进行构建和运行。

提前感谢任何帮助或提示!


Wireshark是否能够捕获回环接口上的数据包(即将数据包发送到同一台机器)取决于平台和您选择的数据包嗅探接口。例如,在Windows+winpcap上,这是不可能的。你在哪个平台上? - tofro
是的,我在使用Windows系统,我会尝试将服务器放到另一台机器上。谢谢! - roof
我已经尝试向远程主机发送数据包。现在,如果我运行本地构建的客户端应用程序,则Wireshark会显示出站数据包,并且对于WebAssembly构建则不会显示任何内容。 - roof
2
WebAssembly仅支持TCP套接字,因为生成的流量嵌入到了Websockets中。你想要的在WebAssembly中无法运行。 - tofro
tofro,没错,谢谢。我已经找到了如何在wasm中实现它。 - roof
2个回答

7

我发现webassembly是如何实现UDP套接字的,实际上它们是通过websockets来模拟的。如果客户端和服务器都是webassembly可能也可以这样做,但我的服务器是本地构建的。由于wasm不支持动态链接,所有代码(包括libc实现)都被捆绑到一个JS文件中,在那里我们可以找到UDP sendto的实现:

// if we're emulating a connection-less dgram socket and don't have
      // a cached connection, queue the buffer to send upon connect and
      // lie, saying the data was sent now.
      if (sock.type === 2) {
        if (!dest || dest.socket.readyState !== dest.socket.OPEN) {
          // if we're not connected, open a new connection
          if (!dest || dest.socket.readyState === dest.socket.CLOSING || dest.socket.readyState === dest.socket.CLOSED) {
            dest = SOCKFS.websocket_sock_ops.createPeer(sock, addr, port);
          }
          dest.dgram_send_queue.push(data);
          return length;
        }
      }

4

在浏览器中运行的任何内容都无法获得原生套接字访问,我怀疑浏览器供应商会强烈反对任何此类访问,因为这可能是一种安全漏洞。

也许随着越来越多的本地应用程序转移到Web上,由于WebAssembly和类似的倡议使性能差异缩小,他们的立场可能会改变,但在此之前,任何想要直接控制套接字的内容都必须保持为本地应用程序。


我希望有一些聪明的UDP套接字实现,可以在WebRTC之上甚至在那个W3C草案的实验性支持之上运行。 - roof
1
你会认为他们会提供一些UDP访问权限,因为它对于某些事情很有用,比如实时音频:\ - rogerdpack
@AdnanY:本地客户端应用程序可以类似于JavaScript应用程序启动,并提供原始套接字。 - user2284570

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接