Annotation of embedaddon/mtr/ui/mtr.c, revision 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>