1*cf5a6c84SAndroid Build Coastguard Worker /* netstat.c - Display Linux networking subsystem.
2*cf5a6c84SAndroid Build Coastguard Worker *
3*cf5a6c84SAndroid Build Coastguard Worker * Copyright 2012 Ranjan Kumar <[email protected]>
4*cf5a6c84SAndroid Build Coastguard Worker * Copyright 2013 Kyungwan Han <[email protected]>
5*cf5a6c84SAndroid Build Coastguard Worker *
6*cf5a6c84SAndroid Build Coastguard Worker * Not in SUSv4.
7*cf5a6c84SAndroid Build Coastguard Worker *
8*cf5a6c84SAndroid Build Coastguard Worker USE_NETSTAT(NEWTOY(netstat, "pWrxwutneal", TOYFLAG_BIN))
9*cf5a6c84SAndroid Build Coastguard Worker config NETSTAT
10*cf5a6c84SAndroid Build Coastguard Worker bool "netstat"
11*cf5a6c84SAndroid Build Coastguard Worker default y
12*cf5a6c84SAndroid Build Coastguard Worker help
13*cf5a6c84SAndroid Build Coastguard Worker usage: netstat [-pWrxwutneal]
14*cf5a6c84SAndroid Build Coastguard Worker
15*cf5a6c84SAndroid Build Coastguard Worker Display networking information. Default is netstat -tuwx
16*cf5a6c84SAndroid Build Coastguard Worker
17*cf5a6c84SAndroid Build Coastguard Worker -r Routing table
18*cf5a6c84SAndroid Build Coastguard Worker -a All sockets (not just connected)
19*cf5a6c84SAndroid Build Coastguard Worker -l Listening server sockets
20*cf5a6c84SAndroid Build Coastguard Worker -t TCP sockets
21*cf5a6c84SAndroid Build Coastguard Worker -u UDP sockets
22*cf5a6c84SAndroid Build Coastguard Worker -w Raw sockets
23*cf5a6c84SAndroid Build Coastguard Worker -x Unix sockets
24*cf5a6c84SAndroid Build Coastguard Worker -e Extended info
25*cf5a6c84SAndroid Build Coastguard Worker -n Don't resolve names
26*cf5a6c84SAndroid Build Coastguard Worker -W Wide display
27*cf5a6c84SAndroid Build Coastguard Worker -p Show PID/program name of sockets
28*cf5a6c84SAndroid Build Coastguard Worker */
29*cf5a6c84SAndroid Build Coastguard Worker
30*cf5a6c84SAndroid Build Coastguard Worker #define FOR_netstat
31*cf5a6c84SAndroid Build Coastguard Worker #include "toys.h"
32*cf5a6c84SAndroid Build Coastguard Worker #include <net/route.h>
33*cf5a6c84SAndroid Build Coastguard Worker
34*cf5a6c84SAndroid Build Coastguard Worker GLOBALS(
35*cf5a6c84SAndroid Build Coastguard Worker struct num_cache *inodes;
36*cf5a6c84SAndroid Build Coastguard Worker int wpad;
37*cf5a6c84SAndroid Build Coastguard Worker )
38*cf5a6c84SAndroid Build Coastguard Worker
39*cf5a6c84SAndroid Build Coastguard Worker struct num_cache {
40*cf5a6c84SAndroid Build Coastguard Worker struct num_cache *next;
41*cf5a6c84SAndroid Build Coastguard Worker long long num;
42*cf5a6c84SAndroid Build Coastguard Worker char data[];
43*cf5a6c84SAndroid Build Coastguard Worker };
44*cf5a6c84SAndroid Build Coastguard Worker
45*cf5a6c84SAndroid Build Coastguard Worker // Find num in cache
get_num_cache(struct num_cache * cache,long long num)46*cf5a6c84SAndroid Build Coastguard Worker static struct num_cache *get_num_cache(struct num_cache *cache, long long num)
47*cf5a6c84SAndroid Build Coastguard Worker {
48*cf5a6c84SAndroid Build Coastguard Worker while (cache) {
49*cf5a6c84SAndroid Build Coastguard Worker if (num==cache->num) return cache;
50*cf5a6c84SAndroid Build Coastguard Worker cache = cache->next;
51*cf5a6c84SAndroid Build Coastguard Worker }
52*cf5a6c84SAndroid Build Coastguard Worker
53*cf5a6c84SAndroid Build Coastguard Worker return 0;
54*cf5a6c84SAndroid Build Coastguard Worker }
55*cf5a6c84SAndroid Build Coastguard Worker
56*cf5a6c84SAndroid Build Coastguard Worker // Uniquely add num+data to cache. Updates *cache, returns pointer to existing
57*cf5a6c84SAndroid Build Coastguard Worker // entry if it was already there.
add_num_cache(struct num_cache ** cache,long long num,void * data,int len)58*cf5a6c84SAndroid Build Coastguard Worker static struct num_cache *add_num_cache(struct num_cache **cache, long long num,
59*cf5a6c84SAndroid Build Coastguard Worker void *data, int len)
60*cf5a6c84SAndroid Build Coastguard Worker {
61*cf5a6c84SAndroid Build Coastguard Worker struct num_cache *old = get_num_cache(*cache, num);
62*cf5a6c84SAndroid Build Coastguard Worker
63*cf5a6c84SAndroid Build Coastguard Worker if (old) return old;
64*cf5a6c84SAndroid Build Coastguard Worker
65*cf5a6c84SAndroid Build Coastguard Worker old = xzalloc(sizeof(struct num_cache)+len);
66*cf5a6c84SAndroid Build Coastguard Worker old->next = *cache;
67*cf5a6c84SAndroid Build Coastguard Worker old->num = num;
68*cf5a6c84SAndroid Build Coastguard Worker memcpy(old->data, data, len);
69*cf5a6c84SAndroid Build Coastguard Worker
70*cf5a6c84SAndroid Build Coastguard Worker *cache = old;
71*cf5a6c84SAndroid Build Coastguard Worker
72*cf5a6c84SAndroid Build Coastguard Worker return 0;
73*cf5a6c84SAndroid Build Coastguard Worker }
74*cf5a6c84SAndroid Build Coastguard Worker
addr2str(int af,void * addr,unsigned port,char * buf,int len,char * proto)75*cf5a6c84SAndroid Build Coastguard Worker static void addr2str(int af, void *addr, unsigned port, char *buf, int len,
76*cf5a6c84SAndroid Build Coastguard Worker char *proto)
77*cf5a6c84SAndroid Build Coastguard Worker {
78*cf5a6c84SAndroid Build Coastguard Worker char pres[INET6_ADDRSTRLEN];
79*cf5a6c84SAndroid Build Coastguard Worker struct servent *se = 0;
80*cf5a6c84SAndroid Build Coastguard Worker int pos, count;
81*cf5a6c84SAndroid Build Coastguard Worker
82*cf5a6c84SAndroid Build Coastguard Worker if (!inet_ntop(af, addr, pres, sizeof(pres))) perror_exit("inet_ntop");
83*cf5a6c84SAndroid Build Coastguard Worker
84*cf5a6c84SAndroid Build Coastguard Worker if (FLAG(n) || !port) {
85*cf5a6c84SAndroid Build Coastguard Worker strcpy(buf, pres);
86*cf5a6c84SAndroid Build Coastguard Worker } else {
87*cf5a6c84SAndroid Build Coastguard Worker struct addrinfo hints, *result, *rp;
88*cf5a6c84SAndroid Build Coastguard Worker char cut[4];
89*cf5a6c84SAndroid Build Coastguard Worker
90*cf5a6c84SAndroid Build Coastguard Worker memset(&hints, 0, sizeof(struct addrinfo));
91*cf5a6c84SAndroid Build Coastguard Worker hints.ai_family = af;
92*cf5a6c84SAndroid Build Coastguard Worker
93*cf5a6c84SAndroid Build Coastguard Worker if (!getaddrinfo(pres, NULL, &hints, &result)) {
94*cf5a6c84SAndroid Build Coastguard Worker socklen_t sock_len = (af == AF_INET) ? sizeof(struct sockaddr_in)
95*cf5a6c84SAndroid Build Coastguard Worker : sizeof(struct sockaddr_in6);
96*cf5a6c84SAndroid Build Coastguard Worker
97*cf5a6c84SAndroid Build Coastguard Worker // We assume that a failing getnameinfo dosn't stomp "buf" here.
98*cf5a6c84SAndroid Build Coastguard Worker for (rp = result; rp; rp = rp->ai_next)
99*cf5a6c84SAndroid Build Coastguard Worker if (!getnameinfo(rp->ai_addr, sock_len, buf, 256, 0, 0, 0)) break;
100*cf5a6c84SAndroid Build Coastguard Worker freeaddrinfo(result);
101*cf5a6c84SAndroid Build Coastguard Worker buf[len] = 0;
102*cf5a6c84SAndroid Build Coastguard Worker }
103*cf5a6c84SAndroid Build Coastguard Worker
104*cf5a6c84SAndroid Build Coastguard Worker // getservbyport() doesn't understand proto "tcp6", so truncate
105*cf5a6c84SAndroid Build Coastguard Worker memcpy(cut, proto, 3);
106*cf5a6c84SAndroid Build Coastguard Worker cut[3] = 0;
107*cf5a6c84SAndroid Build Coastguard Worker se = getservbyport(htons(port), cut);
108*cf5a6c84SAndroid Build Coastguard Worker }
109*cf5a6c84SAndroid Build Coastguard Worker
110*cf5a6c84SAndroid Build Coastguard Worker if (!strcmp(buf, "::")) strcpy(buf, "[::]");
111*cf5a6c84SAndroid Build Coastguard Worker
112*cf5a6c84SAndroid Build Coastguard Worker // Append :service or :* if port == 0.
113*cf5a6c84SAndroid Build Coastguard Worker if (se) {
114*cf5a6c84SAndroid Build Coastguard Worker count = snprintf(0, 0, ":%s", se->s_name);
115*cf5a6c84SAndroid Build Coastguard Worker // NI_MAXSERV == 32, which is greater than our minimum field width.
116*cf5a6c84SAndroid Build Coastguard Worker // (Although the longest service name on Debian in 2021 is only 16 bytes.)
117*cf5a6c84SAndroid Build Coastguard Worker if (count>=len) {
118*cf5a6c84SAndroid Build Coastguard Worker count = len-1;
119*cf5a6c84SAndroid Build Coastguard Worker se->s_name[count] = 0;
120*cf5a6c84SAndroid Build Coastguard Worker }
121*cf5a6c84SAndroid Build Coastguard Worker } else count = port ? snprintf(0, 0, ":%u", port) : 2;
122*cf5a6c84SAndroid Build Coastguard Worker // We always show the port, even if that means clobbering the end of the host.
123*cf5a6c84SAndroid Build Coastguard Worker pos = strlen(buf);
124*cf5a6c84SAndroid Build Coastguard Worker if (len-pos<count) pos = len-count;
125*cf5a6c84SAndroid Build Coastguard Worker if (se) sprintf(buf+pos, ":%s", se->s_name);
126*cf5a6c84SAndroid Build Coastguard Worker else sprintf(buf+pos, port ? ":%u" : ":*", port);
127*cf5a6c84SAndroid Build Coastguard Worker }
128*cf5a6c84SAndroid Build Coastguard Worker
129*cf5a6c84SAndroid Build Coastguard Worker // Display info for tcp/udp/raw
show_ip(char * fname)130*cf5a6c84SAndroid Build Coastguard Worker static void show_ip(char *fname)
131*cf5a6c84SAndroid Build Coastguard Worker {
132*cf5a6c84SAndroid Build Coastguard Worker char *ss_state = "UNKNOWN", buf[12], *s, *label = strrchr(fname, '/')+1;
133*cf5a6c84SAndroid Build Coastguard Worker char *state_label[] = {"", "ESTABLISHED", "SYN_SENT", "SYN_RECV", "FIN_WAIT1",
134*cf5a6c84SAndroid Build Coastguard Worker "FIN_WAIT2", "TIME_WAIT", "CLOSE", "CLOSE_WAIT",
135*cf5a6c84SAndroid Build Coastguard Worker "LAST_ACK", "LISTEN", "CLOSING", "UNKNOWN"};
136*cf5a6c84SAndroid Build Coastguard Worker FILE *fp = xfopen(fname, "r");
137*cf5a6c84SAndroid Build Coastguard Worker
138*cf5a6c84SAndroid Build Coastguard Worker // Skip header.
139*cf5a6c84SAndroid Build Coastguard Worker (void)fgets(toybuf, sizeof(toybuf), fp);
140*cf5a6c84SAndroid Build Coastguard Worker
141*cf5a6c84SAndroid Build Coastguard Worker while (fgets(toybuf, sizeof(toybuf), fp)) {
142*cf5a6c84SAndroid Build Coastguard Worker char lip[256], rip[256];
143*cf5a6c84SAndroid Build Coastguard Worker union {
144*cf5a6c84SAndroid Build Coastguard Worker struct {unsigned u; unsigned char b[4];} i4;
145*cf5a6c84SAndroid Build Coastguard Worker struct {struct {unsigned a, b, c, d;} u; unsigned char b[16];} i6;
146*cf5a6c84SAndroid Build Coastguard Worker } laddr, raddr;
147*cf5a6c84SAndroid Build Coastguard Worker unsigned lport, rport, state, txq, rxq, num, uid, af = AF_INET6;
148*cf5a6c84SAndroid Build Coastguard Worker unsigned long inode;
149*cf5a6c84SAndroid Build Coastguard Worker
150*cf5a6c84SAndroid Build Coastguard Worker // Try ipv6, then try ipv4
151*cf5a6c84SAndroid Build Coastguard Worker if (16 != sscanf(toybuf,
152*cf5a6c84SAndroid Build Coastguard Worker " %d: %8x%8x%8x%8x:%x %8x%8x%8x%8x:%x %x %x:%x %*X:%*X %*X %d %*d %ld",
153*cf5a6c84SAndroid Build Coastguard Worker &num, &laddr.i6.u.a, &laddr.i6.u.b, &laddr.i6.u.c,
154*cf5a6c84SAndroid Build Coastguard Worker &laddr.i6.u.d, &lport, &raddr.i6.u.a, &raddr.i6.u.b,
155*cf5a6c84SAndroid Build Coastguard Worker &raddr.i6.u.c, &raddr.i6.u.d, &rport, &state, &txq, &rxq,
156*cf5a6c84SAndroid Build Coastguard Worker &uid, &inode))
157*cf5a6c84SAndroid Build Coastguard Worker {
158*cf5a6c84SAndroid Build Coastguard Worker af = AF_INET;
159*cf5a6c84SAndroid Build Coastguard Worker if (10 != sscanf(toybuf,
160*cf5a6c84SAndroid Build Coastguard Worker " %d: %x:%x %x:%x %x %x:%x %*X:%*X %*X %d %*d %ld",
161*cf5a6c84SAndroid Build Coastguard Worker &num, &laddr.i4.u, &lport, &raddr.i4.u, &rport, &state, &txq,
162*cf5a6c84SAndroid Build Coastguard Worker &rxq, &uid, &inode)) continue;
163*cf5a6c84SAndroid Build Coastguard Worker }
164*cf5a6c84SAndroid Build Coastguard Worker
165*cf5a6c84SAndroid Build Coastguard Worker // Should we display this? (listening or all or TCP/UDP/RAW)
166*cf5a6c84SAndroid Build Coastguard Worker if (!(FLAG(l) && (!rport && (state&0xA))) && !FLAG(a) && !(rport&0x70))
167*cf5a6c84SAndroid Build Coastguard Worker continue;
168*cf5a6c84SAndroid Build Coastguard Worker
169*cf5a6c84SAndroid Build Coastguard Worker addr2str(af, &laddr, lport, lip, TT.wpad, label);
170*cf5a6c84SAndroid Build Coastguard Worker addr2str(af, &raddr, rport, rip, TT.wpad, label);
171*cf5a6c84SAndroid Build Coastguard Worker
172*cf5a6c84SAndroid Build Coastguard Worker // Display data
173*cf5a6c84SAndroid Build Coastguard Worker s = label;
174*cf5a6c84SAndroid Build Coastguard Worker if (strstart(&s, "tcp")) {
175*cf5a6c84SAndroid Build Coastguard Worker int sz = ARRAY_LEN(state_label);
176*cf5a6c84SAndroid Build Coastguard Worker if (!state || state >= sz) state = sz-1;
177*cf5a6c84SAndroid Build Coastguard Worker ss_state = state_label[state];
178*cf5a6c84SAndroid Build Coastguard Worker } else if (strstart(&s, "udp")) {
179*cf5a6c84SAndroid Build Coastguard Worker if (state == 1) ss_state = state_label[state];
180*cf5a6c84SAndroid Build Coastguard Worker else if (state == 7) ss_state = "";
181*cf5a6c84SAndroid Build Coastguard Worker } else if (strstart(&s, "raw")) sprintf(ss_state = buf, "%u", state);
182*cf5a6c84SAndroid Build Coastguard Worker
183*cf5a6c84SAndroid Build Coastguard Worker printf("%-6s%6d%7d %*.*s %*.*s %-11s", label, rxq, txq, -TT.wpad, TT.wpad,
184*cf5a6c84SAndroid Build Coastguard Worker lip, -TT.wpad, TT.wpad, rip, ss_state);
185*cf5a6c84SAndroid Build Coastguard Worker if (FLAG(e)) {
186*cf5a6c84SAndroid Build Coastguard Worker if (FLAG(n)) sprintf(s = toybuf, "%d", uid);
187*cf5a6c84SAndroid Build Coastguard Worker else s = getusername(uid);
188*cf5a6c84SAndroid Build Coastguard Worker printf(" %-10s %-11ld", s, inode);
189*cf5a6c84SAndroid Build Coastguard Worker }
190*cf5a6c84SAndroid Build Coastguard Worker if (FLAG(p)) {
191*cf5a6c84SAndroid Build Coastguard Worker struct num_cache *nc = get_num_cache(TT.inodes, inode);
192*cf5a6c84SAndroid Build Coastguard Worker
193*cf5a6c84SAndroid Build Coastguard Worker printf(" %s", nc ? nc->data : "-");
194*cf5a6c84SAndroid Build Coastguard Worker }
195*cf5a6c84SAndroid Build Coastguard Worker xputc('\n');
196*cf5a6c84SAndroid Build Coastguard Worker }
197*cf5a6c84SAndroid Build Coastguard Worker fclose(fp);
198*cf5a6c84SAndroid Build Coastguard Worker }
199*cf5a6c84SAndroid Build Coastguard Worker
show_unix_sockets(void)200*cf5a6c84SAndroid Build Coastguard Worker static void show_unix_sockets(void)
201*cf5a6c84SAndroid Build Coastguard Worker {
202*cf5a6c84SAndroid Build Coastguard Worker char *types[] = {"","STREAM","DGRAM","RAW","RDM","SEQPACKET","DCCP","PACKET"},
203*cf5a6c84SAndroid Build Coastguard Worker *states[] = {"","LISTENING","CONNECTING","CONNECTED","DISCONNECTING"},
204*cf5a6c84SAndroid Build Coastguard Worker *filename = 0;
205*cf5a6c84SAndroid Build Coastguard Worker unsigned long refcount, flags, type, state, inode;
206*cf5a6c84SAndroid Build Coastguard Worker FILE *fp = xfopen("/proc/net/unix", "r");
207*cf5a6c84SAndroid Build Coastguard Worker
208*cf5a6c84SAndroid Build Coastguard Worker // Skip header.
209*cf5a6c84SAndroid Build Coastguard Worker (void)fgets(toybuf, sizeof(toybuf), fp);
210*cf5a6c84SAndroid Build Coastguard Worker
211*cf5a6c84SAndroid Build Coastguard Worker while (fscanf(fp, "%*p: %lX %*X %lX %lX %lX %lu%m[^\n]", &refcount, &flags,
212*cf5a6c84SAndroid Build Coastguard Worker &type, &state, &inode, &filename) >= 5) {
213*cf5a6c84SAndroid Build Coastguard Worker // Linux exports only SO_ACCEPTCON since 2.3.15pre3 in 1999, but let's
214*cf5a6c84SAndroid Build Coastguard Worker // filter in case they add more someday.
215*cf5a6c84SAndroid Build Coastguard Worker flags &= 1<<16;
216*cf5a6c84SAndroid Build Coastguard Worker
217*cf5a6c84SAndroid Build Coastguard Worker // Only show unconnected listening sockets with -a or -l.
218*cf5a6c84SAndroid Build Coastguard Worker if (state==1 && flags && !(FLAG(a) || FLAG(l))) continue;
219*cf5a6c84SAndroid Build Coastguard Worker
220*cf5a6c84SAndroid Build Coastguard Worker if (type==10) type = 7; // move SOCK_PACKET into line
221*cf5a6c84SAndroid Build Coastguard Worker if (type>=ARRAY_LEN(types)) type = 0;
222*cf5a6c84SAndroid Build Coastguard Worker if (state>=ARRAY_LEN(states) || (state==1 && !flags)) state = 0;
223*cf5a6c84SAndroid Build Coastguard Worker
224*cf5a6c84SAndroid Build Coastguard Worker if (state!=1 && FLAG(l)) continue;
225*cf5a6c84SAndroid Build Coastguard Worker
226*cf5a6c84SAndroid Build Coastguard Worker sprintf(toybuf, "[ %s]", flags ? "ACC " : "");
227*cf5a6c84SAndroid Build Coastguard Worker printf("unix %-6ld %-11s %-10s %-13s %-8lu ",
228*cf5a6c84SAndroid Build Coastguard Worker refcount, toybuf, types[type], states[state], inode);
229*cf5a6c84SAndroid Build Coastguard Worker if (FLAG(p)) {
230*cf5a6c84SAndroid Build Coastguard Worker struct num_cache *nc = get_num_cache(TT.inodes, inode);
231*cf5a6c84SAndroid Build Coastguard Worker
232*cf5a6c84SAndroid Build Coastguard Worker printf("%-19.19s ", nc ? nc->data : "-");
233*cf5a6c84SAndroid Build Coastguard Worker }
234*cf5a6c84SAndroid Build Coastguard Worker
235*cf5a6c84SAndroid Build Coastguard Worker if (filename) {
236*cf5a6c84SAndroid Build Coastguard Worker printf("%s\n", filename+!FLAG(p));
237*cf5a6c84SAndroid Build Coastguard Worker free(filename);
238*cf5a6c84SAndroid Build Coastguard Worker filename = 0;
239*cf5a6c84SAndroid Build Coastguard Worker } else xputc('\n');
240*cf5a6c84SAndroid Build Coastguard Worker }
241*cf5a6c84SAndroid Build Coastguard Worker fclose(fp);
242*cf5a6c84SAndroid Build Coastguard Worker }
243*cf5a6c84SAndroid Build Coastguard Worker
scan_pids(struct dirtree * node)244*cf5a6c84SAndroid Build Coastguard Worker static int scan_pids(struct dirtree *node)
245*cf5a6c84SAndroid Build Coastguard Worker {
246*cf5a6c84SAndroid Build Coastguard Worker char *s = toybuf+256;
247*cf5a6c84SAndroid Build Coastguard Worker struct dirent *entry;
248*cf5a6c84SAndroid Build Coastguard Worker DIR *dp;
249*cf5a6c84SAndroid Build Coastguard Worker int pid, dirfd;
250*cf5a6c84SAndroid Build Coastguard Worker
251*cf5a6c84SAndroid Build Coastguard Worker if (!node->parent) return DIRTREE_RECURSE;
252*cf5a6c84SAndroid Build Coastguard Worker if (!(pid = atol(node->name))) return 0;
253*cf5a6c84SAndroid Build Coastguard Worker
254*cf5a6c84SAndroid Build Coastguard Worker sprintf(toybuf, "/proc/%d/cmdline", pid);
255*cf5a6c84SAndroid Build Coastguard Worker if (!(readfile(toybuf, toybuf, 256))) return 0;
256*cf5a6c84SAndroid Build Coastguard Worker
257*cf5a6c84SAndroid Build Coastguard Worker sprintf(s, "%d/fd", pid);
258*cf5a6c84SAndroid Build Coastguard Worker if (-1==(dirfd = openat(dirtree_parentfd(node), s, O_RDONLY))) return 0;
259*cf5a6c84SAndroid Build Coastguard Worker if (!(dp = fdopendir(dirfd))) close(dirfd);
260*cf5a6c84SAndroid Build Coastguard Worker else while ((entry = readdir(dp))) {
261*cf5a6c84SAndroid Build Coastguard Worker s = toybuf+256;
262*cf5a6c84SAndroid Build Coastguard Worker if (!readlinkat0(dirfd, entry->d_name, s, sizeof(toybuf)-256)) continue;
263*cf5a6c84SAndroid Build Coastguard Worker // Can the "[0000]:" happen in a modern kernel?
264*cf5a6c84SAndroid Build Coastguard Worker if (strstart(&s, "socket:[") || strstart(&s, "[0000]:")) {
265*cf5a6c84SAndroid Build Coastguard Worker long long ll = atoll(s);
266*cf5a6c84SAndroid Build Coastguard Worker
267*cf5a6c84SAndroid Build Coastguard Worker sprintf(s, "%d/%s", pid, getbasename(toybuf));
268*cf5a6c84SAndroid Build Coastguard Worker add_num_cache(&TT.inodes, ll, s, strlen(s)+1);
269*cf5a6c84SAndroid Build Coastguard Worker }
270*cf5a6c84SAndroid Build Coastguard Worker }
271*cf5a6c84SAndroid Build Coastguard Worker closedir(dp);
272*cf5a6c84SAndroid Build Coastguard Worker
273*cf5a6c84SAndroid Build Coastguard Worker return 0;
274*cf5a6c84SAndroid Build Coastguard Worker }
275*cf5a6c84SAndroid Build Coastguard Worker
276*cf5a6c84SAndroid Build Coastguard Worker // extract inet4 route info from /proc/net/route file and display it.
display_routes(void)277*cf5a6c84SAndroid Build Coastguard Worker static void display_routes(void)
278*cf5a6c84SAndroid Build Coastguard Worker {
279*cf5a6c84SAndroid Build Coastguard Worker static const char flagchars[] = "GHRDMDAC";
280*cf5a6c84SAndroid Build Coastguard Worker static const unsigned flagarray[] = {
281*cf5a6c84SAndroid Build Coastguard Worker RTF_GATEWAY, RTF_HOST, RTF_REINSTATE, RTF_DYNAMIC, RTF_MODIFIED
282*cf5a6c84SAndroid Build Coastguard Worker };
283*cf5a6c84SAndroid Build Coastguard Worker unsigned dest, gate, mask;
284*cf5a6c84SAndroid Build Coastguard Worker int flags, ref, use, metric, mss, win, irtt;
285*cf5a6c84SAndroid Build Coastguard Worker char *out = toybuf, *flag_val;
286*cf5a6c84SAndroid Build Coastguard Worker char iface[64]={0};
287*cf5a6c84SAndroid Build Coastguard Worker FILE *fp = xfopen("/proc/net/route", "r");
288*cf5a6c84SAndroid Build Coastguard Worker
289*cf5a6c84SAndroid Build Coastguard Worker // Skip header.
290*cf5a6c84SAndroid Build Coastguard Worker (void)fgets(toybuf, sizeof(toybuf), fp);
291*cf5a6c84SAndroid Build Coastguard Worker
292*cf5a6c84SAndroid Build Coastguard Worker printf("Kernel IP routing table\n"
293*cf5a6c84SAndroid Build Coastguard Worker "Destination\tGateway \tGenmask \tFlags %s Iface\n",
294*cf5a6c84SAndroid Build Coastguard Worker !FLAG(e) ? " MSS Window irtt" : "Metric Ref Use");
295*cf5a6c84SAndroid Build Coastguard Worker
296*cf5a6c84SAndroid Build Coastguard Worker while (fscanf(fp, "%63s%x%x%X%d%d%d%x%d%d%d", iface, &dest, &gate, &flags,
297*cf5a6c84SAndroid Build Coastguard Worker &ref, &use, &metric, &mask, &mss, &win, &irtt) == 11) {
298*cf5a6c84SAndroid Build Coastguard Worker char *destip = 0, *gateip = 0, *maskip = 0;
299*cf5a6c84SAndroid Build Coastguard Worker
300*cf5a6c84SAndroid Build Coastguard Worker // skip down interfaces.
301*cf5a6c84SAndroid Build Coastguard Worker if (!(flags & RTF_UP)) continue;
302*cf5a6c84SAndroid Build Coastguard Worker
303*cf5a6c84SAndroid Build Coastguard Worker // TODO /proc/net/ipv6_route
304*cf5a6c84SAndroid Build Coastguard Worker
305*cf5a6c84SAndroid Build Coastguard Worker if (dest) {
306*cf5a6c84SAndroid Build Coastguard Worker if (inet_ntop(AF_INET, &dest, out, 16)) destip = out;
307*cf5a6c84SAndroid Build Coastguard Worker } else destip = FLAG(n) ? "0.0.0.0" : "default";
308*cf5a6c84SAndroid Build Coastguard Worker out += 16;
309*cf5a6c84SAndroid Build Coastguard Worker
310*cf5a6c84SAndroid Build Coastguard Worker if (gate) {
311*cf5a6c84SAndroid Build Coastguard Worker if (inet_ntop(AF_INET, &gate, out, 16)) gateip = out;
312*cf5a6c84SAndroid Build Coastguard Worker } else gateip = FLAG(n) ? "0.0.0.0" : "*";
313*cf5a6c84SAndroid Build Coastguard Worker out += 16;
314*cf5a6c84SAndroid Build Coastguard Worker
315*cf5a6c84SAndroid Build Coastguard Worker // TODO /24
316*cf5a6c84SAndroid Build Coastguard Worker //For Mask
317*cf5a6c84SAndroid Build Coastguard Worker if (inet_ntop(AF_INET, &mask, out, 16)) maskip = out;
318*cf5a6c84SAndroid Build Coastguard Worker else maskip = "?";
319*cf5a6c84SAndroid Build Coastguard Worker out += 16;
320*cf5a6c84SAndroid Build Coastguard Worker
321*cf5a6c84SAndroid Build Coastguard Worker //Get flag Values
322*cf5a6c84SAndroid Build Coastguard Worker flag_val = out;
323*cf5a6c84SAndroid Build Coastguard Worker *out++ = 'U';
324*cf5a6c84SAndroid Build Coastguard Worker for (dest = 0; dest < ARRAY_LEN(flagarray); dest++)
325*cf5a6c84SAndroid Build Coastguard Worker if (flags&flagarray[dest]) *out++ = flagchars[dest];
326*cf5a6c84SAndroid Build Coastguard Worker *out = 0;
327*cf5a6c84SAndroid Build Coastguard Worker if (flags & RTF_REJECT) *flag_val = '!';
328*cf5a6c84SAndroid Build Coastguard Worker
329*cf5a6c84SAndroid Build Coastguard Worker printf("%-15.15s %-15.15s %-16s%-6s", destip, gateip, maskip, flag_val);
330*cf5a6c84SAndroid Build Coastguard Worker if (!FLAG(e)) printf("%5d %-5d %6d %s\n", mss, win, irtt, iface);
331*cf5a6c84SAndroid Build Coastguard Worker else printf("%-6d %-2d %7d %s\n", metric, ref, use, iface);
332*cf5a6c84SAndroid Build Coastguard Worker }
333*cf5a6c84SAndroid Build Coastguard Worker fclose(fp);
334*cf5a6c84SAndroid Build Coastguard Worker }
335*cf5a6c84SAndroid Build Coastguard Worker
netstat_main(void)336*cf5a6c84SAndroid Build Coastguard Worker void netstat_main(void)
337*cf5a6c84SAndroid Build Coastguard Worker {
338*cf5a6c84SAndroid Build Coastguard Worker int tuwx = FLAG_t|FLAG_u|FLAG_w|FLAG_x;
339*cf5a6c84SAndroid Build Coastguard Worker char *type = "w/o servers";
340*cf5a6c84SAndroid Build Coastguard Worker
341*cf5a6c84SAndroid Build Coastguard Worker TT.wpad = FLAG(W) ? 51 : 23;
342*cf5a6c84SAndroid Build Coastguard Worker if (!(toys.optflags&(FLAG_r|tuwx))) toys.optflags |= tuwx;
343*cf5a6c84SAndroid Build Coastguard Worker if (FLAG(r)) display_routes();
344*cf5a6c84SAndroid Build Coastguard Worker if (!(toys.optflags&tuwx)) return;
345*cf5a6c84SAndroid Build Coastguard Worker
346*cf5a6c84SAndroid Build Coastguard Worker if (FLAG(a)) type = "servers and established";
347*cf5a6c84SAndroid Build Coastguard Worker else if (FLAG(l)) type = "only servers";
348*cf5a6c84SAndroid Build Coastguard Worker
349*cf5a6c84SAndroid Build Coastguard Worker if (FLAG(p)) dirtree_read("/proc", scan_pids);
350*cf5a6c84SAndroid Build Coastguard Worker
351*cf5a6c84SAndroid Build Coastguard Worker if (toys.optflags&(FLAG_t|FLAG_u|FLAG_w)) {
352*cf5a6c84SAndroid Build Coastguard Worker printf("Active Internet connections (%s)\n", type);
353*cf5a6c84SAndroid Build Coastguard Worker printf("Proto Recv-Q Send-Q %*s %*s State ", -TT.wpad, "Local Address",
354*cf5a6c84SAndroid Build Coastguard Worker -TT.wpad, "Foreign Address");
355*cf5a6c84SAndroid Build Coastguard Worker if (FLAG(e)) printf(" User Inode ");
356*cf5a6c84SAndroid Build Coastguard Worker if (FLAG(p)) printf(" PID/Program Name");
357*cf5a6c84SAndroid Build Coastguard Worker xputc('\n');
358*cf5a6c84SAndroid Build Coastguard Worker
359*cf5a6c84SAndroid Build Coastguard Worker if (FLAG(t)) {
360*cf5a6c84SAndroid Build Coastguard Worker show_ip("/proc/net/tcp");
361*cf5a6c84SAndroid Build Coastguard Worker show_ip("/proc/net/tcp6");
362*cf5a6c84SAndroid Build Coastguard Worker }
363*cf5a6c84SAndroid Build Coastguard Worker if (FLAG(u)) {
364*cf5a6c84SAndroid Build Coastguard Worker show_ip("/proc/net/udp");
365*cf5a6c84SAndroid Build Coastguard Worker show_ip("/proc/net/udp6");
366*cf5a6c84SAndroid Build Coastguard Worker }
367*cf5a6c84SAndroid Build Coastguard Worker if (FLAG(w)) {
368*cf5a6c84SAndroid Build Coastguard Worker show_ip("/proc/net/raw");
369*cf5a6c84SAndroid Build Coastguard Worker show_ip("/proc/net/raw6");
370*cf5a6c84SAndroid Build Coastguard Worker }
371*cf5a6c84SAndroid Build Coastguard Worker }
372*cf5a6c84SAndroid Build Coastguard Worker
373*cf5a6c84SAndroid Build Coastguard Worker if (FLAG(x)) {
374*cf5a6c84SAndroid Build Coastguard Worker printf("Active UNIX domain sockets (%s)\n", type);
375*cf5a6c84SAndroid Build Coastguard Worker printf("Proto RefCnt Flags Type State I-Node%sPath\n",
376*cf5a6c84SAndroid Build Coastguard Worker FLAG(p) ? " PID/Program Name " : " ");
377*cf5a6c84SAndroid Build Coastguard Worker show_unix_sockets();
378*cf5a6c84SAndroid Build Coastguard Worker }
379*cf5a6c84SAndroid Build Coastguard Worker
380*cf5a6c84SAndroid Build Coastguard Worker if (FLAG(p) && CFG_TOYBOX_FREE) llist_traverse(TT.inodes, free);
381*cf5a6c84SAndroid Build Coastguard Worker toys.exitval = 0;
382*cf5a6c84SAndroid Build Coastguard Worker }
383