Annotation of embedaddon/mtr/ui/mtr.c, revision 1.1.1.1
1.1 misho 1: /*
2: mtr -- a network diagnostic tool
3: Copyright (C) 1997,1998 Matt Kimball
4:
5: This program is free software; you can redistribute it and/or modify
6: it under the terms of the GNU General Public License version 2 as
7: published by the Free Software Foundation.
8:
9: This program is distributed in the hope that it will be useful,
10: but WITHOUT ANY WARRANTY; without even the implied warranty of
11: MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12: GNU General Public License for more details.
13:
14: You should have received a copy of the GNU General Public License
15: along with this program; if not, write to the Free Software
16: Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
17: */
18:
19: #include "config.h"
20:
21: #include <sys/types.h>
22: #include <stdio.h>
23: #include <stdlib.h>
24: #include <string.h>
25: #include <unistd.h>
26: #include <errno.h>
27: #include <string.h>
28: #include <strings.h>
29: #ifdef HAVE_ERROR_H
30: #include <error.h>
31: #else
32: #include "portability/error.h"
33: #endif
34: #ifdef HAVE_VALUES_H
35: #include <values.h>
36: #endif
37: #ifdef HAVE_SYS_LIMITS_H
38: #include <sys/limits.h>
39: #endif
40:
41: #include <netdb.h>
42: #include <netinet/in.h>
43: #include <sys/socket.h>
44: #include <ctype.h>
45: #include <assert.h>
46: #include <fcntl.h>
47: #include <limits.h>
48: #include <sys/stat.h>
49: #include <sys/time.h>
50:
51: #include "mtr.h"
52: #include "mtr-curses.h"
53: #include "display.h"
54: #include "dns.h"
55: #include "report.h"
56: #include "net.h"
57: #include "asn.h"
58: #include "utils.h"
59:
60: #ifdef HAVE_GETOPT
61: #include <getopt.h>
62: #else
63: #include "portability/getopt.h"
64: #endif
65:
66: #ifdef ENABLE_IPV6
67: #define DEFAULT_AF AF_UNSPEC
68: #else
69: #define DEFAULT_AF AF_INET
70: #endif
71:
72:
73: const struct fields data_fields[MAXFLD] = {
74: /* key, Remark, Header, Format, Width, CallBackFunc */
75: {' ', "<sp>: Space between fields", " ", " ", 1, &net_drop},
76: {'L', "L: Loss Ratio", "Loss%", " %4.1f%%", 6, &net_loss},
77: {'D', "D: Dropped Packets", "Drop", " %4d", 5, &net_drop},
78: {'R', "R: Received Packets", "Rcv", " %5d", 6, &net_returned},
79: {'S', "S: Sent Packets", "Snt", " %5d", 6, &net_xmit},
80: {'N', "N: Newest RTT(ms)", "Last", " %5.1f", 6, &net_last},
81: {'B', "B: Min/Best RTT(ms)", "Best", " %5.1f", 6, &net_best},
82: {'A', "A: Average RTT(ms)", "Avg", " %5.1f", 6, &net_avg},
83: {'W', "W: Max/Worst RTT(ms)", "Wrst", " %5.1f", 6, &net_worst},
84: {'V', "V: Standard Deviation", "StDev", " %5.1f", 6, &net_stdev},
85: {'G', "G: Geometric Mean", "Gmean", " %5.1f", 6, &net_gmean},
86: {'J', "J: Current Jitter", "Jttr", " %4.1f", 5, &net_jitter},
87: {'M', "M: Jitter Mean/Avg.", "Javg", " %4.1f", 5, &net_javg},
88: {'X', "X: Worst Jitter", "Jmax", " %4.1f", 5, &net_jworst},
89: {'I', "I: Interarrival Jitter", "Jint", " %4.1f", 5, &net_jinta},
90: {'\0', NULL, NULL, NULL, 0, NULL}
91: };
92:
93: typedef struct names {
94: char *name;
95: struct names *next;
96: } names_t;
97:
98: static void __attribute__ ((__noreturn__)) usage(FILE * out)
99: {
100: fputs("\nUsage:\n", out);
101: fputs(" mtr [options] hostname\n", out);
102: fputs("\n", out);
103: fputs(" -F, --filename FILE read hostname(s) from a file\n",
104: out);
105: fputs(" -4 use IPv4 only\n", out);
106: #ifdef ENABLE_IPV6
107: fputs(" -6 use IPv6 only\n", out);
108: #endif
109: fputs(" -u, --udp use UDP instead of ICMP echo\n",
110: out);
111: fputs(" -T, --tcp use TCP instead of ICMP echo\n",
112: out);
113: fputs
114: (" -a, --address ADDRESS bind the outgoing socket to ADDRESS\n",
115: out);
116: fputs(" -f, --first-ttl NUMBER set what TTL to start\n", out);
117: fputs(" -m, --max-ttl NUMBER maximum number of hops\n", out);
118: fputs(" -U, --max-unknown NUMBER maximum unknown host\n", out);
119: fputs
120: (" -P, --port PORT target port number for TCP, SCTP, or UDP\n",
121: out);
122: fputs(" -L, --localport LOCALPORT source port number for UDP\n", out);
123: fputs
124: (" -s, --psize PACKETSIZE set the packet size used for probing\n",
125: out);
126: fputs
127: (" -B, --bitpattern NUMBER set bit pattern to use in payload\n",
128: out);
129: fputs(" -i, --interval SECONDS ICMP echo request interval\n", out);
130: fputs
131: (" -G, --gracetime SECONDS number of seconds to wait for responses\n",
132: out);
133: fputs
134: (" -Q, --tos NUMBER type of service field in IP header\n",
135: out);
136: fputs
137: (" -e, --mpls display information from ICMP extensions\n",
138: out);
139: fputs
140: (" -Z, --timeout SECONDS seconds to keep probe sockets open\n",
141: out);
142: #ifdef SO_MARK
143: fputs(" -M, --mark MARK mark each sent packet\n", out);
144: #endif
145: fputs(" -r, --report output using report mode\n", out);
146: fputs(" -w, --report-wide output wide report\n", out);
147: fputs(" -c, --report-cycles COUNT set the number of pings sent\n",
148: out);
149: fputs(" -j, --json output json\n", out);
150: fputs(" -x, --xml output xml\n", out);
151: fputs(" -C, --csv output comma separated values\n",
152: out);
153: fputs(" -l, --raw output raw format\n", out);
154: fputs(" -p, --split split output\n", out);
155: #ifdef HAVE_CURSES
156: fputs(" -t, --curses use curses terminal interface\n",
157: out);
158: #endif
159: fputs(" --displaymode MODE select initial display mode\n",
160: out);
161: #ifdef HAVE_GTK
162: fputs(" -g, --gtk use GTK+ xwindow interface\n", out);
163: #endif
164: fputs(" -n, --no-dns do not resove host names\n", out);
165: fputs(" -b, --show-ips show IP numbers and host names\n",
166: out);
167: fputs(" -o, --order FIELDS select output fields\n", out);
168: #ifdef HAVE_IPINFO
169: fputs(" -y, --ipinfo NUMBER select IP information in output\n",
170: out);
171: fputs(" -z, --aslookup display AS number\n", out);
172: #endif
173: fputs(" -h, --help display this help and exit\n", out);
174: fputs
175: (" -v, --version output version information and exit\n",
176: out);
177: fputs("\n", out);
178: fputs("See the 'man 8 mtr' for details.\n", out);
179: exit(out == stderr ? EXIT_FAILURE : EXIT_SUCCESS);
180: }
181:
182:
183: static void append_to_names(
184: names_t ** names_head,
185: const char *item)
186: {
187: names_t **name_tail = names_head;
188:
189: while (*name_tail) {
190: name_tail = &(*name_tail)->next;
191: }
192:
193: names_t *name = calloc(1, sizeof(names_t));
194: if (name == NULL) {
195: error(EXIT_FAILURE, errno, "memory allocation failure");
196: }
197: name->name = xstrdup(item);
198: name->next = NULL;
199:
200: *name_tail = name;
201: }
202:
203: static void read_from_file(
204: names_t ** names,
205: const char *filename)
206: {
207:
208: FILE *in;
209: char line[512];
210:
211: if (!filename || strcmp(filename, "-") == 0) {
212: clearerr(stdin);
213: in = stdin;
214: } else {
215: in = fopen(filename, "r");
216: if (!in) {
217: error(EXIT_FAILURE, errno, "open %s", filename);
218: }
219: }
220:
221: while (fgets(line, sizeof(line), in)) {
222: char *name = trim(line, '\0');
223: append_to_names(names, name);
224: }
225:
226: if (ferror(in)) {
227: error(EXIT_FAILURE, errno, "ferror %s", filename);
228: }
229:
230: if (in != stdin)
231: fclose(in);
232: }
233:
234: /*
235: * If the file stream is associated with a regular file, lock the file
236: * in order coordinate writes to a common file from multiple mtr
237: * instances. This is useful if, for example, multiple mtr instances
238: * try to append results to a common file.
239: */
240:
241: static void lock(
242: FILE * f)
243: {
244: int fd;
245: struct stat buf;
246: static struct flock lock;
247:
248: assert(f);
249:
250: lock.l_type = F_WRLCK;
251: lock.l_start = 0;
252: lock.l_whence = SEEK_END;
253: lock.l_len = 0;
254: lock.l_pid = getpid();
255:
256: fd = fileno(f);
257: if ((fstat(fd, &buf) == 0) && S_ISREG(buf.st_mode)) {
258: if (fcntl(fd, F_SETLKW, &lock) == -1) {
259: error(0, errno, "fcntl (ignored)");
260: }
261: }
262: }
263:
264: /*
265: * If the file stream is associated with a regular file, unlock the
266: * file (which presumably has previously been locked).
267: */
268:
269: static void unlock(
270: FILE * f)
271: {
272: int fd;
273: struct stat buf;
274: static struct flock lock;
275:
276: assert(f);
277:
278: lock.l_type = F_UNLCK;
279: lock.l_start = 0;
280: lock.l_whence = SEEK_END;
281: lock.l_len = 0;
282: lock.l_pid = getpid();
283:
284: fd = fileno(f);
285: if ((fstat(fd, &buf) == 0) && S_ISREG(buf.st_mode)) {
286: if (fcntl(fd, F_SETLKW, &lock) == -1) {
287: error(0, errno, "fcntl (ignored)");
288: }
289: }
290: }
291:
292:
293: static void init_fld_options(
294: struct mtr_ctl *ctl)
295: {
296: int i;
297:
298: memset(ctl->fld_index, -1, FLD_INDEX_SZ);
299:
300: for (i = 0; data_fields[i].key != 0; i++) {
301: ctl->available_options[i] = data_fields[i].key;
302: ctl->fld_index[data_fields[i].key] = i;
303: }
304: ctl->available_options[i] = 0;
305: }
306:
307:
308: static void parse_arg(
309: struct mtr_ctl *ctl,
310: names_t ** names,
311: int argc,
312: char **argv)
313: {
314: int opt;
315: int i;
316: /* IMPORTANT: when adding or modifying an option:
317: 0/ try to find a somewhat logical order;
318: 1/ add the long option name in "long_options" below;
319: 2/ update the man page (use the same order);
320: 3/ update the help message (see usage() function).
321: */
322: enum {
323: OPT_DISPLAYMODE = CHAR_MAX + 1
324: };
325: static const struct option long_options[] = {
326: /* option name, has argument, NULL, short name */
327: {"help", 0, NULL, 'h'},
328: {"version", 0, NULL, 'v'},
329:
330: {"inet", 0, NULL, '4'}, /* IPv4 only */
331: #ifdef ENABLE_IPV6
332: {"inet6", 0, NULL, '6'}, /* IPv6 only */
333: #endif
334: {"filename", 1, NULL, 'F'},
335:
336: {"report", 0, NULL, 'r'},
337: {"report-wide", 0, NULL, 'w'},
338: {"xml", 0, NULL, 'x'},
339: #ifdef HAVE_CURSES
340: {"curses", 0, NULL, 't'},
341: #endif
342: #ifdef HAVE_GTK
343: {"gtk", 0, NULL, 'g'},
344: #endif
345: {"raw", 0, NULL, 'l'},
346: {"csv", 0, NULL, 'C'},
347: {"json", 0, NULL, 'j'},
348: {"displaymode", 1, NULL, OPT_DISPLAYMODE},
349: {"split", 0, NULL, 'p'}, /* BL */
350: /* maybe above should change to -d 'x' */
351:
352: {"no-dns", 0, NULL, 'n'},
353: {"show-ips", 0, NULL, 'b'},
354: {"order", 1, NULL, 'o'}, /* fields to display & their order */
355: #ifdef HAVE_IPINFO
356: {"ipinfo", 1, NULL, 'y'}, /* IP info lookup */
357: {"aslookup", 0, NULL, 'z'}, /* Do AS lookup (--ipinfo 0) */
358: #endif
359:
360: {"interval", 1, NULL, 'i'},
361: {"report-cycles", 1, NULL, 'c'},
362: {"psize", 1, NULL, 's'}, /* overload psize<0, ->rand(min,max) */
363: {"bitpattern", 1, NULL, 'B'}, /* overload B>255, ->rand(0,255) */
364: {"tos", 1, NULL, 'Q'}, /* typeof service (0,255) */
365: {"mpls", 0, NULL, 'e'},
366: {"address", 1, NULL, 'a'},
367: {"first-ttl", 1, NULL, 'f'}, /* -f & -m are borrowed from traceroute */
368: {"max-ttl", 1, NULL, 'm'},
369: {"max-unknown", 1, NULL, 'U'},
370: {"udp", 0, NULL, 'u'}, /* UDP (default is ICMP) */
371: {"tcp", 0, NULL, 'T'}, /* TCP (default is ICMP) */
372: #ifdef HAS_SCTP
373: {"sctp", 0, NULL, 'S'}, /* SCTP (default is ICMP) */
374: #endif
375: {"port", 1, NULL, 'P'}, /* target port number for TCP/SCTP/UDP */
376: {"localport", 1, NULL, 'L'}, /* source port number for UDP */
377: {"timeout", 1, NULL, 'Z'}, /* timeout for probe sockets */
378: {"gracetime", 1, NULL, 'G'}, /* gracetime for replies after last probe */
379: #ifdef SO_MARK
380: {"mark", 1, NULL, 'M'}, /* use SO_MARK */
381: #endif
382: {NULL, 0, NULL, 0}
383: };
384: enum { num_options = sizeof(long_options) / sizeof(struct option) };
385: char short_options[num_options * 2];
386: size_t n, p;
387:
388: for (n = p = 0; n < num_options; n++) {
389: if (CHAR_MAX < long_options[n].val) {
390: continue;
391: }
392: short_options[p] = long_options[n].val;
393: p++;
394: if (long_options[n].has_arg == 1) {
395: short_options[p] = ':';
396: p++;
397: }
398: /* optional options need two ':', but ignore them now as they are not in use */
399: }
400:
401: opt = 0;
402: while (1) {
403: opt = getopt_long(argc, argv, short_options, long_options, NULL);
404: if (opt == -1)
405: break;
406:
407: switch (opt) {
408: case 'v':
409: printf("mtr " PACKAGE_VERSION "\n");
410: exit(EXIT_SUCCESS);
411: break;
412: case 'h':
413: usage(stdout);
414: break;
415:
416: case 'r':
417: ctl->DisplayMode = DisplayReport;
418: break;
419: case 'w':
420: ctl->reportwide = 1;
421: ctl->DisplayMode = DisplayReport;
422: break;
423: #ifdef HAVE_CURSES
424: case 't':
425: ctl->DisplayMode = DisplayCurses;
426: break;
427: #endif
428: #ifdef HAVE_GTK
429: case 'g':
430: ctl->DisplayMode = DisplayGTK;
431: break;
432: #endif
433: case 'p': /* BL */
434: ctl->DisplayMode = DisplaySplit;
435: break;
436: case 'l':
437: ctl->DisplayMode = DisplayRaw;
438: break;
439: case 'C':
440: ctl->DisplayMode = DisplayCSV;
441: break;
442: case 'j':
443: ctl->DisplayMode = DisplayJSON;
444: break;
445: case 'x':
446: ctl->DisplayMode = DisplayXML;
447: break;
448:
449: case OPT_DISPLAYMODE:
450: ctl->display_mode =
451: strtonum_or_err(optarg, "invalid argument", STRTO_INT);
452: if ((DisplayModeMAX - 1) < ctl->display_mode)
453: error(EXIT_FAILURE, 0, "value out of range (%d - %d): %s",
454: DisplayModeDefault, (DisplayModeMAX - 1), optarg);
455: break;
456: case 'c':
457: ctl->MaxPing =
458: strtonum_or_err(optarg, "invalid argument", STRTO_INT);
459: ctl->ForceMaxPing = 1;
460: break;
461: case 's':
462: ctl->cpacketsize =
463: strtonum_or_err(optarg, "invalid argument", STRTO_INT);
464: break;
465: case 'a':
466: ctl->InterfaceAddress = optarg;
467: break;
468: case 'e':
469: ctl->enablempls = 1;
470: break;
471: case 'n':
472: ctl->dns = 0;
473: break;
474: case 'i':
475: ctl->WaitTime = strtofloat_or_err(optarg, "invalid argument");
476: if (ctl->WaitTime <= 0.0) {
477: error(EXIT_FAILURE, 0, "wait time must be positive");
478: }
479: if (getuid() != 0 && ctl->WaitTime < 1.0) {
480: error(EXIT_FAILURE, 0,
481: "non-root users cannot request an interval < 1.0 seconds");
482: }
483: break;
484: case 'f':
485: ctl->fstTTL =
486: strtonum_or_err(optarg, "invalid argument", STRTO_INT);
487: if (ctl->fstTTL > ctl->maxTTL) {
488: ctl->fstTTL = ctl->maxTTL;
489: }
490: if (ctl->fstTTL < 1) { /* prevent 0 hop */
491: ctl->fstTTL = 1;
492: }
493: break;
494: case 'F':
495: read_from_file(names, optarg);
496: break;
497: case 'm':
498: ctl->maxTTL =
499: strtonum_or_err(optarg, "invalid argument", STRTO_INT);
500: if (ctl->maxTTL > (MaxHost - 1)) {
501: ctl->maxTTL = MaxHost - 1;
502: }
503: if (ctl->maxTTL < 1) { /* prevent 0 hop */
504: ctl->maxTTL = 1;
505: }
506: if (ctl->fstTTL > ctl->maxTTL) { /* don't know the pos of -m or -f */
507: ctl->fstTTL = ctl->maxTTL;
508: }
509: break;
510: case 'U':
511: ctl->maxUnknown =
512: strtonum_or_err(optarg, "invalid argument", STRTO_INT);
513: if (ctl->maxUnknown < 1) {
514: ctl->maxUnknown = 1;
515: }
516: break;
517: case 'o':
518: /* Check option before passing it on to fld_active. */
519: if (strlen(optarg) > MAXFLD) {
520: error(EXIT_FAILURE, 0, "Too many fields: %s", optarg);
521: }
522: for (i = 0; optarg[i]; i++) {
523: if (!strchr(ctl->available_options, optarg[i])) {
524: error(EXIT_FAILURE, 0, "Unknown field identifier: %c",
525: optarg[i]);
526: }
527: }
528: xstrncpy(ctl->fld_active, optarg, 2 * MAXFLD);
529: break;
530: case 'B':
531: ctl->bitpattern =
532: strtonum_or_err(optarg, "invalid argument", STRTO_INT);
533: if (ctl->bitpattern > 255)
534: ctl->bitpattern = -1;
535: break;
536: case 'G':
537: ctl->GraceTime = strtofloat_or_err(optarg, "invalid argument");
538: if (ctl->GraceTime <= 0.0) {
539: error(EXIT_FAILURE, 0, "wait time must be positive");
540: }
541: break;
542: case 'Q':
543: ctl->tos =
544: strtonum_or_err(optarg, "invalid argument", STRTO_INT);
545: if (ctl->tos > 255 || ctl->tos < 0) {
546: /* error message, should do more checking for valid values,
547: * details in rfc2474 */
548: ctl->tos = 0;
549: }
550: break;
551: case 'u':
552: if (ctl->mtrtype != IPPROTO_ICMP) {
553: error(EXIT_FAILURE, 0,
554: "-u , -T and -S are mutually exclusive");
555: }
556: ctl->mtrtype = IPPROTO_UDP;
557: break;
558: case 'T':
559: if (ctl->mtrtype != IPPROTO_ICMP) {
560: error(EXIT_FAILURE, 0,
561: "-u , -T and -S are mutually exclusive");
562: }
563: if (!ctl->remoteport) {
564: ctl->remoteport = 80;
565: }
566: ctl->mtrtype = IPPROTO_TCP;
567: break;
568: #ifdef HAS_SCTP
569: case 'S':
570: if (ctl->mtrtype != IPPROTO_ICMP) {
571: error(EXIT_FAILURE, 0,
572: "-u , -T and -S are mutually exclusive");
573: }
574: if (!ctl->remoteport) {
575: ctl->remoteport = 80;
576: }
577: ctl->mtrtype = IPPROTO_SCTP;
578: break;
579: #endif
580: case 'b':
581: ctl->show_ips = 1;
582: break;
583: case 'P':
584: ctl->remoteport =
585: strtonum_or_err(optarg, "invalid argument", STRTO_INT);
586: if (ctl->remoteport < 1 || MaxPort < ctl->remoteport) {
587: error(EXIT_FAILURE, 0, "Illegal port number: %d",
588: ctl->remoteport);
589: }
590: break;
591: case 'L':
592: ctl->localport =
593: strtonum_or_err(optarg, "invalid argument", STRTO_INT);
594: if (ctl->localport < MinPort || MaxPort < ctl->localport) {
595: error(EXIT_FAILURE, 0, "Illegal port number: %d",
596: ctl->localport);
597: }
598: break;
599: case 'Z':
600: ctl->probe_timeout =
601: strtonum_or_err(optarg, "invalid argument", STRTO_INT);
602: ctl->probe_timeout *= 1000000;
603: break;
604: case '4':
605: ctl->af = AF_INET;
606: break;
607: #ifdef ENABLE_IPV6
608: case '6':
609: ctl->af = AF_INET6;
610: break;
611: #endif
612: #ifdef HAVE_IPINFO
613: case 'y':
614: ctl->ipinfo_no =
615: strtonum_or_err(optarg, "invalid argument", STRTO_INT);
616: if (ctl->ipinfo_no < 0 || 4 < ctl->ipinfo_no) {
617: error(EXIT_FAILURE, 0, "value %d out of range (0 - 4)",
618: ctl->ipinfo_no);
619: }
620: break;
621: case 'z':
622: ctl->ipinfo_no = 0;
623: break;
624: #endif
625: #ifdef SO_MARK
626: case 'M':
627: ctl->mark =
628: strtonum_or_err(optarg, "invalid argument", STRTO_U32INT);
629: break;
630: #endif
631: default:
632: usage(stderr);
633: }
634: }
635:
636: if (ctl->DisplayMode == DisplayReport ||
637: ctl->DisplayMode == DisplayTXT ||
638: ctl->DisplayMode == DisplayJSON ||
639: ctl->DisplayMode == DisplayXML ||
640: ctl->DisplayMode == DisplayRaw || ctl->DisplayMode == DisplayCSV)
641: ctl->Interactive = 0;
642:
643: if (optind > argc - 1)
644: return;
645:
646: }
647:
648:
649: static void parse_mtr_options(
650: struct mtr_ctl *ctl,
651: names_t ** names,
652: char *string)
653: {
654: int argc = 1;
655: char *argv[128], *p;
656:
657: if (!string)
658: return;
659: argv[0] = xstrdup(PACKAGE_NAME);
660: argc = 1;
661: p = strtok(string, " \t");
662: while (p != NULL && ((size_t) argc < (sizeof(argv) / sizeof(argv[0])))) {
663: argv[argc++] = p;
664: p = strtok(NULL, " \t");
665: }
666: if (p != NULL) {
667: error(0, 0, "Warning: extra arguments ignored: %s", p);
668: }
669:
670: parse_arg(ctl, names, argc, argv);
671: free(argv[0]);
672: optind = 0;
673: }
674:
675: static void init_rand(
676: void)
677: {
678: struct timeval tv;
679:
680: gettimeofday(&tv, NULL);
681: srand((getpid() << 16) ^ getuid() ^ tv.tv_sec ^ tv.tv_usec);
682: }
683:
684: int main(
685: int argc,
686: char **argv)
687: {
688: struct hostent *host = NULL;
689: struct addrinfo hints, *res;
690: int gai_error;
691: struct hostent trhost;
692: char *alptr[2];
693: struct sockaddr_in *sa4;
694: #ifdef ENABLE_IPV6
695: struct sockaddr_in6 *sa6;
696: #endif
697: names_t *names_head = NULL;
698: names_t *names_walk;
699:
700: struct mtr_ctl ctl;
701: memset(&ctl, 0, sizeof(ctl));
702: /* initialize non-null values */
703: ctl.Interactive = 1;
704: ctl.MaxPing = 10;
705: ctl.WaitTime = 1.0;
706: ctl.GraceTime = 5.0;
707: ctl.dns = 1;
708: ctl.use_dns = 1;
709: ctl.cpacketsize = 64;
710: ctl.af = DEFAULT_AF;
711: ctl.mtrtype = IPPROTO_ICMP;
712: ctl.fstTTL = 1;
713: ctl.maxTTL = 30;
714: ctl.maxUnknown = 12;
715: ctl.probe_timeout = 10 * 1000000;
716: ctl.ipinfo_no = -1;
717: ctl.ipinfo_max = -1;
718: xstrncpy(ctl.fld_active, "LS NABWV", 2 * MAXFLD);
719:
720: /*
721: mtr used to be suid root. It should not be with this version.
722: We'll check so that we can notify people using installation
723: mechanisms with obsolete assumptions.
724: */
725: if ((geteuid() != getuid()) || (getegid() != getgid())) {
726: error(EXIT_FAILURE, errno, "mtr should not run suid");
727: }
728:
729: /* This will check if stdout/stderr writing is successful */
730: atexit(close_stdout);
731:
732: /* reset the random seed */
733: init_rand();
734:
735: display_detect(&ctl, &argc, &argv);
736: ctl.display_mode = DisplayModeDefault;
737:
738: /* The field options are now in a static array all together,
739: but that requires a run-time initialization. */
740: init_fld_options(&ctl);
741:
742: parse_mtr_options(&ctl, &names_head, getenv("MTR_OPTIONS"));
743:
744: parse_arg(&ctl, &names_head, argc, argv);
745:
746: while (optind < argc) {
747: char *name = argv[optind++];
748: append_to_names(&names_head, name);
749: }
750:
751: /* default: localhost. */
752: if (!names_head)
753: append_to_names(&names_head, "localhost");
754:
755: names_walk = names_head;
756: while (names_walk != NULL) {
757:
758: ctl.Hostname = names_walk->name;
759: if (gethostname(ctl.LocalHostname, sizeof(ctl.LocalHostname))) {
760: xstrncpy(ctl.LocalHostname, "UNKNOWNHOST",
761: sizeof(ctl.LocalHostname));
762: }
763:
764: /* gethostbyname2() is deprecated so we'll use getaddrinfo() instead. */
765: memset(&hints, 0, sizeof hints);
766: hints.ai_family = ctl.af;
767: hints.ai_socktype = SOCK_DGRAM;
768: gai_error = getaddrinfo(ctl.Hostname, NULL, &hints, &res);
769: if (gai_error) {
770: if (gai_error == EAI_SYSTEM)
771: error(0, 0, "Failed to resolve host: %s", ctl.Hostname);
772: else
773: error(0, 0, "Failed to resolve host: %s: %s", ctl.Hostname,
774: gai_strerror(gai_error));
775:
776: if (ctl.Interactive)
777: exit(EXIT_FAILURE);
778: else {
779: names_walk = names_walk->next;
780: continue;
781: }
782: }
783: /* Convert the first addrinfo into a hostent. */
784: host = &trhost;
785: memset(host, 0, sizeof trhost);
786: host->h_name = res->ai_canonname;
787: host->h_aliases = NULL;
788: host->h_addrtype = res->ai_family;
789: ctl.af = res->ai_family;
790: host->h_length = res->ai_addrlen;
791: host->h_addr_list = alptr;
792: switch (ctl.af) {
793: case AF_INET:
794: sa4 = (struct sockaddr_in *) res->ai_addr;
795: alptr[0] = (void *) &(sa4->sin_addr);
796: break;
797: #ifdef ENABLE_IPV6
798: case AF_INET6:
799: sa6 = (struct sockaddr_in6 *) res->ai_addr;
800: alptr[0] = (void *) &(sa6->sin6_addr);
801: break;
802: #endif
803: default:
804: error(0, 0, "unknown address type");
805: if (ctl.Interactive)
806: exit(EXIT_FAILURE);
807: else {
808: names_walk = names_walk->next;
809: continue;
810: }
811: }
812: alptr[1] = NULL;
813:
814: if (net_open(&ctl, host) != 0) {
815: error(0, 0, "Unable to start net module");
816: if (ctl.Interactive)
817: exit(EXIT_FAILURE);
818: else {
819: names_walk = names_walk->next;
820: continue;
821: }
822: }
823:
824: lock(stdout);
825: dns_open(&ctl);
826: display_open(&ctl);
827:
828: display_loop(&ctl);
829:
830: net_end_transit();
831: display_close(&ctl);
832: unlock(stdout);
833:
834: if (ctl.Interactive)
835: break;
836: else
837: names_walk = names_walk->next;
838:
839: }
840:
841: net_close();
842:
843: while (names_head != NULL) {
844: names_t *item = names_head;
845: free(item->name);
846: item->name = NULL;
847: names_head = item->next;
848: free(item);
849: item = NULL;
850: }
851:
852: return 0;
853: }
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>