【问题标题】:UDP broadcast Is there something wrong here?UDP广播这里有什么问题吗?
【发布时间】:2018-02-02 01:13:55
【问题描述】:

我有一个 UDP 服务器在我的本地网络上侦听 IP:192.168.0.53 端口 1337。

我在本地网络设置的同一子网上也有一个 UDP 客户端,用于将数据包发送到 255.255.255.255 端口 1337 它永远不会到达我的服务器。这种情况反复发生,如你所见,我使用的是广播地址。

我尝试从我的客户端向 192.168.0.53 端口 1337 发送一个数据包它可以正常到达。显示更简单的非广播路由效果很好。

我已经尝试了 2 个客户端,因为我认为需要为广播设置一些特殊标志。我的客户没有收到PacketSender 和这个specific broadcast application

我的服务器是使用ESP8266WiFi UDP class编写的。

我在这里遗漏了什么吗?广播应该被我的服务器接收,但不是。我的服务器上可能需要一个特殊的标志吗?

服务器代码(基于上面链接的 Udp 类)就是:

     int packetSize = Udp.parsePacket();
  if (packetSize){
     // receive incoming UDP packets
    String msg = "Received ";
    msg += String(packetSize)+" bytes from IP:";
    msg += Udp.remoteIP().toString()+" on port:"+Udp.remotePort();
    this->DCPrintf(msg);

    int len = Udp.read(incomingPacket, 255);
    if (len > 0)
    {
      incomingPacket[len] = 0;
    }
    msg = "UDP packet contents:";
    msg +=incomingPacket;
    this->DCPrintf(msg);

  }

从这个githubhttps://github.com/stanwu/udp-broadcast发送代码:

/* fpont 12/99 */
/* pont.net    */
/* udpClient.c */

#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <netdb.h>
#include <stdio.h>
#include <unistd.h>
#include <string.h> /* memset() */
#include <sys/time.h> /* select() */ 

//for Mac OS X
#include <stdlib.h>

#define REMOTE_SERVER_PORT 1500
#define MAX_MSG 100


int main(int argc, char *argv[]) {

  int sd, rc, i;
  struct sockaddr_in cliAddr, remoteServAddr;
  struct hostent *h;
  int broadcast = 1;

  /* check command line args */
  if(argc<3) {
    printf("usage : %s <server> <data1> ... <dataN> \n", argv[0]);
    exit(1);
  }

  /* get server IP address (no check if input is IP address or DNS name */
  h = gethostbyname(argv[1]);
  if(h==NULL) {
    printf("%s: unknown host '%s' \n", argv[0], argv[1]);
    exit(1);
  }

  printf("%s: sending data to '%s' (IP : %s) \n", argv[0], h->h_name,
     inet_ntoa(*(struct in_addr *)h->h_addr_list[0]));

  remoteServAddr.sin_family = h->h_addrtype;
  memcpy((char *) &remoteServAddr.sin_addr.s_addr, 
     h->h_addr_list[0], h->h_length);
  remoteServAddr.sin_port = htons(REMOTE_SERVER_PORT);

  /* socket creation */
  sd = socket(AF_INET,SOCK_DGRAM,0);
  if(sd<0) {
    printf("%s: cannot open socket \n",argv[0]);
    exit(1);
  }


  if (setsockopt(sd, SOL_SOCKET, SO_BROADCAST, &broadcast,sizeof broadcast) == -1) {
          perror("setsockopt (SO_BROADCAST)");
          exit(1);
  }

  /* bind any port */
  cliAddr.sin_family = AF_INET;
  cliAddr.sin_addr.s_addr = htonl(INADDR_ANY);
  cliAddr.sin_port = htons(0);

  rc = bind(sd, (struct sockaddr *) &cliAddr, sizeof(cliAddr));
  if(rc<0) {
    printf("%s: cannot bind port\n", argv[0]);
    exit(1);
  }


  /* send data */
  for(i=2;i<argc;i++) {
    rc = sendto(sd, argv[i], strlen(argv[i])+1, 0, 
        (struct sockaddr *) &remoteServAddr, 
        sizeof(remoteServAddr));

    if(rc<0) {
      printf("%s: cannot send data %d \n",argv[0],i-1);
      close(sd);
      exit(1);
    }

  }

  return 1;

}

另外一个客户端就是PacketSender软件。

【问题讨论】:

  • 发送到子网广播地址192.168.0.255。向255.255.255.255 广播已被弃用二十多年。
  • @EJP 奇怪做了一些阅读,它建议 255.255.255.255 应该可以工作。然而 192.168.0.255 没有从任何一个客户端到达我的服务器。同样,单播直接工作正常。
  • 您需要在发送方设置SO_BROADCAST。请在此处发布您的发送代码。
  • 添加代码@EJP
  • @EJP 你能解释一下为什么我的服务器没有收到我的广播包吗?

标签: networking arduino udp broadcast esp8266


【解决方案1】:

解决方案是改用多播。

我的服务器代码已更改为加入多播组并侦听特定端口:

int port=12345;
  IPAddress local=WiFi.localIP();

  IPAddress multicastGroup(233, 100, 100, 53);
  Udp.beginMulticast(local,multicastGroup,port); 

然后我从任何客户端向 IP 233.100.100.53 port 12345 发送了一个 UDP 数据包,这是多播组,瞧——它的工作原理!

【讨论】:

  • 还是更好的选择。在不监听这些数据包的节点上效率更高。
猜你喜欢
  • 2011-02-22
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2023-04-10
  • 2021-10-03
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多