1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30#define FOR_netstat
31#include "toys.h"
32#include <net/route.h>
33
34GLOBALS(
35 struct num_cache *inodes;
36 int wpad;
37);
38
39
40static void addr2str(int af, void *addr, unsigned port, char *buf, int len,
41 char *proto)
42{
43 int pos, count;
44 struct servent *ser = 0;
45
46
47 if (!inet_ntop(af, addr, buf, 256)) {
48 *buf = 0;
49
50 return;
51 }
52 buf[len] = 0;
53 pos = strlen(buf);
54
55
56 if (!port) {
57 if (len-pos<2) pos = len-2;
58 strcpy(buf+pos, ":*");
59
60 return;
61 }
62
63 if (!(toys.optflags & FLAG_n)) {
64 struct addrinfo hints, *result, *rp;
65 char cut[4];
66
67 memset(&hints, 0, sizeof(struct addrinfo));
68 hints.ai_family = af;
69
70 if (!getaddrinfo(buf, NULL, &hints, &result)) {
71 socklen_t sock_len = (af == AF_INET) ? sizeof(struct sockaddr_in)
72 : sizeof(struct sockaddr_in6);
73
74
75 for (rp = result; rp; rp = rp->ai_next)
76 if (!getnameinfo(rp->ai_addr, sock_len, buf, 256, 0, 0, 0)) break;
77 freeaddrinfo(result);
78 buf[len] = 0;
79 pos = strlen(buf);
80 }
81
82
83 memcpy(cut, proto, 3);
84 cut[3] = 0;
85 ser = getservbyport(htons(port), cut);
86 }
87
88
89 count = snprintf(0, 0, ":%u", port);
90 if (ser) {
91 count = snprintf(0, 0, ":%s", ser->s_name);
92
93 if (count>=len) {
94 count = len-1;
95 ser->s_name[count] = 0;
96 }
97 }
98 if (len-pos<count) pos = len-count;
99 if (ser) sprintf(buf+pos, ":%s", ser->s_name);
100 else sprintf(buf+pos, ":%u", port);
101}
102
103
104static void show_ip(char *fname)
105{
106 char *ss_state = "UNKNOWN", buf[12], *s, *label = strrchr(fname, '/')+1;
107 char *state_label[] = {"", "ESTABLISHED", "SYN_SENT", "SYN_RECV", "FIN_WAIT1",
108 "FIN_WAIT2", "TIME_WAIT", "CLOSE", "CLOSE_WAIT",
109 "LAST_ACK", "LISTEN", "CLOSING", "UNKNOWN"};
110 struct passwd *pw;
111 FILE *fp = fopen(fname, "r");
112
113 if (!fp) {
114 perror_msg("'%s'", fname);
115 return;
116 }
117
118 if(!fgets(toybuf, sizeof(toybuf), fp)) return;
119
120 while (fgets(toybuf, sizeof(toybuf), fp)) {
121 char lip[256], rip[256];
122 union {
123 struct {unsigned u; unsigned char b[4];} i4;
124 struct {struct {unsigned a, b, c, d;} u; unsigned char b[16];} i6;
125 } laddr, raddr;
126 unsigned lport, rport, state, txq, rxq, num, uid, nitems;
127 unsigned long inode;
128
129
130 nitems = sscanf(toybuf,
131 " %d: %8x%8x%8x%8x:%x %8x%8x%8x%8x:%x %x %x:%x %*X:%*X %*X %d %*d %ld",
132 &num, &laddr.i6.u.a, &laddr.i6.u.b, &laddr.i6.u.c,
133 &laddr.i6.u.d, &lport, &raddr.i6.u.a, &raddr.i6.u.b,
134 &raddr.i6.u.c, &raddr.i6.u.d, &rport, &state, &txq, &rxq,
135 &uid, &inode);
136
137 if (nitems!=16) {
138 nitems = sscanf(toybuf,
139 " %d: %x:%x %x:%x %x %x:%x %*X:%*X %*X %d %*d %ld",
140 &num, &laddr.i4.u, &lport, &raddr.i4.u, &rport, &state, &txq,
141 &rxq, &uid, &inode);
142
143 if (nitems!=10) continue;
144 nitems = AF_INET;
145 } else nitems = AF_INET6;
146
147
148 if (!((toys.optflags & FLAG_l) && (!rport && (state & 0xA)))
149 && !(toys.optflags & FLAG_a) && !(rport & (0x10 | 0x20 | 0x40)))
150 continue;
151
152 addr2str(nitems, &laddr, lport, lip, TT.wpad, label);
153 addr2str(nitems, &raddr, rport, rip, TT.wpad, label);
154
155
156 s = label;
157 if (strstart(&s, "tcp")) {
158 int sz = ARRAY_LEN(state_label);
159 if (!state || state >= sz) state = sz-1;
160 ss_state = state_label[state];
161 } else if (strstart(&s, "udp")) {
162 if (state == 1) ss_state = state_label[state];
163 else if (state == 7) ss_state = "";
164 } else if (strstart(&s, "raw")) sprintf(ss_state = buf, "%u", state);
165
166 if (!(toys.optflags & FLAG_n) && (pw = bufgetpwuid(uid)))
167 snprintf(toybuf, sizeof(toybuf), "%s", pw->pw_name);
168 else snprintf(toybuf, sizeof(toybuf), "%d", uid);
169
170 printf("%-6s%6d%7d ", label, rxq, txq);
171 printf("%*.*s %*.*s ", -TT.wpad, TT.wpad, lip, -TT.wpad, TT.wpad, rip);
172 printf("%-11s", ss_state);
173 if ((toys.optflags & FLAG_e)) printf(" %-10s %-11ld", toybuf, inode);
174 if ((toys.optflags & FLAG_p)) {
175 struct num_cache *nc = get_num_cache(TT.inodes, inode);
176
177 printf(" %s", nc ? nc->data : "-");
178 }
179 xputc('\n');
180 }
181 fclose(fp);
182}
183
184static void show_unix_sockets(void)
185{
186 char *types[] = {"","STREAM","DGRAM","RAW","RDM","SEQPACKET","DCCP","PACKET"},
187 *states[] = {"","LISTENING","CONNECTING","CONNECTED","DISCONNECTING"},
188 *s, *ss;
189 unsigned long refcount, flags, type, state, inode;
190 FILE *fp = xfopen("/proc/net/unix", "r");
191
192 if(!fgets(toybuf, sizeof(toybuf), fp)) return;
193
194 while (fgets(toybuf, sizeof(toybuf), fp)) {
195 unsigned offset = 0;
196
197
198 if (6<sscanf(toybuf, "%*p: %lX %*X %lX %lX %lX %lu %n",
199 &refcount, &flags, &type, &state, &inode, &offset))
200 continue;
201
202
203
204 flags &= 1<<16;
205
206
207 if (state==1 && flags && !(toys.optflags&FLAG_a)) continue;
208
209 if (type==10) type = 7;
210 if (type>ARRAY_LEN(types)) type = 0;
211 if (state>ARRAY_LEN(states) || (state==1 && !flags)) state = 0;
212 sprintf(toybuf, "[ %s]", flags ? "ACC " : "");
213
214 printf("unix %-6ld %-11s %-10s %-13s %8lu ",
215 refcount, toybuf, types[type], states[state], inode);
216 if (toys.optflags & FLAG_p) {
217 struct num_cache *nc = get_num_cache(TT.inodes, inode);
218
219 printf("%-19.19s", nc ? nc->data : "-");
220 }
221
222 if (offset) {
223 if ((ss = strrchr(s = toybuf+offset, '\n'))) *ss = 0;
224 printf("%s", s);
225 }
226 xputc('\n');
227 }
228
229 fclose(fp);
230}
231
232static int scan_pids(struct dirtree *node)
233{
234 char *s = toybuf+256;
235 struct dirent *entry;
236 DIR *dp;
237 int pid, dirfd;
238
239 if (!node->parent) return DIRTREE_RECURSE;
240 if (!(pid = atol(node->name))) return 0;
241
242 sprintf(toybuf, "/proc/%d/cmdline", pid);
243 if (!(readfile(toybuf, toybuf, 256))) return 0;
244
245 sprintf(s, "%d/fd", pid);
246 if (-1==(dirfd = openat(dirtree_parentfd(node), s, O_RDONLY))) return 0;
247 if (!(dp = fdopendir(dirfd))) {
248 close(dirfd);
249
250 return 0;
251 }
252
253 while ((entry = readdir(dp))) {
254 s = toybuf+256;
255 if (!readlinkat0(dirfd, entry->d_name, s, sizeof(toybuf)-256)) continue;
256
257 if (strstart(&s, "socket:[") || strstart(&s, "[0000]:")) {
258 long long ll = atoll(s);
259
260 sprintf(s, "%d/%s", pid, getbasename(toybuf));
261 add_num_cache(&TT.inodes, ll, s, strlen(s)+1);
262 }
263 }
264 closedir(dp);
265
266 return 0;
267}
268
269
270
271
272static void display_routes(void)
273{
274 static const char flagchars[] = "GHRDMDAC";
275 static const unsigned flagarray[] = {
276 RTF_GATEWAY, RTF_HOST, RTF_REINSTATE, RTF_DYNAMIC, RTF_MODIFIED
277 };
278 unsigned dest, gate, mask;
279 int flags, ref, use, metric, mss, win, irtt;
280 char *out = toybuf, *flag_val;
281 char iface[64]={0};
282 FILE *fp = xfopen("/proc/net/route", "r");
283
284 if(!fgets(toybuf, sizeof(toybuf), fp)) return;
285
286 printf("Kernel IP routing table\n"
287 "Destination\tGateway \tGenmask \tFlags %s Iface\n",
288 !(toys.optflags&FLAG_e) ? " MSS Window irtt" : "Metric Ref Use");
289
290 while (fgets(toybuf, sizeof(toybuf), fp)) {
291 char *destip = 0, *gateip = 0, *maskip = 0;
292
293 if (11 != sscanf(toybuf, "%63s%x%x%X%d%d%d%x%d%d%d", iface, &dest,
294 &gate, &flags, &ref, &use, &metric, &mask, &mss, &win, &irtt))
295 break;
296
297
298 if (!(flags & RTF_UP)) continue;
299
300
301
302 if (dest) {
303 if (inet_ntop(AF_INET, &dest, out, 16)) destip = out;
304 } else destip = (toys.optflags&FLAG_n) ? "0.0.0.0" : "default";
305 out += 16;
306
307 if (gate) {
308 if (inet_ntop(AF_INET, &gate, out, 16)) gateip = out;
309 } else gateip = (toys.optflags&FLAG_n) ? "0.0.0.0" : "*";
310 out += 16;
311
312
313
314 if (inet_ntop(AF_INET, &mask, out, 16)) maskip = out;
315 else maskip = "?";
316 out += 16;
317
318
319 flag_val = out;
320 *out++ = 'U';
321 for (dest = 0; dest < ARRAY_LEN(flagarray); dest++)
322 if (flags&flagarray[dest]) *out++ = flagchars[dest];
323 *out = 0;
324 if (flags & RTF_REJECT) *flag_val = '!';
325
326 printf("%-15.15s %-15.15s %-16s%-6s", destip, gateip, maskip, flag_val);
327 if (!(toys.optflags & FLAG_e))
328 printf("%5d %-5d %6d %s\n", mss, win, irtt, iface);
329 else printf("%-6d %-2d %7d %s\n", metric, ref, use, iface);
330 }
331
332 fclose(fp);
333}
334
335void netstat_main(void)
336{
337 int tuwx = FLAG_t|FLAG_u|FLAG_w|FLAG_x;
338 char *type = "w/o";
339
340 TT.wpad = (toys.optflags&FLAG_W) ? 51 : 23;
341 if (!(toys.optflags&(FLAG_r|tuwx))) toys.optflags |= tuwx;
342 if (toys.optflags & FLAG_r) display_routes();
343 if (!(toys.optflags&tuwx)) return;
344
345 if (toys.optflags & FLAG_a) type = "established and";
346 else if (toys.optflags & FLAG_l) type = "only";
347
348 if (toys.optflags & FLAG_p) dirtree_read("/proc", scan_pids);
349
350 if (toys.optflags&(FLAG_t|FLAG_u|FLAG_w)) {
351 printf("Active %s (%s servers)\n", "Internet connections", type);
352 printf("Proto Recv-Q Send-Q %*s %*s State ", -TT.wpad, "Local Address",
353 -TT.wpad, "Foreign Address");
354 if (toys.optflags & FLAG_e) printf(" User Inode ");
355 if (toys.optflags & FLAG_p) printf(" PID/Program Name");
356 xputc('\n');
357
358 if (toys.optflags & FLAG_t) {
359 show_ip("/proc/net/tcp");
360 show_ip("/proc/net/tcp6");
361 }
362 if (toys.optflags & FLAG_u) {
363 show_ip("/proc/net/udp");
364 show_ip("/proc/net/udp6");
365 }
366 if (toys.optflags & FLAG_w) {
367 show_ip("/proc/net/raw");
368 show_ip("/proc/net/raw6");
369 }
370 }
371
372 if (toys.optflags & FLAG_x) {
373 printf("Active %s (%s servers)\n", "UNIX domain sockets", type);
374
375 printf("Proto RefCnt Flags\t Type\t State\t %s Path\n",
376 (toys.optflags&FLAG_p) ? "PID/Program Name" : "I-Node");
377 show_unix_sockets();
378 }
379
380 if ((toys.optflags & FLAG_p) && CFG_TOYBOX_FREE)
381 llist_traverse(TT.inodes, free);
382 toys.exitval = 0;
383}
384