]>
git.saurik.com Git - apple/network_cmds.git/blob - netstat.tproj/main.c
9d686078ff13464b95fc0f7aafa3647ac6a9de34
2 * Copyright (c) 2008 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>
71 #include <netinet/in.h>
72 #include <net/pfkeyv2.h>
86 #include <sys/types.h>
87 #include <sys/sysctl.h>
90 #include <TargetConditionals.h>
94 * ----------------------------------------------------------------------------
95 * "THE BEER-WARE LICENSE" (Revision 42):
96 * <phk@FreeBSD.org> wrote this file. As long as you retain this notice you
97 * can do whatever you want with this stuff. If we meet some day, and you think
98 * this stuff is worth it, you can buy me a beer in return. Poul-Henning Kamp
99 * ----------------------------------------------------------------------------
101 * $Id: main.c,v 1.8 2004/10/14 22:24:09 lindak Exp $
105 static struct nlist nl
[] = {
130 #define N_ESISSTAT 12
138 #define N_CLTPSTAT 16
158 #define N_ICMP6STAT 29
160 #define N_IPSECSTAT 30
162 #define N_IPSEC6STAT 31
164 #define N_PIM6STAT 32
166 #define N_MRT6PROTO 33
168 #define N_MRT6STAT 34
170 #define N_MF6CTABLE 35
172 #define N_MIF6TABLE 36
174 #define N_PFKEYSTAT 37
180 #define N_NMBCLUSTERS 40
191 u_char pr_index
; /* index into nlist of cb head */
192 u_char pr_sindex
; /* index into nlist of stat block */
193 u_char pr_wanted
; /* 1 if wanted, 0 otherwise */
194 void (*pr_cblocks
)(uint32_t, char *, int);
195 /* control blocks printing routine */
196 void (*pr_stats
)(uint32_t, char *, int);
197 /* statistics printing routine */
198 void (*pr_istats
)(char *); /* per/if statistics printing routine */
199 char *pr_name
; /* well-known name */
200 int pr_usesysctl
; /* true if we use sysctl, not kvm */
202 { -1, -1, 1, protopr
,
203 tcp_stats
, NULL
, "tcp", IPPROTO_TCP
},
204 { -1, -1, 1, protopr
,
205 udp_stats
, NULL
, "udp", IPPROTO_UDP
},
206 { -1, -1, 1, protopr
,
207 NULL
, NULL
, "divert",IPPROTO_DIVERT
},
208 { -1, -1, 1, protopr
,
209 ip_stats
, NULL
, "ip", IPPROTO_RAW
},
210 { -1, -1, 1, protopr
,
211 icmp_stats
, NULL
, "icmp", IPPROTO_ICMP
},
212 { -1, -1, 1, protopr
,
213 igmp_stats
, NULL
, "igmp", IPPROTO_IGMP
},
216 ipsec_stats
, NULL
, "ipsec", IPPROTO_ESP
},
223 struct protox ip6protox
[] = {
224 { -1, -1, 1, protopr
,
225 tcp_stats
, NULL
, "tcp", IPPROTO_TCP
},
226 { -1, -1, 1, protopr
,
227 udp_stats
, NULL
, "udp", IPPROTO_UDP
},
228 { -1, N_IP6STAT
, 1, protopr
,
229 ip6_stats
, ip6_ifstats
, "ip6", IPPROTO_RAW
},
230 { -1, N_ICMP6STAT
, 1, protopr
,
231 icmp6_stats
, icmp6_ifstats
, "icmp6",IPPROTO_ICMPV6
},
233 { -1, N_IPSEC6STAT
, 1, 0,
234 ipsec_stats
, NULL
, "ipsec6",IPPROTO_ESP
},
237 { -1, N_PIM6STAT
, 1, 0,
238 pim6_stats
, NULL
, "pim6", 0 },
241 rip6_stats
, NULL
, "rip6", IPPROTO_RAW
},
248 struct protox pfkeyprotox
[] = {
249 { -1, N_PFKEYSTAT
, 1, 0,
250 pfkey_stats
, NULL
, "pfkey", PF_KEY_V2
},
256 struct protox
*protoprotox
[] = {
266 static void printproto (struct protox
*, char *);
267 static void usage (void);
268 static struct protox
*name2protox (char *);
269 static struct protox
*knownname (char *);
271 extern void _serv_cache_close();
274 int Aflag
; /* show addresses of protocol control block */
275 int aflag
; /* show all sockets (including servers) */
276 int bflag
; /* show i/f total bytes in/out */
277 int dflag
; /* show i/f dropped packets */
278 #if defined(__APPLE__) && !TARGET_OS_EMBEDDED
279 int gflag
; /* show group (multicast) routing or stats */
281 int iflag
; /* show interfaces */
282 int lflag
; /* show routing table with use and ref */
283 int Lflag
; /* show size of listen queues */
284 int mflag
; /* show memory stats */
285 int nflag
; /* show addresses numerically */
286 static int pflag
; /* show given protocol */
287 int rflag
; /* show routing tables (or routing stats) */
288 int sflag
; /* show protocol statistics */
289 int tflag
; /* show i/f watchdog timers */
290 int Wflag
; /* wide display */
292 int interval
; /* repeat interval for i/f stats */
294 char *interface
; /* desired i/f for stats, or NULL for all i/fs */
295 int unit
; /* unit number for above */
297 int af
; /* address family */
304 register struct protox
*tp
= NULL
; /* for printing cblocks & stats */
309 while ((ch
= getopt(argc
, argv
, "Aabdf:gI:iLlmnp:rRstuWw:")) != -1)
324 if (strcmp(optarg
, "ipx") == 0)
326 else if (strcmp(optarg
, "inet") == 0)
329 else if (strcmp(optarg
, "inet6") == 0)
333 else if (strcmp(optarg
, "pfkey") == 0)
336 else if (strcmp(optarg
, "unix") == 0)
339 errx(1, "%s: unknown address family", optarg
);
342 #if defined(__APPLE__) && !TARGET_OS_EMBEDDED
351 for (cp
= interface
= optarg
; isalpha(*cp
); cp
++)
372 if ((tp
= name2protox(optarg
)) == NULL
) {
374 "%s: unknown or uninstrumented protocol",
395 interval
= atoi(optarg
);
405 #define BACKWARD_COMPATIBILITY
406 #ifdef BACKWARD_COMPATIBILITY
408 if (isdigit(**argv
)) {
409 interval
= atoi(*argv
);
422 if (iflag
&& !sflag
) {
430 routepr(nl
[N_RTREE
].n_value
);
433 #if defined(__APPLE__) && !TARGET_OS_EMBEDDED
436 if (af
== AF_INET
|| af
== AF_UNSPEC
)
439 if (af
== AF_INET6
|| af
== AF_UNSPEC
)
443 if (af
== AF_INET
|| af
== AF_UNSPEC
)
446 if (af
== AF_INET6
|| af
== AF_UNSPEC
)
456 printproto(tp
, tp
->pr_name
);
459 if (af
== AF_INET
|| af
== AF_UNSPEC
)
460 for (tp
= protox
; tp
->pr_name
; tp
++)
461 printproto(tp
, tp
->pr_name
);
463 if (af
== AF_INET6
|| af
== AF_UNSPEC
)
464 for (tp
= ip6protox
; tp
->pr_name
; tp
++)
465 printproto(tp
, tp
->pr_name
);
468 if (af
== PF_KEY
|| af
== AF_UNSPEC
)
469 for (tp
= pfkeyprotox
; tp
->pr_name
; tp
++)
470 printproto(tp
, tp
->pr_name
);
472 if ((af
== AF_UNIX
|| af
== AF_UNSPEC
) && !Lflag
&& !sflag
)
481 * Print out protocol statistics or control blocks (per sflag).
482 * If the interface was not specifically requested, and the symbol
483 * is not in the namelist, ignore this one.
487 register struct protox
*tp
;
490 void (*pr
)(uint32_t, char *, int);
494 if (iflag
&& !pflag
) {
496 intpr(tp
->pr_istats
);
498 printf("%s: no per-interface stats routine\n",
506 printf("%s: no stats routine\n",
510 off
= tp
->pr_usesysctl
? tp
->pr_usesysctl
511 : nl
[tp
->pr_sindex
].n_value
;
517 printf("%s: no PCB routine\n", tp
->pr_name
);
520 off
= tp
->pr_usesysctl
? tp
->pr_usesysctl
521 : nl
[tp
->pr_index
].n_value
;
523 if (pr
!= NULL
&& (off
|| af
!= AF_UNSPEC
)) {
524 if (sflag
&& iflag
&& pflag
)
525 intervalpr(pr
, off
, name
, af
);
527 (*pr
)(off
, name
, af
);
529 printf("### no stats for %s\n", name
);
536 return (n
!= 1 ? "s" : "");
542 return (n
!= 1 ? "es" : "");
546 * Find the protox for the given "well-known" name.
548 static struct protox
*
549 knownname(char *name
)
551 struct protox
**tpp
, *tp
;
553 for (tpp
= protoprotox
; *tpp
; tpp
++)
554 for (tp
= *tpp
; tp
->pr_name
; tp
++)
555 if (strcmp(tp
->pr_name
, name
) == 0)
561 * Find the protox corresponding to name.
563 static struct protox
*
564 name2protox(char *name
)
567 char **alias
; /* alias from p->aliases */
571 * Try to find the name in the list of "well-known" names. If that
572 * fails, check if name is an alias for an Internet protocol.
574 if ((tp
= knownname(name
)) != NULL
)
577 setprotoent(1); /* make protocol lookup cheaper */
578 while ((p
= getprotoent()) != NULL
) {
579 /* assert: name not same as p->name */
580 for (alias
= p
->p_aliases
; *alias
; alias
++)
581 if (strcmp(name
, *alias
) == 0) {
583 return (knownname(p
->p_name
));
590 #define NETSTAT_USAGE "\
591 Usage: netstat [-AaLlnW] [-f address_family | -p protocol]\n\
592 netstat [-gilns] [-f address_family]\n\
593 netstat -i | -I interface [-w wait] [-abdgt]\n\
594 netstat -s [-s] [-f address_family | -p protocol] [-w wait]\n\
595 netstat -i | -I interface -s [-f address_family | -p protocol]\n\
597 netstat -r [-Aaln] [-f address_family]\n\
604 (void) fprintf(stderr
, "%s\n", NETSTAT_USAGE
);