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