netstat: fix -r display in usage text
[oweals/busybox.git] / networking / netstat.c
1 /* vi: set sw=4 ts=4: */
2 /*
3  * Mini netstat implementation(s) for busybox
4  * based in part on the netstat implementation from net-tools.
5  *
6  * Copyright (C) 2002 by Bart Visscher <magick@linux-fan.com>
7  *
8  * 2002-04-20
9  * IPV6 support added by Bart Visscher <magick@linux-fan.com>
10  *
11  * 2008-07-10
12  * optional '-p' flag support ported from net-tools by G. Somlo <somlo@cmu.edu>
13  *
14  * Licensed under GPLv2 or later, see file LICENSE in this source tree.
15  */
16
17 #include "libbb.h"
18 #include "inet_common.h"
19
20 //usage:#define netstat_trivial_usage
21 //usage:       "[-"IF_ROUTE("r")"al] [-tuwx] [-en"IF_FEATURE_NETSTAT_WIDE("W")IF_FEATURE_NETSTAT_PRG("p")"]"
22 //usage:#define netstat_full_usage "\n\n"
23 //usage:       "Display networking information\n"
24 //usage:     "\nOptions:"
25 //usage:        IF_ROUTE(
26 //usage:     "\n        -r      Routing table"
27 //usage:        )
28 //usage:     "\n        -a      All sockets"
29 //usage:     "\n        -l      Listening sockets"
30 //usage:     "\n                Else: connected sockets"
31 //usage:     "\n        -t      TCP sockets"
32 //usage:     "\n        -u      UDP sockets"
33 //usage:     "\n        -w      Raw sockets"
34 //usage:     "\n        -x      Unix sockets"
35 //usage:     "\n                Else: all socket types"
36 //usage:     "\n        -e      Other/more information"
37 //usage:     "\n        -n      Don't resolve names"
38 //usage:        IF_FEATURE_NETSTAT_WIDE(
39 //usage:     "\n        -W      Wide display"
40 //usage:        )
41 //usage:        IF_FEATURE_NETSTAT_PRG(
42 //usage:     "\n        -p      Show PID/program name for sockets"
43 //usage:        )
44
45 #define NETSTAT_OPTS "laentuwx" \
46         IF_ROUTE(               "r") \
47         IF_FEATURE_NETSTAT_WIDE("W") \
48         IF_FEATURE_NETSTAT_PRG( "p")
49
50 enum {
51         OPT_sock_listen = 1 << 0, // l
52         OPT_sock_all    = 1 << 1, // a
53         OPT_extended    = 1 << 2, // e
54         OPT_noresolve   = 1 << 3, // n
55         OPT_sock_tcp    = 1 << 4, // t
56         OPT_sock_udp    = 1 << 5, // u
57         OPT_sock_raw    = 1 << 6, // w
58         OPT_sock_unix   = 1 << 7, // x
59         OPTBIT_x        = 7,
60         IF_ROUTE(               OPTBIT_ROUTE,)
61         IF_FEATURE_NETSTAT_WIDE(OPTBIT_WIDE ,)
62         IF_FEATURE_NETSTAT_PRG( OPTBIT_PRG  ,)
63         OPT_route       = IF_ROUTE(               (1 << OPTBIT_ROUTE)) + 0, // r
64         OPT_wide        = IF_FEATURE_NETSTAT_WIDE((1 << OPTBIT_WIDE )) + 0, // W
65         OPT_prg         = IF_FEATURE_NETSTAT_PRG( (1 << OPTBIT_PRG  )) + 0, // p
66 };
67
68 #define NETSTAT_CONNECTED 0x01
69 #define NETSTAT_LISTENING 0x02
70 #define NETSTAT_NUMERIC   0x04
71 /* Must match getopt32 option string */
72 #define NETSTAT_TCP       0x10
73 #define NETSTAT_UDP       0x20
74 #define NETSTAT_RAW       0x40
75 #define NETSTAT_UNIX      0x80
76 #define NETSTAT_ALLPROTO  (NETSTAT_TCP|NETSTAT_UDP|NETSTAT_RAW|NETSTAT_UNIX)
77
78
79 enum {
80         TCP_ESTABLISHED = 1,
81         TCP_SYN_SENT,
82         TCP_SYN_RECV,
83         TCP_FIN_WAIT1,
84         TCP_FIN_WAIT2,
85         TCP_TIME_WAIT,
86         TCP_CLOSE,
87         TCP_CLOSE_WAIT,
88         TCP_LAST_ACK,
89         TCP_LISTEN,
90         TCP_CLOSING, /* now a valid state */
91 };
92
93 static const char *const tcp_state[] = {
94         "",
95         "ESTABLISHED",
96         "SYN_SENT",
97         "SYN_RECV",
98         "FIN_WAIT1",
99         "FIN_WAIT2",
100         "TIME_WAIT",
101         "CLOSE",
102         "CLOSE_WAIT",
103         "LAST_ACK",
104         "LISTEN",
105         "CLOSING"
106 };
107
108 typedef enum {
109         SS_FREE = 0,     /* not allocated                */
110         SS_UNCONNECTED,  /* unconnected to any socket    */
111         SS_CONNECTING,   /* in process of connecting     */
112         SS_CONNECTED,    /* connected to socket          */
113         SS_DISCONNECTING /* in process of disconnecting  */
114 } socket_state;
115
116 #define SO_ACCEPTCON (1<<16)    /* performed a listen           */
117 #define SO_WAITDATA  (1<<17)    /* wait data to read            */
118 #define SO_NOSPACE   (1<<18)    /* no space to write            */
119
120 /* Standard printout size */
121 #define PRINT_IP_MAX_SIZE           23
122 #define PRINT_NET_CONN              "%s   %6ld %6ld %-23s %-23s %-12s"
123 #define PRINT_NET_CONN_HEADER       "\nProto Recv-Q Send-Q %-23s %-23s State       "
124
125 /* When there are IPv6 connections the IPv6 addresses will be
126  * truncated to none-recognition. The '-W' option makes the
127  * address columns wide enough to accomodate for longest possible
128  * IPv6 addresses, i.e. addresses of the form
129  * xxxx:xxxx:xxxx:xxxx:xxxx:xxxx:ddd.ddd.ddd.ddd
130  */
131 #define PRINT_IP_MAX_SIZE_WIDE      51  /* INET6_ADDRSTRLEN + 5 for the port number */
132 #define PRINT_NET_CONN_WIDE         "%s   %6ld %6ld %-51s %-51s %-12s"
133 #define PRINT_NET_CONN_HEADER_WIDE  "\nProto Recv-Q Send-Q %-51s %-51s State       "
134
135 #define PROGNAME_WIDTH     20
136 #define PROGNAME_WIDTH_STR "20"
137 /* PROGNAME_WIDTH chars: 12345678901234567890 */
138 #define PROGNAME_BANNER "PID/Program name    "
139
140 struct prg_node {
141         struct prg_node *next;
142         long inode;
143         char name[PROGNAME_WIDTH];
144 };
145
146 #define PRG_HASH_SIZE 211
147
148 struct globals {
149         const char *net_conn_line;
150         smallint flags;
151 #if ENABLE_FEATURE_NETSTAT_PRG
152         smallint prg_cache_loaded;
153         struct prg_node *prg_hash[PRG_HASH_SIZE];
154 #endif
155 };
156 #define G (*ptr_to_globals)
157 #define flags            (G.flags           )
158 #define net_conn_line    (G.net_conn_line   )
159 #define prg_hash         (G.prg_hash        )
160 #define prg_cache_loaded (G.prg_cache_loaded)
161 #define INIT_G() do { \
162         SET_PTR_TO_GLOBALS(xzalloc(sizeof(G))); \
163         flags = NETSTAT_CONNECTED | NETSTAT_ALLPROTO; \
164         net_conn_line = PRINT_NET_CONN; \
165 } while (0)
166
167
168 #if ENABLE_FEATURE_NETSTAT_PRG
169
170 /* Deliberately truncating long to unsigned *int* */
171 #define PRG_HASHIT(x) ((unsigned)(x) % PRG_HASH_SIZE)
172
173 #define print_progname_banner() do { \
174         if (option_mask32 & OPT_prg) printf(PROGNAME_BANNER); \
175 } while (0)
176
177 static void prg_cache_add(long inode, char *name)
178 {
179         unsigned hi = PRG_HASHIT(inode);
180         struct prg_node **pnp, *pn;
181
182         prg_cache_loaded = 2;
183         for (pnp = prg_hash + hi; (pn = *pnp) != NULL; pnp = &pn->next) {
184                 if (pn->inode == inode) {
185                         /* Some warning should be appropriate here
186                            as we got multiple processes for one i-node */
187                         return;
188                 }
189         }
190         *pnp = xzalloc(sizeof(struct prg_node));
191         pn = *pnp;
192         pn->inode = inode;
193         safe_strncpy(pn->name, name, PROGNAME_WIDTH);
194 }
195
196 static const char *prg_cache_get(long inode)
197 {
198         unsigned hi = PRG_HASHIT(inode);
199         struct prg_node *pn;
200
201         for (pn = prg_hash[hi]; pn; pn = pn->next)
202                 if (pn->inode == inode)
203                         return pn->name;
204         return "-";
205 }
206
207 #if ENABLE_FEATURE_CLEAN_UP
208 static void prg_cache_clear(void)
209 {
210         struct prg_node **pnp, *pn;
211
212         for (pnp = prg_hash; pnp < prg_hash + PRG_HASH_SIZE; pnp++) {
213                 while ((pn = *pnp) != NULL) {
214                         *pnp = pn->next;
215                         free(pn);
216                 }
217         }
218 }
219 #else
220 #define prg_cache_clear() ((void)0)
221 #endif
222
223 static long extract_socket_inode(const char *lname)
224 {
225         long inode = -1;
226
227         if (strncmp(lname, "socket:[", sizeof("socket:[")-1) == 0) {
228                 /* "socket:[12345]", extract the "12345" as inode */
229                 inode = bb_strtol(lname + sizeof("socket:[")-1, (char**)&lname, 0);
230                 if (*lname != ']')
231                         inode = -1;
232         } else if (strncmp(lname, "[0000]:", sizeof("[0000]:")-1) == 0) {
233                 /* "[0000]:12345", extract the "12345" as inode */
234                 inode = bb_strtol(lname + sizeof("[0000]:")-1, NULL, 0);
235                 if (errno) /* not NUL terminated? */
236                         inode = -1;
237         }
238
239 #if 0 /* bb_strtol returns all-ones bit pattern on ERANGE anyway */
240         if (errno == ERANGE)
241                 inode = -1;
242 #endif
243         return inode;
244 }
245
246 static int FAST_FUNC file_act(const char *fileName,
247                 struct stat *statbuf UNUSED_PARAM,
248                 void *pid_slash_progname,
249                 int depth UNUSED_PARAM)
250 {
251         char *linkname;
252         long inode;
253
254         linkname = xmalloc_readlink(fileName);
255         if (linkname != NULL) {
256                 inode = extract_socket_inode(linkname);
257                 free(linkname);
258                 if (inode >= 0)
259                         prg_cache_add(inode, (char *)pid_slash_progname);
260         }
261         return TRUE;
262 }
263
264 static int FAST_FUNC dir_act(const char *fileName,
265                 struct stat *statbuf UNUSED_PARAM,
266                 void *userData UNUSED_PARAM,
267                 int depth)
268 {
269         const char *pid;
270         char *p, *pid_slash_progname;
271         char cmdline_buf[512];
272         int i;
273
274         if (depth == 0) /* "/proc" itself */
275                 return TRUE; /* continue looking one level below /proc */
276
277         pid = fileName + sizeof("/proc/")-1; /* point after "/proc/" */
278         if (!isdigit(pid[0])) /* skip /proc entries which aren't processes */
279                 return SKIP;
280
281         p = concat_path_file(fileName, "cmdline"); /* "/proc/PID/cmdline" */
282         i = open_read_close(p, cmdline_buf, sizeof(cmdline_buf) - 1);
283         free(p);
284         if (i < 0)
285                 return FALSE;
286         cmdline_buf[i] = '\0';
287
288         /* go through all files in /proc/PID/fd */
289         pid_slash_progname = concat_path_file(pid, bb_basename(cmdline_buf)); /* "PID/argv0" */
290         p = concat_path_file(fileName, "fd");
291         i = recursive_action(p, ACTION_RECURSE | ACTION_QUIET,
292                                 file_act, NULL, (void *)pid_slash_progname, 0);
293         free(p);
294         free(pid_slash_progname);
295
296         if (!i)
297                 return FALSE; /* signal permissions error to caller */
298
299         return SKIP; /* caller should not recurse further into this dir */
300 }
301
302 static void prg_cache_load(void)
303 {
304         int load_ok;
305
306         prg_cache_loaded = 1;
307         load_ok = recursive_action("/proc", ACTION_RECURSE | ACTION_QUIET,
308                                 NULL, dir_act, NULL, 0);
309         if (load_ok)
310                 return;
311
312         if (prg_cache_loaded == 1)
313                 bb_error_msg("can't scan /proc - are you root?");
314         else
315                 bb_error_msg("showing only processes with your user ID");
316 }
317
318 #else
319
320 #define prg_cache_clear()       ((void)0)
321 #define print_progname_banner() ((void)0)
322
323 #endif //ENABLE_FEATURE_NETSTAT_PRG
324
325
326 #if ENABLE_FEATURE_IPV6
327 static void build_ipv6_addr(char* local_addr, struct sockaddr_in6* localaddr)
328 {
329         char addr6[INET6_ADDRSTRLEN];
330         struct in6_addr in6;
331
332         sscanf(local_addr, "%08X%08X%08X%08X",
333                    &in6.s6_addr32[0], &in6.s6_addr32[1],
334                    &in6.s6_addr32[2], &in6.s6_addr32[3]);
335         inet_ntop(AF_INET6, &in6, addr6, sizeof(addr6));
336         inet_pton(AF_INET6, addr6, &localaddr->sin6_addr);
337
338         localaddr->sin6_family = AF_INET6;
339 }
340 #endif
341
342 static void build_ipv4_addr(char* local_addr, struct sockaddr_in* localaddr)
343 {
344         sscanf(local_addr, "%X", &localaddr->sin_addr.s_addr);
345         localaddr->sin_family = AF_INET;
346 }
347
348 static const char *get_sname(int port, const char *proto, int numeric)
349 {
350         if (!port)
351                 return "*";
352         if (!numeric) {
353                 struct servent *se = getservbyport(port, proto);
354                 if (se)
355                         return se->s_name;
356         }
357         /* hummm, we may return static buffer here!! */
358         return itoa(ntohs(port));
359 }
360
361 static char *ip_port_str(struct sockaddr *addr, int port, const char *proto, int numeric)
362 {
363         char *host, *host_port;
364
365         /* Code which used "*" for INADDR_ANY is removed: it's ambiguous
366          * in IPv6, while "0.0.0.0" is not. */
367
368         host = numeric ? xmalloc_sockaddr2dotted_noport(addr)
369                        : xmalloc_sockaddr2host_noport(addr);
370
371         host_port = xasprintf("%s:%s", host, get_sname(htons(port), proto, numeric));
372         free(host);
373         return host_port;
374 }
375
376 struct inet_params {
377         int local_port, rem_port, state, uid;
378         union {
379                 struct sockaddr     sa;
380                 struct sockaddr_in  sin;
381 #if ENABLE_FEATURE_IPV6
382                 struct sockaddr_in6 sin6;
383 #endif
384         } localaddr, remaddr;
385         unsigned long rxq, txq, inode;
386 };
387
388 static int scan_inet_proc_line(struct inet_params *param, char *line)
389 {
390         int num;
391         char local_addr[64], rem_addr[64];
392
393         num = sscanf(line,
394                         "%*d: %64[0-9A-Fa-f]:%X "
395                         "%64[0-9A-Fa-f]:%X %X "
396                         "%lX:%lX %*X:%*X "
397                         "%*X %d %*d %ld ",
398                         local_addr, &param->local_port,
399                         rem_addr, &param->rem_port, &param->state,
400                         &param->txq, &param->rxq,
401                         &param->uid, &param->inode);
402         if (num < 9) {
403                 return 1; /* error */
404         }
405
406         if (strlen(local_addr) > 8) {
407 #if ENABLE_FEATURE_IPV6
408                 build_ipv6_addr(local_addr, &param->localaddr.sin6);
409                 build_ipv6_addr(rem_addr, &param->remaddr.sin6);
410 #endif
411         } else {
412                 build_ipv4_addr(local_addr, &param->localaddr.sin);
413                 build_ipv4_addr(rem_addr, &param->remaddr.sin);
414         }
415         return 0;
416 }
417
418 static void print_inet_line(struct inet_params *param,
419                 const char *state_str, const char *proto, int is_connected)
420 {
421         if ((is_connected && (flags & NETSTAT_CONNECTED))
422          || (!is_connected && (flags & NETSTAT_LISTENING))
423         ) {
424                 char *l = ip_port_str(
425                                 &param->localaddr.sa, param->local_port,
426                                 proto, flags & NETSTAT_NUMERIC);
427                 char *r = ip_port_str(
428                                 &param->remaddr.sa, param->rem_port,
429                                 proto, flags & NETSTAT_NUMERIC);
430                 printf(net_conn_line,
431                         proto, param->rxq, param->txq, l, r, state_str);
432 #if ENABLE_FEATURE_NETSTAT_PRG
433                 if (option_mask32 & OPT_prg)
434                         printf("%."PROGNAME_WIDTH_STR"s", prg_cache_get(param->inode));
435 #endif
436                 bb_putchar('\n');
437                 free(l);
438                 free(r);
439         }
440 }
441
442 static int FAST_FUNC tcp_do_one(char *line)
443 {
444         struct inet_params param;
445
446         if (scan_inet_proc_line(&param, line))
447                 return 1;
448
449         print_inet_line(&param, tcp_state[param.state], "tcp", param.rem_port);
450         return 0;
451 }
452
453 #if ENABLE_FEATURE_IPV6
454 # define NOT_NULL_ADDR(A) ( \
455         ( (A.sa.sa_family == AF_INET6) \
456           && (A.sin6.sin6_addr.s6_addr32[0] | A.sin6.sin6_addr.s6_addr32[1] | \
457               A.sin6.sin6_addr.s6_addr32[2] | A.sin6.sin6_addr.s6_addr32[3])  \
458         ) || ( \
459           (A.sa.sa_family == AF_INET) \
460           && A.sin.sin_addr.s_addr != 0 \
461         ) \
462 )
463 #else
464 # define NOT_NULL_ADDR(A) (A.sin.sin_addr.s_addr)
465 #endif
466
467 static int FAST_FUNC udp_do_one(char *line)
468 {
469         int have_remaddr;
470         const char *state_str;
471         struct inet_params param;
472
473         if (scan_inet_proc_line(&param, line))
474                 return 1;
475
476         state_str = "UNKNOWN";
477         switch (param.state) {
478         case TCP_ESTABLISHED:
479                 state_str = "ESTABLISHED";
480                 break;
481         case TCP_CLOSE:
482                 state_str = "";
483                 break;
484         }
485
486         have_remaddr = NOT_NULL_ADDR(param.remaddr);
487         print_inet_line(&param, state_str, "udp", have_remaddr);
488         return 0;
489 }
490
491 static int FAST_FUNC raw_do_one(char *line)
492 {
493         int have_remaddr;
494         struct inet_params param;
495
496         if (scan_inet_proc_line(&param, line))
497                 return 1;
498
499         have_remaddr = NOT_NULL_ADDR(param.remaddr);
500         print_inet_line(&param, itoa(param.state), "raw", have_remaddr);
501         return 0;
502 }
503
504 static int FAST_FUNC unix_do_one(char *line)
505 {
506         unsigned long refcnt, proto, unix_flags;
507         unsigned long inode;
508         int type, state;
509         int num, path_ofs;
510         const char *ss_proto, *ss_state, *ss_type;
511         char ss_flags[32];
512
513         /* 2.6.15 may report lines like "... @/tmp/fam-user-^@^@^@^@^@^@^@..."
514          * Other users report long lines filled by NUL bytes.
515          * (those ^@ are NUL bytes too). We see them as empty lines. */
516         if (!line[0])
517                 return 0;
518
519         path_ofs = 0; /* paranoia */
520         num = sscanf(line, "%*p: %lX %lX %lX %X %X %lu %n",
521                         &refcnt, &proto, &unix_flags, &type, &state, &inode, &path_ofs);
522         if (num < 6) {
523                 return 1; /* error */
524         }
525         if ((flags & (NETSTAT_LISTENING|NETSTAT_CONNECTED)) != (NETSTAT_LISTENING|NETSTAT_CONNECTED)) {
526                 if ((state == SS_UNCONNECTED) && (unix_flags & SO_ACCEPTCON)) {
527                         if (!(flags & NETSTAT_LISTENING))
528                                 return 0;
529                 } else {
530                         if (!(flags & NETSTAT_CONNECTED))
531                                 return 0;
532                 }
533         }
534
535         switch (proto) {
536                 case 0:
537                         ss_proto = "unix";
538                         break;
539                 default:
540                         ss_proto = "??";
541         }
542
543         switch (type) {
544                 case SOCK_STREAM:
545                         ss_type = "STREAM";
546                         break;
547                 case SOCK_DGRAM:
548                         ss_type = "DGRAM";
549                         break;
550                 case SOCK_RAW:
551                         ss_type = "RAW";
552                         break;
553                 case SOCK_RDM:
554                         ss_type = "RDM";
555                         break;
556                 case SOCK_SEQPACKET:
557                         ss_type = "SEQPACKET";
558                         break;
559                 default:
560                         ss_type = "UNKNOWN";
561         }
562
563         switch (state) {
564                 case SS_FREE:
565                         ss_state = "FREE";
566                         break;
567                 case SS_UNCONNECTED:
568                         /*
569                          * Unconnected sockets may be listening
570                          * for something.
571                          */
572                         if (unix_flags & SO_ACCEPTCON) {
573                                 ss_state = "LISTENING";
574                         } else {
575                                 ss_state = "";
576                         }
577                         break;
578                 case SS_CONNECTING:
579                         ss_state = "CONNECTING";
580                         break;
581                 case SS_CONNECTED:
582                         ss_state = "CONNECTED";
583                         break;
584                 case SS_DISCONNECTING:
585                         ss_state = "DISCONNECTING";
586                         break;
587                 default:
588                         ss_state = "UNKNOWN";
589         }
590
591         strcpy(ss_flags, "[ ");
592         if (unix_flags & SO_ACCEPTCON)
593                 strcat(ss_flags, "ACC ");
594         if (unix_flags & SO_WAITDATA)
595                 strcat(ss_flags, "W ");
596         if (unix_flags & SO_NOSPACE)
597                 strcat(ss_flags, "N ");
598         strcat(ss_flags, "]");
599
600         printf("%-5s %-6ld %-11s %-10s %-13s %6lu ",
601                 ss_proto, refcnt, ss_flags, ss_type, ss_state, inode
602                 );
603
604 #if ENABLE_FEATURE_NETSTAT_PRG
605         if (option_mask32 & OPT_prg)
606                 printf("%-"PROGNAME_WIDTH_STR"s", prg_cache_get(inode));
607 #endif
608
609         /* TODO: currently we stop at first NUL byte. Is it a problem? */
610         line += path_ofs;
611         *strchrnul(line, '\n') = '\0';
612         while (*line)
613                 fputc_printable(*line++, stdout);
614         bb_putchar('\n');
615         return 0;
616 }
617
618 static void do_info(const char *file, int FAST_FUNC (*proc)(char *))
619 {
620         int lnr;
621         FILE *procinfo;
622         char *buffer;
623
624         /* _stdin is just to save "r" param */
625         procinfo = fopen_or_warn_stdin(file);
626         if (procinfo == NULL) {
627                 return;
628         }
629         lnr = 0;
630         /* Why xmalloc_fgets_str? because it doesn't stop on NULs */
631         while ((buffer = xmalloc_fgets_str(procinfo, "\n")) != NULL) {
632                 /* line 0 is skipped */
633                 if (lnr && proc(buffer))
634                         bb_error_msg("%s: bogus data on line %d", file, lnr + 1);
635                 lnr++;
636                 free(buffer);
637         }
638         fclose(procinfo);
639 }
640
641 int netstat_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
642 int netstat_main(int argc UNUSED_PARAM, char **argv)
643 {
644         const char *net_conn_line_header = PRINT_NET_CONN_HEADER;
645         unsigned opt;
646
647         INIT_G();
648
649         /* Option string must match NETSTAT_xxx constants */
650         opt = getopt32(argv, NETSTAT_OPTS);
651         if (opt & OPT_sock_listen) { // -l
652                 flags &= ~NETSTAT_CONNECTED;
653                 flags |= NETSTAT_LISTENING;
654         }
655         if (opt & OPT_sock_all) flags |= NETSTAT_LISTENING | NETSTAT_CONNECTED; // -a
656         //if (opt & OPT_extended) // -e
657         if (opt & OPT_noresolve) flags |= NETSTAT_NUMERIC; // -n
658         //if (opt & OPT_sock_tcp) // -t: NETSTAT_TCP
659         //if (opt & OPT_sock_udp) // -u: NETSTAT_UDP
660         //if (opt & OPT_sock_raw) // -w: NETSTAT_RAW
661         //if (opt & OPT_sock_unix) // -x: NETSTAT_UNIX
662 #if ENABLE_ROUTE
663         if (opt & OPT_route) { // -r
664                 bb_displayroutes(flags & NETSTAT_NUMERIC, !(opt & OPT_extended));
665                 return 0;
666         }
667 #endif
668         if (opt & OPT_wide) { // -W
669                 net_conn_line = PRINT_NET_CONN_WIDE;
670                 net_conn_line_header = PRINT_NET_CONN_HEADER_WIDE;
671         }
672 #if ENABLE_FEATURE_NETSTAT_PRG
673         if (opt & OPT_prg) { // -p
674                 prg_cache_load();
675         }
676 #endif
677
678         opt &= NETSTAT_ALLPROTO;
679         if (opt) {
680                 flags &= ~NETSTAT_ALLPROTO;
681                 flags |= opt;
682         }
683         if (flags & (NETSTAT_TCP|NETSTAT_UDP|NETSTAT_RAW)) {
684                 printf("Active Internet connections "); /* xxx */
685
686                 if ((flags & (NETSTAT_LISTENING|NETSTAT_CONNECTED)) == (NETSTAT_LISTENING|NETSTAT_CONNECTED))
687                         printf("(servers and established)");
688                 else if (flags & NETSTAT_LISTENING)
689                         printf("(only servers)");
690                 else
691                         printf("(w/o servers)");
692                 printf(net_conn_line_header, "Local Address", "Foreign Address");
693                 print_progname_banner();
694                 bb_putchar('\n');
695         }
696         if (flags & NETSTAT_TCP) {
697                 do_info("/proc/net/tcp", tcp_do_one);
698 #if ENABLE_FEATURE_IPV6
699                 do_info("/proc/net/tcp6", tcp_do_one);
700 #endif
701         }
702         if (flags & NETSTAT_UDP) {
703                 do_info("/proc/net/udp", udp_do_one);
704 #if ENABLE_FEATURE_IPV6
705                 do_info("/proc/net/udp6", udp_do_one);
706 #endif
707         }
708         if (flags & NETSTAT_RAW) {
709                 do_info("/proc/net/raw", raw_do_one);
710 #if ENABLE_FEATURE_IPV6
711                 do_info("/proc/net/raw6", raw_do_one);
712 #endif
713         }
714         if (flags & NETSTAT_UNIX) {
715                 printf("Active UNIX domain sockets ");
716                 if ((flags & (NETSTAT_LISTENING|NETSTAT_CONNECTED)) == (NETSTAT_LISTENING|NETSTAT_CONNECTED))
717                         printf("(servers and established)");
718                 else if (flags & NETSTAT_LISTENING)
719                         printf("(only servers)");
720                 else
721                         printf("(w/o servers)");
722                 printf("\nProto RefCnt Flags       Type       State         I-Node ");
723                 print_progname_banner();
724                 printf("Path\n");
725                 do_info("/proc/net/unix", unix_do_one);
726         }
727         prg_cache_clear();
728         return 0;
729 }