kaps@orange.ucsb.edu (Vikas Kapur) (12/04/89)
I'm posting this for a friend who has no current net access. Your help would be *very* much appreciated - he has had no luck so far. Please reply to the address listed in the header. Like I said this is a *long* posting..source code included... ^L <* He has been trying to broadcast to 5 networked Sun workstations on a dedicated Ethernet *> I tried several variations to the basic approach outlined in the Sun-3 IPC Manual. The basic problem is to get the correct internet address corresponding to a broadcast, for which the following call is suggested : struct in_addr inet_makeaddr(net,bcast_lna) int net, bcast_lna; where net= the network number of the net on which the broadcast is required, and bcast_lna= INADDR_ANY ( I also tried INADDR_BCAST ), specifies the host_addr (last byte of the internet address). One would thus expect, for a host with, for instance, the address : "128.111.63.26" that the broadcast address on the same network would be (either "128.111.63.0" - using INADDR_ANY, as suggested by the IPC documentation from Sun, or) "128.111.63.255" using INADDR_BCAST. It turns out that variations of this call (see inet(3) of manual pages), give vastly differing return values, from "128.111.255.255" to "255.255.255.255" (whih happens to be the *physical* ethernet broadcast address (4 bytes, all 1's). Incidentally, there's a call for converting inet_addresses to the form "a.b.c.d" : inet_ntoa() , which I used to look at the values returned by the inet_makeaddr() and other call that I used (viz., inet_addr()). However, not one of these variations seemed to work. In fact, the sendto() call using the address returned as above always returned without error, but none of the processes doing a receivefrom() on the appropriate UDP-port on any of the machines on the network ever received the "broadcast". In the absence of a LANalyser to monitor the packets appearing on the network, it's difficult to say whether the broadcast didnot happen (error at the sender), or the broadcast happened but was not received by any of the machines (error at the receiver(s)). (P.S. - In all cases the UDP-port of the sending machine & receiving machines were also the same.) A simple pair of test routines for the sending party & receiving parties is listed hereunder : /*----------------------------------------------------------------*/ /* COMMON CODE */ #include <sys/types.h> #include <netinet/in.h> #include <arpa/inet.h> #include <sys/socket.h> #include <netdb.h> #include <errno.h> #define SERVENT struct servent #define HOSTENT struct hostent #define INET_ADDR struct sockaddr_in #define MAX_HOST_NAMELEN 128 #define MAX_TRY 30 #define MAX_MSG 100 #define SLEEP_INT 2 char MY_HOST_NAME[MAX_HOST_NAMELEN]; INET_ADDR MY_ADDR; int MY_SOCK; phys_init() { SERVENT *srv_ent; HOSTENT *hst_ent; int on=1; MY_ADDR.sin_family=AF_INET; /* my address family */ if((srv_ent=getservbyname("TRANS_SERVER","udp"))==(SERVENT *)0) err_exit(-1,"Unable to find TRANS_SERVER's port number\n"); MY_ADDR.sin_port=srv_ent->s_port; /* get the server's port # */ gethostname(MY_HOST_NAME,MAX_HOST_NAMELEN); /* get my host's name */ if((hst_ent=gethostbyname(MY_HOST_NAME))==(HOSTENT *)0) err_exit(-2,"Can't find my host entry\n"); bcopy(hst_ent->h_addr,&MY_ADDR.sin_addr,hst_ent->h_length); /* get my internet address */ if((MY_SOCK=socket(PF_INET,SOCK_DGRAM,0))<0) err_exit(-3,"Can't open datagram socket\n"); if(setsockopt(MY_SOCK,SOL_SOCKET,SO_BROADCAST,&on,sizeof(on))<0) err_exit(-1,"Can't set broadcast option on socket\n"); /* set broadcasting option on socket */ if(bind(MY_SOCK,&MY_ADDR,sizeof(INET_ADDR))<0) err_exit(-1,"Error in binding name to socket\n"); /* bind my full internet address (including port number) to socket */ return; } send_bcast() { INET_ADDR BCAST_ADDR; struct in_addr bcast_addr; int i,size,nbytes; char *msg="Did you get this broadcast message\n"; BCAST_ADDR.sin_family=AF_INET; BCAST_ADDR.sin_port=MY_ADDR.sin_port; bcast_addr=inet_makeaddr(inet_netof(MY_ADDR.sin_addr),INADDR_BROADCAST); printf("The broadcast address is %s",inet_ntoa(bcast_addr)); bcopy(&bcast_addr,&BCAST_ADDR.sin_addr,sizeof(struct in_addr)); /* form the broadcast socket address, preparatory to sending */ size=strlen(msg); for(i=0;i<MAX_TRY;i++){ if(sendto(MY_SOCK,msg,size,0,&BCAST_ADDR,sizeof(INET_ADDR))!=size) err_exit(-1,"Error in sending broadcast\n"); delay(); } printf("%d broadcast transmissions made\n", MAX_TRY); return; } recv_msg() { char msgbuf[MAX_MSG]; INET_ADDR src_addr; int size,nbytes; size=sizeof(INET_ADDR); if((nbytes=recvfrom(MY_SOCK,msgbuf,MAX_MSG,0,&src_addr,&size))<0) err_exit(-1,"Error in receive\n"); printf("%d byte message received : %s\n",nbytes,msgbuf); return; } delay() { sleep(SLEEP_INT); return; } err_exit(err,msg) int err; char *msg; { perror(""); printf("\nErr %d : %s",err,msg); exit(err); } /* end of common code */ /*---------------------------------------------------------------------*/ /* This is the BROADCASTER's code */ #ifdef BROADCASTER main() { phys_init(); send_bcast(); exit(0); } #endif BROADCASTER /* end of Broadcaster's code */ /*---------------------------------------------------------------------*/ /* This is the RECEIVERS' code */ #ifdef RECEIVER main() { phys_init(); recv_msg(); exit(0); } #endif RECEIVER /* end of receivers' code */ /*---------------------------------------------------------------------*/ You will notice that the "broadcast address" that the above inet_makeaddr call returns is simply the machine internet address with the last two bytes replaced by 255.255 (which is obviously wrong). I also tried the inet_addr() call and directly passed to it "a.b.c.255", but even that doesnot work. Any advice or help that you can offer will be highly appreciated. ________________________________________________________________________ Vikas Kapur