#ifndef lint static char sccsid[] = "@(#)query.c 4.6 12/21/83"; #endif #include #include #include #include #include #include #include #include #include "../protocol.h" #define WTIME 5 /* Time to wait for responses */ int s; char packet[MAXPACKETSIZE]; main(argc, argv) int argc; char *argv[]; { int cc, count, bits; struct sockaddr from; int fromlen = sizeof(from); struct timeval notime; if (argc < 2) { printf("usage: query hosts...\n"); exit(1); } s = socket(AF_INET, SOCK_DGRAM, 0); if (s < 0) { perror("socket"); exit(2); } argv++, argc--; count = argc; while (argc > 0) { query(*argv); argv++, argc--; } /* * Listen for returning packets; * may be more than one packet per host. */ bits = 1 << s; bzero(¬ime, sizeof(notime)); while (count > 0 || select(20, &bits, 0, 0, ¬ime) > 0) { cc = recvfrom(s, packet, sizeof (packet), 0, &from, &fromlen); if (cc <= 0) { if (cc < 0) { perror("recvfrom"); (void) close(s); exit(1); } continue; } rip_input(&from, cc); count--; } } query(host) char *host; { struct sockaddr_in router; register struct rip *msg = (struct rip *)packet; struct hostent *hp; struct servent *sp; bzero((char *)&router, sizeof (router)); hp = gethostbyname(host); if (hp == 0) { printf("%s: unknown\n", host); exit(1); } bcopy(hp->h_addr, &router.sin_addr, hp->h_length); router.sin_family = AF_INET; sp = getservbyname("router", "udp"); if (sp == 0) { printf("udp/router: service unknown\n"); exit(1); } router.sin_port = sp->s_port; msg->rip_cmd = RIPCMD_REQUEST; msg->rip_vers = RIPVERSION; msg->rip_nets[0].rip_dst.sa_family = htons(AF_UNSPEC); msg->rip_nets[0].rip_metric = htonl(HOPCNT_INFINITY); if (sendto(s, packet, sizeof (struct rip), 0, &router, sizeof(router)) < 0) perror(host); } /* * Handle an incoming routing packet. */ rip_input(from, size) struct sockaddr_in *from; int size; { register struct rip *msg = (struct rip *)packet; struct netinfo *n; char *name; struct hostent *hp; struct netent *np; if (msg->rip_cmd != RIPCMD_RESPONSE) return; hp = gethostbyaddr(&from->sin_addr, sizeof (struct in_addr), AF_INET); name = hp == 0 ? "???" : hp->h_name; printf("from %s(%s):\n", name, inet_ntoa(from->sin_addr)); size -= sizeof (int); n = msg->rip_nets; while (size > 0) { register struct sockaddr_in *sin; if (size < sizeof (struct netinfo)) break; if (msg->rip_vers > 0) { n->rip_dst.sa_family = ntohs(n->rip_dst.sa_family); n->rip_metric = ntohl(n->rip_metric); } sin = (struct sockaddr_in *)&n->rip_dst; if (inet_lnaof(sin->sin_addr) == INADDR_ANY) { np = getnetbyaddr(inet_netof(sin->sin_addr), AF_INET); name = np ? np->n_name : "???"; } else { hp = gethostbyaddr(&sin->sin_addr, sizeof (struct in_addr), AF_INET); name = hp ? hp->h_name : "???"; } printf("\t%s(%s), metric %d\n", name, inet_ntoa(sin->sin_addr), n->rip_metric); size -= sizeof (struct netinfo), n++; } } timeout() { timedout = 1; }