19b50d902SRodney W. Grimes /*- 29b50d902SRodney W. Grimes * Copyright (c) 1980, 1992, 1993 39b50d902SRodney W. Grimes * The Regents of the University of California. All rights reserved. 49b50d902SRodney W. Grimes * 59b50d902SRodney W. Grimes * Redistribution and use in source and binary forms, with or without 69b50d902SRodney W. Grimes * modification, are permitted provided that the following conditions 79b50d902SRodney W. Grimes * are met: 89b50d902SRodney W. Grimes * 1. Redistributions of source code must retain the above copyright 99b50d902SRodney W. Grimes * notice, this list of conditions and the following disclaimer. 109b50d902SRodney W. Grimes * 2. Redistributions in binary form must reproduce the above copyright 119b50d902SRodney W. Grimes * notice, this list of conditions and the following disclaimer in the 129b50d902SRodney W. Grimes * documentation and/or other materials provided with the distribution. 139b50d902SRodney W. Grimes * 3. All advertising materials mentioning features or use of this software 149b50d902SRodney W. Grimes * must display the following acknowledgement: 159b50d902SRodney W. Grimes * This product includes software developed by the University of 169b50d902SRodney W. Grimes * California, Berkeley and its contributors. 179b50d902SRodney W. Grimes * 4. Neither the name of the University nor the names of its contributors 189b50d902SRodney W. Grimes * may be used to endorse or promote products derived from this software 199b50d902SRodney W. Grimes * without specific prior written permission. 209b50d902SRodney W. Grimes * 219b50d902SRodney W. Grimes * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND 229b50d902SRodney W. Grimes * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 239b50d902SRodney W. Grimes * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 249b50d902SRodney W. Grimes * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE 259b50d902SRodney W. Grimes * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 269b50d902SRodney W. Grimes * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 279b50d902SRodney W. Grimes * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 289b50d902SRodney W. Grimes * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 299b50d902SRodney W. Grimes * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 309b50d902SRodney W. Grimes * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 319b50d902SRodney W. Grimes * SUCH DAMAGE. 329b50d902SRodney W. Grimes */ 339b50d902SRodney W. Grimes 349ff712b0SMark Murray #include <sys/cdefs.h> 359ff712b0SMark Murray 369ff712b0SMark Murray __FBSDID("$FreeBSD$"); 379ff712b0SMark Murray 389ff712b0SMark Murray #ifdef lint 399ff712b0SMark Murray static const char sccsid[] = "@(#)netcmds.c 8.1 (Berkeley) 6/6/93"; 409ff712b0SMark Murray #endif 419b50d902SRodney W. Grimes 429b50d902SRodney W. Grimes /* 439b50d902SRodney W. Grimes * Common network command support routines. 449b50d902SRodney W. Grimes */ 459b50d902SRodney W. Grimes #include <sys/param.h> 46d8d89152SDavid Greenman #include <sys/queue.h> 479b50d902SRodney W. Grimes #include <sys/socket.h> 489b50d902SRodney W. Grimes #include <sys/socketvar.h> 499b50d902SRodney W. Grimes #include <sys/protosw.h> 509b50d902SRodney W. Grimes 519b50d902SRodney W. Grimes #include <net/route.h> 529b50d902SRodney W. Grimes #include <netinet/in.h> 539b50d902SRodney W. Grimes #include <netinet/in_systm.h> 549b50d902SRodney W. Grimes #include <netinet/ip.h> 559b50d902SRodney W. Grimes #include <netinet/in_pcb.h> 569ec49abdSPeter Wemm #include <arpa/inet.h> 579b50d902SRodney W. Grimes 589ff712b0SMark Murray #include <ctype.h> 599b50d902SRodney W. Grimes #include <netdb.h> 609b50d902SRodney W. Grimes #include <stdlib.h> 619b50d902SRodney W. Grimes #include <string.h> 629ff712b0SMark Murray 639b50d902SRodney W. Grimes #include "systat.h" 649b50d902SRodney W. Grimes #include "extern.h" 659b50d902SRodney W. Grimes 669b50d902SRodney W. Grimes #define streq(a,b) (strcmp(a,b)==0) 679b50d902SRodney W. Grimes 689b50d902SRodney W. Grimes static struct hitem { 699b50d902SRodney W. Grimes struct in_addr addr; 709b50d902SRodney W. Grimes int onoff; 719b50d902SRodney W. Grimes } *hosts; 729b50d902SRodney W. Grimes 739b50d902SRodney W. Grimes int nports, nhosts, protos; 749b50d902SRodney W. Grimes 753f330d7dSWarner Losh static void changeitems(const char *, int); 763f330d7dSWarner Losh static int selectproto(const char *); 773f330d7dSWarner Losh static void showprotos(void); 783f330d7dSWarner Losh static int selectport(long, int); 793f330d7dSWarner Losh static void showports(void); 803f330d7dSWarner Losh static int selecthost(struct in_addr *, int); 813f330d7dSWarner Losh static void showhosts(void); 829b50d902SRodney W. Grimes 839b50d902SRodney W. Grimes int 8493b9f504SXin LI netcmd(const char *cmd, const char *args) 859b50d902SRodney W. Grimes { 869b50d902SRodney W. Grimes 8779431394SSteve Price if (prefix(cmd, "proto")) { 8879431394SSteve Price if (*args == '\0') { 8979431394SSteve Price move(CMDLINE, 0); 9079431394SSteve Price clrtoeol(); 9179431394SSteve Price addstr("which proto?"); 9279431394SSteve Price } else if (!selectproto(args)) { 9379431394SSteve Price error("%s: Unknown protocol.", args); 9479431394SSteve Price } 959b50d902SRodney W. Grimes return (1); 969b50d902SRodney W. Grimes } 979b50d902SRodney W. Grimes if (prefix(cmd, "ignore") || prefix(cmd, "display")) { 989b50d902SRodney W. Grimes changeitems(args, prefix(cmd, "display")); 999b50d902SRodney W. Grimes return (1); 1009b50d902SRodney W. Grimes } 1019b50d902SRodney W. Grimes if (prefix(cmd, "reset")) { 1029b50d902SRodney W. Grimes selectproto(0); 1039b50d902SRodney W. Grimes selecthost(0, 0); 1049b50d902SRodney W. Grimes selectport(-1, 0); 1059b50d902SRodney W. Grimes return (1); 1069b50d902SRodney W. Grimes } 1079b50d902SRodney W. Grimes if (prefix(cmd, "show")) { 1089b50d902SRodney W. Grimes move(CMDLINE, 0); clrtoeol(); 1099b50d902SRodney W. Grimes if (*args == '\0') { 1109b50d902SRodney W. Grimes showprotos(); 1119b50d902SRodney W. Grimes showhosts(); 1129b50d902SRodney W. Grimes showports(); 1139b50d902SRodney W. Grimes return (1); 1149b50d902SRodney W. Grimes } 1159b50d902SRodney W. Grimes if (prefix(args, "protos")) 1169b50d902SRodney W. Grimes showprotos(); 1179b50d902SRodney W. Grimes else if (prefix(args, "hosts")) 1189b50d902SRodney W. Grimes showhosts(); 1199b50d902SRodney W. Grimes else if (prefix(args, "ports")) 1209b50d902SRodney W. Grimes showports(); 1219b50d902SRodney W. Grimes else 1229b50d902SRodney W. Grimes addstr("show what?"); 1239b50d902SRodney W. Grimes return (1); 1249b50d902SRodney W. Grimes } 1259b50d902SRodney W. Grimes return (0); 1269b50d902SRodney W. Grimes } 1279b50d902SRodney W. Grimes 1289b50d902SRodney W. Grimes 1299b50d902SRodney W. Grimes static void 13093b9f504SXin LI changeitems(const char *args, int onoff) 1319b50d902SRodney W. Grimes { 1329ff712b0SMark Murray char *cp, *tmpstr, *tmpstr1; 1339b50d902SRodney W. Grimes struct servent *sp; 1349b50d902SRodney W. Grimes struct hostent *hp; 1359b50d902SRodney W. Grimes struct in_addr in; 1369b50d902SRodney W. Grimes 1379ff712b0SMark Murray tmpstr = tmpstr1 = strdup(args); 1389ff712b0SMark Murray cp = index(tmpstr1, '\n'); 1399b50d902SRodney W. Grimes if (cp) 1409b50d902SRodney W. Grimes *cp = '\0'; 1419ff712b0SMark Murray for (;;tmpstr1 = cp) { 1429ff712b0SMark Murray for (cp = tmpstr1; *cp && isspace(*cp); cp++) 1439b50d902SRodney W. Grimes ; 1449ff712b0SMark Murray tmpstr1 = cp; 1459b50d902SRodney W. Grimes for (; *cp && !isspace(*cp); cp++) 1469b50d902SRodney W. Grimes ; 1479b50d902SRodney W. Grimes if (*cp) 1489b50d902SRodney W. Grimes *cp++ = '\0'; 1499ff712b0SMark Murray if (cp - tmpstr1 == 0) 1509b50d902SRodney W. Grimes break; 1519ff712b0SMark Murray sp = getservbyname(tmpstr1, 1529b50d902SRodney W. Grimes protos == TCP ? "tcp" : protos == UDP ? "udp" : 0); 1539b50d902SRodney W. Grimes if (sp) { 1549b50d902SRodney W. Grimes selectport(sp->s_port, onoff); 1559b50d902SRodney W. Grimes continue; 1569b50d902SRodney W. Grimes } 1579ff712b0SMark Murray hp = gethostbyname(tmpstr1); 1589b50d902SRodney W. Grimes if (hp == 0) { 1599ff712b0SMark Murray in.s_addr = inet_addr(tmpstr1); 1609ff712b0SMark Murray if ((int)in.s_addr == -1) { 1619ff712b0SMark Murray error("%s: unknown host or port", tmpstr1); 1629b50d902SRodney W. Grimes continue; 1639b50d902SRodney W. Grimes } 1649b50d902SRodney W. Grimes } else 1659b50d902SRodney W. Grimes in = *(struct in_addr *)hp->h_addr; 1669b50d902SRodney W. Grimes selecthost(&in, onoff); 1679b50d902SRodney W. Grimes } 1689ff712b0SMark Murray free(tmpstr); 1699b50d902SRodney W. Grimes } 1709b50d902SRodney W. Grimes 1719b50d902SRodney W. Grimes static int 17293b9f504SXin LI selectproto(const char *proto) 1739b50d902SRodney W. Grimes { 1749b50d902SRodney W. Grimes 1759b50d902SRodney W. Grimes if (proto == 0 || streq(proto, "all")) 17679431394SSteve Price protos = TCP | UDP; 1779b50d902SRodney W. Grimes else if (streq(proto, "tcp")) 17879431394SSteve Price protos = TCP; 1799b50d902SRodney W. Grimes else if (streq(proto, "udp")) 18079431394SSteve Price protos = UDP; 18179431394SSteve Price else 18279431394SSteve Price return (0); 18379431394SSteve Price 18479431394SSteve Price return (protos); 1859b50d902SRodney W. Grimes } 1869b50d902SRodney W. Grimes 1879b50d902SRodney W. Grimes static void 18893b9f504SXin LI showprotos(void) 1899b50d902SRodney W. Grimes { 1909b50d902SRodney W. Grimes 1919b50d902SRodney W. Grimes if ((protos&TCP) == 0) 1929b50d902SRodney W. Grimes addch('!'); 1939b50d902SRodney W. Grimes addstr("tcp "); 1949b50d902SRodney W. Grimes if ((protos&UDP) == 0) 1959b50d902SRodney W. Grimes addch('!'); 1969b50d902SRodney W. Grimes addstr("udp "); 1979b50d902SRodney W. Grimes } 1989b50d902SRodney W. Grimes 1999b50d902SRodney W. Grimes static struct pitem { 2009b50d902SRodney W. Grimes long port; 2019b50d902SRodney W. Grimes int onoff; 2029b50d902SRodney W. Grimes } *ports; 2039b50d902SRodney W. Grimes 2049b50d902SRodney W. Grimes static int 20593b9f504SXin LI selectport(long port, int onoff) 2069b50d902SRodney W. Grimes { 20793b9f504SXin LI struct pitem *p; 2089b50d902SRodney W. Grimes 2099b50d902SRodney W. Grimes if (port == -1) { 2109b50d902SRodney W. Grimes if (ports == 0) 2119b50d902SRodney W. Grimes return (0); 2129b50d902SRodney W. Grimes free((char *)ports), ports = 0; 2139b50d902SRodney W. Grimes nports = 0; 2149b50d902SRodney W. Grimes return (1); 2159b50d902SRodney W. Grimes } 2169b50d902SRodney W. Grimes for (p = ports; p < ports+nports; p++) 2179b50d902SRodney W. Grimes if (p->port == port) { 2189b50d902SRodney W. Grimes p->onoff = onoff; 2199b50d902SRodney W. Grimes return (0); 2209b50d902SRodney W. Grimes } 2219b50d902SRodney W. Grimes if (nports == 0) 2229b50d902SRodney W. Grimes ports = (struct pitem *)malloc(sizeof (*p)); 2239b50d902SRodney W. Grimes else 2249b50d902SRodney W. Grimes ports = (struct pitem *)realloc(ports, (nports+1)*sizeof (*p)); 2259b50d902SRodney W. Grimes p = &ports[nports++]; 2269b50d902SRodney W. Grimes p->port = port; 2279b50d902SRodney W. Grimes p->onoff = onoff; 2289b50d902SRodney W. Grimes return (1); 2299b50d902SRodney W. Grimes } 2309b50d902SRodney W. Grimes 2319b50d902SRodney W. Grimes int 23293b9f504SXin LI checkport(struct inpcb *inp) 2339b50d902SRodney W. Grimes { 23493b9f504SXin LI struct pitem *p; 2359b50d902SRodney W. Grimes 2369b50d902SRodney W. Grimes if (ports) 2379b50d902SRodney W. Grimes for (p = ports; p < ports+nports; p++) 2389b50d902SRodney W. Grimes if (p->port == inp->inp_lport || p->port == inp->inp_fport) 2399b50d902SRodney W. Grimes return (p->onoff); 2409b50d902SRodney W. Grimes return (1); 2419b50d902SRodney W. Grimes } 2429b50d902SRodney W. Grimes 2439b50d902SRodney W. Grimes static void 24493b9f504SXin LI showports(void) 2459b50d902SRodney W. Grimes { 24693b9f504SXin LI struct pitem *p; 2479b50d902SRodney W. Grimes struct servent *sp; 2489b50d902SRodney W. Grimes 2499b50d902SRodney W. Grimes for (p = ports; p < ports+nports; p++) { 2509b50d902SRodney W. Grimes sp = getservbyport(p->port, 251342e2faaSThomas Moestl protos == (TCP|UDP) ? 0 : protos == TCP ? "tcp" : "udp"); 2529b50d902SRodney W. Grimes if (!p->onoff) 2539b50d902SRodney W. Grimes addch('!'); 2549b50d902SRodney W. Grimes if (sp) 2559b50d902SRodney W. Grimes printw("%s ", sp->s_name); 2569b50d902SRodney W. Grimes else 2579b50d902SRodney W. Grimes printw("%d ", p->port); 2589b50d902SRodney W. Grimes } 2599b50d902SRodney W. Grimes } 2609b50d902SRodney W. Grimes 2619b50d902SRodney W. Grimes static int 26293b9f504SXin LI selecthost(struct in_addr *in, int onoff) 2639b50d902SRodney W. Grimes { 26493b9f504SXin LI struct hitem *p; 2659b50d902SRodney W. Grimes 2669b50d902SRodney W. Grimes if (in == 0) { 2679b50d902SRodney W. Grimes if (hosts == 0) 2689b50d902SRodney W. Grimes return (0); 2699b50d902SRodney W. Grimes free((char *)hosts), hosts = 0; 2709b50d902SRodney W. Grimes nhosts = 0; 2719b50d902SRodney W. Grimes return (1); 2729b50d902SRodney W. Grimes } 2739b50d902SRodney W. Grimes for (p = hosts; p < hosts+nhosts; p++) 2749b50d902SRodney W. Grimes if (p->addr.s_addr == in->s_addr) { 2759b50d902SRodney W. Grimes p->onoff = onoff; 2769b50d902SRodney W. Grimes return (0); 2779b50d902SRodney W. Grimes } 2789b50d902SRodney W. Grimes if (nhosts == 0) 2799b50d902SRodney W. Grimes hosts = (struct hitem *)malloc(sizeof (*p)); 2809b50d902SRodney W. Grimes else 2819b50d902SRodney W. Grimes hosts = (struct hitem *)realloc(hosts, (nhosts+1)*sizeof (*p)); 2829b50d902SRodney W. Grimes p = &hosts[nhosts++]; 2839b50d902SRodney W. Grimes p->addr = *in; 2849b50d902SRodney W. Grimes p->onoff = onoff; 2859b50d902SRodney W. Grimes return (1); 2869b50d902SRodney W. Grimes } 2879b50d902SRodney W. Grimes 2889b50d902SRodney W. Grimes int 28993b9f504SXin LI checkhost(struct inpcb *inp) 2909b50d902SRodney W. Grimes { 29193b9f504SXin LI struct hitem *p; 2929b50d902SRodney W. Grimes 2939b50d902SRodney W. Grimes if (hosts) 2949b50d902SRodney W. Grimes for (p = hosts; p < hosts+nhosts; p++) 2959b50d902SRodney W. Grimes if (p->addr.s_addr == inp->inp_laddr.s_addr || 2969b50d902SRodney W. Grimes p->addr.s_addr == inp->inp_faddr.s_addr) 2979b50d902SRodney W. Grimes return (p->onoff); 2989b50d902SRodney W. Grimes return (1); 2999b50d902SRodney W. Grimes } 3009b50d902SRodney W. Grimes 3019b50d902SRodney W. Grimes static void 30293b9f504SXin LI showhosts(void) 3039b50d902SRodney W. Grimes { 30493b9f504SXin LI struct hitem *p; 3059b50d902SRodney W. Grimes struct hostent *hp; 3069b50d902SRodney W. Grimes 3079b50d902SRodney W. Grimes for (p = hosts; p < hosts+nhosts; p++) { 3089b50d902SRodney W. Grimes hp = gethostbyaddr((char *)&p->addr, sizeof (p->addr), AF_INET); 3099b50d902SRodney W. Grimes if (!p->onoff) 3109b50d902SRodney W. Grimes addch('!'); 3119b50d902SRodney W. Grimes printw("%s ", hp ? hp->h_name : (char *)inet_ntoa(p->addr)); 3129b50d902SRodney W. Grimes } 3139b50d902SRodney W. Grimes } 314