UDP广播这里有问题吗?
UDP broadcast Is there something wrong here?
我有一个 UDP 服务器在 IP:192.168.0.53 端口 1337 上侦听我的本地网络。
我在本地网络设置的同一子网上也有一个 UDP 客户端,用于将数据包发送到 255.255.255.255 端口 1337 但它从未到达我的服务器。这种情况反复发生,如您所见,我正在使用广播地址。
我尝试从我的客户端发送一个数据包到 192.168.0.53 端口 1337 并且它顺利到达。显示更简单的非广播路由效果很好。
我试过2个客户端,因为我认为需要设置一些特殊标志才能进行广播。 PacketSender and this specific broadcast application 我的客户都没有收到。
我的服务器是用ESP8266WiFi UDP class写的。
我是不是漏掉了什么?广播应该由我的服务器接收但不是。我的服务器可能需要一个特殊标志吗?
服务器代码(基于上面链接的 Udp class)只是:
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);
}
从这个 github https://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 软件。
解决方案是改用多播。
我的服务器代码已更改为加入多播组并侦听特定端口:
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 数据包,这是多播组,瞧 - 它很有用!
我有一个 UDP 服务器在 IP:192.168.0.53 端口 1337 上侦听我的本地网络。
我在本地网络设置的同一子网上也有一个 UDP 客户端,用于将数据包发送到 255.255.255.255 端口 1337 但它从未到达我的服务器。这种情况反复发生,如您所见,我正在使用广播地址。
我尝试从我的客户端发送一个数据包到 192.168.0.53 端口 1337 并且它顺利到达。显示更简单的非广播路由效果很好。
我试过2个客户端,因为我认为需要设置一些特殊标志才能进行广播。 PacketSender and this specific broadcast application 我的客户都没有收到。
我的服务器是用ESP8266WiFi UDP class写的。
我是不是漏掉了什么?广播应该由我的服务器接收但不是。我的服务器可能需要一个特殊标志吗?
服务器代码(基于上面链接的 Udp class)只是:
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);
}
从这个 github https://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 软件。
解决方案是改用多播。
我的服务器代码已更改为加入多播组并侦听特定端口:
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 数据包,这是多播组,瞧 - 它很有用!