]>
git.saurik.com Git - apple/network_cmds.git/blob - netstat.tproj/main.c
7a9d7d7342869495548cef0030d5d9eaf9be1f00
2 * Copyright (c) 2008-2015 Apple Inc. All rights reserved.
4 * @APPLE_OSREFERENCE_LICENSE_HEADER_START@
6 * This file contains Original Code and/or Modifications of Original Code
7 * as defined in and that are subject to the Apple Public Source License
8 * Version 2.0 (the 'License'). You may not use this file except in
9 * compliance with the License. The rights granted to you under the License
10 * may not be used to create, or enable the creation or redistribution of,
11 * unlawful or unlicensed copies of an Apple operating system, or to
12 * circumvent, violate, or enable the circumvention or violation of, any
13 * terms of an Apple operating system software license agreement.
15 * Please obtain a copy of the License at
16 * http://www.opensource.apple.com/apsl/ and read it before using this file.
18 * The Original Code and all software distributed under the License are
19 * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
20 * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
21 * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
22 * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
23 * Please see the License for the specific language governing rights and
24 * limitations under the License.
26 * @APPLE_OSREFERENCE_LICENSE_HEADER_END@
29 * Copyright (c) 1983, 1988, 1993
30 * Regents of the University of California. All rights reserved.
32 * Redistribution and use in source and binary forms, with or without
33 * modification, are permitted provided that the following conditions
35 * 1. Redistributions of source code must retain the above copyright
36 * notice, this list of conditions and the following disclaimer.
37 * 2. Redistributions in binary form must reproduce the above copyright
38 * notice, this list of conditions and the following disclaimer in the
39 * documentation and/or other materials provided with the distribution.
40 * 3. All advertising materials mentioning features or use of this software
41 * must display the following acknowledgement:
42 * This product includes software developed by the University of
43 * California, Berkeley and its contributors.
44 * 4. Neither the name of the University nor the names of its contributors
45 * may be used to endorse or promote products derived from this software
46 * without specific prior written permission.
48 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
49 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
50 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
51 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
52 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
53 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
54 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
55 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
56 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
57 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
62 char const copyright
[] =
63 "@(#) Copyright (c) 1983, 1988, 1993\n\
64 Regents of the University of California. All rights reserved.\n";
67 #include <sys/param.h>
69 #include <sys/socket.h>
70 #include <sys/sys_domain.h>
72 #include <netinet/in.h>
73 #include <net/pfkeyv2.h>
87 #include <sys/types.h>
88 #include <sys/sysctl.h>
91 #include <TargetConditionals.h>
95 * ----------------------------------------------------------------------------
96 * "THE BEER-WARE LICENSE" (Revision 42):
97 * <phk@FreeBSD.org> wrote this file. As long as you retain this notice you
98 * can do whatever you want with this stuff. If we meet some day, and you think
99 * this stuff is worth it, you can buy me a beer in return. Poul-Henning Kamp
100 * ----------------------------------------------------------------------------
102 * $Id: main.c,v 1.8 2004/10/14 22:24:09 lindak Exp $
107 void (*pr_cblocks
)(uint32_t, char *, int);
108 /* control blocks printing routine */
109 void (*pr_stats
)(uint32_t, char *, int);
110 /* statistics printing routine */
111 void (*pr_istats
)(char *); /* per/if statistics printing routine */
112 char *pr_name
; /* well-known name */
115 { protopr
, tcp_stats
, NULL
, "tcp", IPPROTO_TCP
},
116 { protopr
, udp_stats
, NULL
, "udp", IPPROTO_UDP
},
117 { protopr
, NULL
, NULL
, "divert", IPPROTO_DIVERT
},
118 { protopr
, ip_stats
, NULL
, "ip", IPPROTO_RAW
},
119 { protopr
, icmp_stats
, NULL
, "icmp", IPPROTO_ICMP
},
120 { protopr
, igmp_stats
, NULL
, "igmp", IPPROTO_IGMP
},
122 { NULL
, ipsec_stats
, NULL
, "ipsec", IPPROTO_ESP
},
124 { NULL
, arp_stats
, NULL
, "arp", 0 },
125 { mptcppr
, mptcp_stats
, NULL
, "mptcp", IPPROTO_TCP
},
126 { NULL
, NULL
, NULL
, NULL
, 0 }
130 struct protox ip6protox
[] = {
131 { protopr
, tcp_stats
, NULL
, "tcp", IPPROTO_TCP
},
132 { protopr
, udp_stats
, NULL
, "udp", IPPROTO_UDP
},
133 { protopr
, ip6_stats
, ip6_ifstats
, "ip6", IPPROTO_RAW
},
134 { protopr
, icmp6_stats
, icmp6_ifstats
, "icmp6",IPPROTO_ICMPV6
},
136 { NULL
, ipsec_stats
, NULL
, "ipsec6", IPPROTO_ESP
},
138 { NULL
, rip6_stats
, NULL
, "rip6", IPPROTO_RAW
},
139 { mptcppr
, mptcp_stats
, NULL
, "mptcp", IPPROTO_TCP
},
140 { NULL
, NULL
, NULL
, NULL
, 0 }
145 struct protox pfkeyprotox
[] = {
146 { NULL
, pfkey_stats
, NULL
, "pfkey", PF_KEY_V2
},
147 { NULL
, NULL
, NULL
, NULL
, 0 }
152 struct protox systmprotox
[] = {
153 { systmpr
, NULL
, NULL
, "reg", 0 },
154 { systmpr
, kevt_stats
, NULL
, "kevt", SYSPROTO_EVENT
},
155 { systmpr
, kctl_stats
, NULL
, "kctl", SYSPROTO_CONTROL
},
156 { NULL
, NULL
, NULL
, NULL
, 0 }
159 struct protox nstatprotox
[] = {
160 { NULL
, print_nstat_stats
, NULL
, "nstat", 0 },
161 { NULL
, NULL
, NULL
, NULL
, 0 }
164 struct protox ipcprotox
[] = {
165 { NULL
, print_extbkidle_stats
, NULL
, "xbkidle", 0 },
166 { NULL
, NULL
, NULL
, NULL
, 0 }
169 struct protox kernprotox
[] = {
170 { NULL
, print_net_api_stats
, NULL
, "net_api", 0 },
171 { NULL
, NULL
, NULL
, NULL
, 0 }
174 struct protox
*protoprotox
[] = {
189 static void printproto (struct protox
*, char *);
190 static void usage (void);
191 static struct protox
*name2protox (char *);
192 static struct protox
*knownname (char *);
194 extern void _serv_cache_close();
197 int Aflag
; /* show addresses of protocol control block */
198 int aflag
; /* show all sockets (including servers) */
199 int bflag
; /* show i/f total bytes in/out */
200 int cflag
; /* show specific classq */
201 int dflag
; /* show i/f dropped packets */
202 int Fflag
; /* show i/f forwarded packets */
203 #if defined(__APPLE__)
204 int gflag
; /* show group (multicast) routing or stats */
206 int iflag
; /* show interfaces */
207 int lflag
; /* show routing table with use and ref */
208 int Lflag
; /* show size of listen queues */
209 int mflag
; /* show memory stats */
210 int nflag
; /* show addresses numerically */
211 static int pflag
; /* show given protocol */
212 int prioflag
= -1; /* show packet priority statistics */
213 int Rflag
; /* show reachability information */
214 int rflag
; /* show routing tables (or routing stats) */
215 int sflag
; /* show protocol statistics */
216 int Sflag
; /* show additional i/f link status */
217 int tflag
; /* show i/f watchdog timers */
218 int vflag
; /* more verbose */
219 int Wflag
; /* wide display */
220 int qflag
; /* classq stats display */
221 int Qflag
; /* opportunistic polling stats display */
222 int xflag
; /* show extended link-layer reachability information */
223 int zflag
; /* show only entries with non zero rtt metrics */
225 int cq
= -1; /* send classq index (-1 for all) */
226 int interval
; /* repeat interval for i/f stats */
228 char *interface
; /* desired i/f for stats, or NULL for all i/fs */
229 int unit
; /* unit number for above */
231 int af
; /* address family */
238 register struct protox
*tp
= NULL
; /* for printing cblocks & stats */
243 while ((ch
= getopt(argc
, argv
, "Aabc:dFf:gI:ikLlmnP:p:qQrRsStuvWw:xz")) != -1)
265 if (strcmp(optarg
, "ipx") == 0)
267 else if (strcmp(optarg
, "inet") == 0)
270 else if (strcmp(optarg
, "inet6") == 0)
274 else if (strcmp(optarg
, "pfkey") == 0)
277 else if (strcmp(optarg
, "unix") == 0)
279 else if (strcmp(optarg
, "systm") == 0)
282 errx(1, "%s: unknown address family", optarg
);
285 #if defined(__APPLE__)
294 for (cp
= interface
= optarg
; isalpha(*cp
); cp
++)
315 prioflag
= atoi(optarg
);
318 if ((tp
= name2protox(optarg
)) == NULL
) {
320 "%s: unknown or uninstrumented protocol",
356 interval
= atoi(optarg
);
373 #define BACKWARD_COMPATIBILITY
374 #ifdef BACKWARD_COMPATIBILITY
376 if (isdigit(**argv
)) {
377 interval
= atoi(*argv
);
390 if (iflag
&& !sflag
&& !Sflag
&& !gflag
&& !qflag
&& !Qflag
) {
404 if (qflag
|| Qflag
) {
405 if (interface
== NULL
) {
406 fprintf(stderr
, "%s statistics option "
407 "requires interface name\n", qflag
? "Queue" :
417 if (interface
== NULL
) {
418 fprintf(stderr
, "additional link status option"
419 " requires interface name\n");
421 print_link_status(interface
);
426 #if defined(__APPLE__)
434 printproto(tp
, tp
->pr_name
);
437 if (af
== AF_INET
|| af
== AF_UNSPEC
)
438 for (tp
= protox
; tp
->pr_name
; tp
++)
439 printproto(tp
, tp
->pr_name
);
441 if (af
== AF_INET6
|| af
== AF_UNSPEC
)
442 for (tp
= ip6protox
; tp
->pr_name
; tp
++)
443 printproto(tp
, tp
->pr_name
);
446 if (af
== PF_KEY
|| af
== AF_UNSPEC
)
447 for (tp
= pfkeyprotox
; tp
->pr_name
; tp
++)
448 printproto(tp
, tp
->pr_name
);
450 if ((af
== AF_UNIX
|| af
== AF_UNSPEC
) && !Lflag
&& !sflag
)
453 if ((af
== AF_SYSTEM
|| af
== AF_UNSPEC
) && !Lflag
)
454 for (tp
= systmprotox
; tp
->pr_name
; tp
++)
455 printproto(tp
, tp
->pr_name
);
457 if (af
== AF_UNSPEC
&& !Lflag
)
458 for (tp
= nstatprotox
; tp
->pr_name
; tp
++)
459 printproto(tp
, tp
->pr_name
);
460 #endif /* TARGET_OS_IPHONE */
462 if (af
== AF_UNSPEC
&& !Lflag
)
463 for (tp
= ipcprotox
; tp
->pr_name
; tp
++)
464 printproto(tp
, tp
->pr_name
);
466 if (af
== AF_UNSPEC
&& !Lflag
)
467 for (tp
= kernprotox
; tp
->pr_name
; tp
++)
468 printproto(tp
, tp
->pr_name
);
478 * Print out protocol statistics or control blocks (per sflag).
479 * If the interface was not specifically requested, and the symbol
480 * is not in the namelist, ignore this one.
484 register struct protox
*tp
;
487 void (*pr
)(uint32_t, char *, int);
491 if (iflag
&& !pflag
) {
493 intpr(tp
->pr_istats
);
495 printf("%s: no per-interface stats routine\n",
503 printf("%s: no stats routine\n",
507 off
= tp
->pr_protocol
;
513 printf("%s: no PCB routine\n", tp
->pr_name
);
516 off
= tp
->pr_protocol
;
519 if (sflag
&& iflag
&& pflag
)
520 intervalpr(pr
, off
, name
, af
);
522 (*pr
)(off
, name
, af
);
524 printf("### no stats for %s\n", name
);
531 return (n
> 1 ? "s" : "");
537 return (n
> 1 ? "es" : "");
543 return (n
> 1 ? "ies" : "y");
547 * Find the protox for the given "well-known" name.
549 static struct protox
*
550 knownname(char *name
)
552 struct protox
**tpp
, *tp
;
554 for (tpp
= protoprotox
; *tpp
; tpp
++)
555 for (tp
= *tpp
; tp
->pr_name
; tp
++)
556 if (strcmp(tp
->pr_name
, name
) == 0)
562 * Find the protox corresponding to name.
564 static struct protox
*
565 name2protox(char *name
)
568 char **alias
; /* alias from p->aliases */
572 * Try to find the name in the list of "well-known" names. If that
573 * fails, check if name is an alias for an Internet protocol.
575 if ((tp
= knownname(name
)) != NULL
)
578 setprotoent(1); /* make protocol lookup cheaper */
579 while ((p
= getprotoent()) != NULL
) {
580 /* assert: name not same as p->name */
581 for (alias
= p
->p_aliases
; *alias
; alias
++)
582 if (strcmp(name
, *alias
) == 0) {
584 return (knownname(p
->p_name
));
591 #define NETSTAT_USAGE "\
592 Usage: netstat [-AaLlnW] [-f address_family | -p protocol]\n\
593 netstat [-gilns] [-f address_family]\n\
594 netstat -i | -I interface [-w wait] [-abdgRtS]\n\
595 netstat -s [-s] [-f address_family | -p protocol] [-w wait]\n\
596 netstat -i | -I interface -s [-f address_family | -p protocol]\n\
598 netstat -r [-Aaln] [-f address_family]\n\
605 (void) fprintf(stderr
, "%s\n", NETSTAT_USAGE
);
617 (void) localtime_r(&now
, &tm
);
619 num_written
+= printf("%02d:%02d:%02d ", tm
.tm_hour
, tm
.tm_min
, tm
.tm_sec
);
621 return (num_written
);