Annotation of embedaddon/mtr/ui/curses.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 "mtr.h"
! 22:
! 23: #include <strings.h>
! 24: #include <unistd.h>
! 25:
! 26: #include <ctype.h>
! 27: #include <stdlib.h>
! 28: #include <string.h>
! 29: #include <time.h>
! 30:
! 31: /* MacOSX may need this before socket.h...*/
! 32: #if defined(HAVE_SYS_TYPES_H)
! 33: #include <sys/types.h>
! 34: #endif
! 35:
! 36: #include <sys/socket.h>
! 37: #include <netinet/in.h>
! 38: #include <arpa/inet.h>
! 39:
! 40: #if defined(HAVE_NCURSES_H)
! 41: #include <ncurses.h>
! 42: #elif defined(HAVE_NCURSES_CURSES_H)
! 43: #include <ncurses/curses.h>
! 44: #elif defined(HAVE_CURSES_H)
! 45: #include <curses.h>
! 46: #elif defined(HAVE_CURSESX_H)
! 47: #include <cursesX.h>
! 48: #else
! 49: #error No curses header file available
! 50: #endif
! 51:
! 52: /* This go-around is needed only when compiling with antique version of curses.
! 53: getmaxyx is part of Technical Standard X/Open Curses Issue 4, Version 2 (1996).
! 54: http://pubs.opengroup.org/onlinepubs/9693989999/toc.pdf see page 106 */
! 55: #ifndef getmaxyx
! 56: #define getmaxyx(win,y,x) ((y) = (win)->_maxy + 1, (x) = (win)->_maxx + 1)
! 57: #endif
! 58:
! 59: #include "mtr.h"
! 60: #include "mtr-curses.h"
! 61: #include "net.h"
! 62: #include "dns.h"
! 63: #include "asn.h"
! 64: #include "display.h"
! 65: #include "utils.h"
! 66:
! 67:
! 68: enum { NUM_FACTORS = 8 };
! 69: static double factors[NUM_FACTORS];
! 70: static int scale[NUM_FACTORS];
! 71: static char block_map[NUM_FACTORS];
! 72:
! 73: enum { black = 1, red, green, yellow, blue, magenta, cyan, white };
! 74: static const int block_col[NUM_FACTORS + 1] = {
! 75: COLOR_PAIR(red) | A_BOLD,
! 76: A_NORMAL,
! 77: COLOR_PAIR(green),
! 78: COLOR_PAIR(green) | A_BOLD,
! 79: COLOR_PAIR(yellow) | A_BOLD,
! 80: COLOR_PAIR(magenta) | A_BOLD,
! 81: COLOR_PAIR(magenta),
! 82: COLOR_PAIR(red),
! 83: COLOR_PAIR(red) | A_BOLD
! 84: };
! 85:
! 86: static void pwcenter(
! 87: char *str)
! 88: {
! 89: int maxx;
! 90: size_t cx;
! 91: int __unused_int ATTRIBUTE_UNUSED;
! 92:
! 93: getmaxyx(stdscr, __unused_int, maxx);
! 94: cx = (size_t) (maxx - strlen(str)) / 2;
! 95: printw("%*s%s", (int) cx, "", str);
! 96: }
! 97:
! 98:
! 99: static char *format_number(
! 100: int n,
! 101: int w,
! 102: char *buf)
! 103: {
! 104: if (w != 5)
! 105: /* XXX todo: implement w != 5.. */
! 106: snprintf(buf, w + 1, "%s", "unimpl");
! 107: else if (n < 100000)
! 108: /* buf is good as-is */ ;
! 109: else if (n < 1000000)
! 110: snprintf(buf, w + 1, "%3dk%1d", n / 1000, (n % 1000) / 100);
! 111: else if (n < 10000000)
! 112: snprintf(buf, w + 1, "%1dM%03d", n / 1000000,
! 113: (n % 1000000) / 1000);
! 114: else if (n < 100000000)
! 115: snprintf(buf, w + 1, "%2dM%02d", n / 1000000,
! 116: (n % 1000000) / 10000);
! 117: else if (n < 1000000000)
! 118: snprintf(buf, w + 1, "%3dM%01d", n / 1000000,
! 119: (n % 1000000) / 100000);
! 120: else /* if (n < 10000000000) */
! 121: snprintf(buf, w + 1, "%1dG%03d", n / 1000000000,
! 122: (n % 1000000000) / 1000000);
! 123:
! 124: return buf;
! 125: }
! 126:
! 127:
! 128: int mtr_curses_keyaction(
! 129: struct mtr_ctl *ctl)
! 130: {
! 131: int c = getch();
! 132: int i = 0;
! 133: float f = 0.0;
! 134: char buf[MAXFLD + 1];
! 135:
! 136: if (c == 'Q') { /* must be checked before c = tolower(c) */
! 137: mvprintw(2, 0, "Type of Service(tos): %d\n", ctl->tos);
! 138: mvprintw(3, 0,
! 139: "default 0x00, min cost 0x02, rel 0x04,, thr 0x08, low del 0x10...\n");
! 140: move(2, 22);
! 141: refresh();
! 142: while ((c = getch()) != '\n' && i < MAXFLD) {
! 143: attron(A_BOLD);
! 144: printw("%c", c);
! 145: attroff(A_BOLD);
! 146: refresh();
! 147: buf[i++] = c; /* need more checking on 'c' */
! 148: }
! 149: buf[i] = '\0';
! 150: ctl->tos = atoi(buf);
! 151: if (ctl->tos > 255 || ctl->tos < 0)
! 152: ctl->tos = 0;
! 153: return ActionNone;
! 154: }
! 155:
! 156: c = tolower(c);
! 157:
! 158: switch (c) {
! 159: case 'q':
! 160: case 3:
! 161: return ActionQuit;
! 162: case 12:
! 163: return ActionClear;
! 164: case 19:
! 165: case 'p':
! 166: return ActionPause;
! 167: case 17:
! 168: case ' ':
! 169: return ActionResume;
! 170: case 'r':
! 171: return ActionReset;
! 172: case 'd':
! 173: return ActionDisplay;
! 174: case 'e':
! 175: return ActionMPLS;
! 176: case 'n':
! 177: return ActionDNS;
! 178: #ifdef HAVE_IPINFO
! 179: case 'y':
! 180: return ActionII;
! 181: case 'z':
! 182: return ActionAS;
! 183: #endif
! 184: case '+':
! 185: return ActionScrollDown;
! 186: case '-':
! 187: return ActionScrollUp;
! 188: case 's':
! 189: mvprintw(2, 0, "Change Packet Size: %d\n", ctl->cpacketsize);
! 190: mvprintw(3, 0, "Size Range: %d-%d, < 0:random.\n", MINPACKET,
! 191: MAXPACKET);
! 192: move(2, 20);
! 193: refresh();
! 194: while ((c = getch()) != '\n' && i < MAXFLD) {
! 195: attron(A_BOLD);
! 196: printw("%c", c);
! 197: attroff(A_BOLD);
! 198: refresh();
! 199: buf[i++] = c; /* need more checking on 'c' */
! 200: }
! 201: buf[i] = '\0';
! 202: ctl->cpacketsize = atoi(buf);
! 203: return ActionNone;
! 204: case 'b':
! 205: mvprintw(2, 0, "Ping Bit Pattern: %d\n", ctl->bitpattern);
! 206: mvprintw(3, 0, "Pattern Range: 0(0x00)-255(0xff), <0 random.\n");
! 207: move(2, 18);
! 208: refresh();
! 209: while ((c = getch()) != '\n' && i < MAXFLD) {
! 210: attron(A_BOLD);
! 211: printw("%c", c);
! 212: attroff(A_BOLD);
! 213: refresh();
! 214: buf[i++] = c; /* need more checking on 'c' */
! 215: }
! 216: buf[i] = '\0';
! 217: ctl->bitpattern = atoi(buf);
! 218: if (ctl->bitpattern > 255)
! 219: ctl->bitpattern = -1;
! 220: return ActionNone;
! 221: case 'i':
! 222: mvprintw(2, 0, "Interval : %0.0f\n\n", ctl->WaitTime);
! 223: move(2, 11);
! 224: refresh();
! 225: while ((c = getch()) != '\n' && i < MAXFLD) {
! 226: attron(A_BOLD);
! 227: printw("%c", c);
! 228: attroff(A_BOLD);
! 229: refresh();
! 230: buf[i++] = c; /* need more checking on 'c' */
! 231: }
! 232: buf[i] = '\0';
! 233:
! 234: f = atof(buf);
! 235:
! 236: if (f <= 0.0)
! 237: return ActionNone;
! 238: if (getuid() != 0 && f < 1.0)
! 239: return ActionNone;
! 240: ctl->WaitTime = f;
! 241:
! 242: return ActionNone;
! 243: case 'f':
! 244: mvprintw(2, 0, "First TTL: %d\n\n", ctl->fstTTL);
! 245: move(2, 11);
! 246: refresh();
! 247: while ((c = getch()) != '\n' && i < MAXFLD) {
! 248: attron(A_BOLD);
! 249: printw("%c", c);
! 250: attroff(A_BOLD);
! 251: refresh();
! 252: buf[i++] = c; /* need more checking on 'c' */
! 253: }
! 254: buf[i] = '\0';
! 255: i = atoi(buf);
! 256:
! 257: if (i < 1 || i > ctl->maxTTL)
! 258: return ActionNone;
! 259: ctl->fstTTL = i;
! 260:
! 261: return ActionNone;
! 262: case 'm':
! 263: mvprintw(2, 0, "Max TTL: %d\n\n", ctl->maxTTL);
! 264: move(2, 9);
! 265: refresh();
! 266: while ((c = getch()) != '\n' && i < MAXFLD) {
! 267: attron(A_BOLD);
! 268: printw("%c", c);
! 269: attroff(A_BOLD);
! 270: refresh();
! 271: buf[i++] = c; /* need more checking on 'c' */
! 272: }
! 273: buf[i] = '\0';
! 274: i = atoi(buf);
! 275:
! 276: if (i < ctl->fstTTL || i > (MaxHost - 1))
! 277: return ActionNone;
! 278: ctl->maxTTL = i;
! 279:
! 280: return ActionNone;
! 281: /* fields to display & their ordering */
! 282: case 'o':
! 283: mvprintw(2, 0, "Fields: %s\n\n", ctl->fld_active);
! 284:
! 285: for (i = 0; i < MAXFLD; i++) {
! 286: if (data_fields[i].descr != NULL)
! 287: printw(" %s\n", data_fields[i].descr);
! 288: }
! 289: printw("\n");
! 290: move(2, 8); /* length of "Fields: " */
! 291: refresh();
! 292:
! 293: i = 0;
! 294: while ((c = getch()) != '\n' && i < MAXFLD) {
! 295: if (strchr(ctl->available_options, c)) {
! 296: attron(A_BOLD);
! 297: printw("%c", c);
! 298: attroff(A_BOLD);
! 299: refresh();
! 300: buf[i++] = c; /* Only permit values in "available_options" be entered */
! 301: } else {
! 302: printf("\a"); /* Illegal character. Beep, ring the bell. */
! 303: }
! 304: }
! 305: buf[i] = '\0';
! 306: if (strlen(buf) > 0)
! 307: xstrncpy(ctl->fld_active, buf, 2 * MAXFLD);
! 308:
! 309: return ActionNone;
! 310: case 'j':
! 311: if (strchr(ctl->fld_active, 'N'))
! 312: /* GeoMean and jitter */
! 313: xstrncpy(ctl->fld_active, "DR AGJMXI", 2 * MAXFLD);
! 314: else
! 315: /* default */
! 316: xstrncpy(ctl->fld_active, "LS NABWV", 2 * MAXFLD);
! 317: return ActionNone;
! 318: case 'u':
! 319: switch (ctl->mtrtype) {
! 320: case IPPROTO_ICMP:
! 321: case IPPROTO_TCP:
! 322: ctl->mtrtype = IPPROTO_UDP;
! 323: break;
! 324: case IPPROTO_UDP:
! 325: ctl->mtrtype = IPPROTO_ICMP;
! 326: break;
! 327: }
! 328: return ActionNone;
! 329: case 't':
! 330: switch (ctl->mtrtype) {
! 331: case IPPROTO_ICMP:
! 332: case IPPROTO_UDP:
! 333: ctl->mtrtype = IPPROTO_TCP;
! 334: break;
! 335: case IPPROTO_TCP:
! 336: ctl->mtrtype = IPPROTO_ICMP;
! 337: break;
! 338: }
! 339: return ActionNone;
! 340: /* reserve to display help message -Min */
! 341: case '?':
! 342: case 'h':
! 343: mvprintw(2, 0, "Command:\n");
! 344: printw(" ?|h help\n");
! 345: printw(" p pause (SPACE to resume)\n");
! 346: printw(" d switching display mode\n");
! 347: printw(" e toggle MPLS information on/off\n");
! 348: printw(" n toggle DNS on/off\n");
! 349: printw(" r reset all counters\n");
! 350: printw
! 351: (" o str set the columns to display, default str='LRS N BAWV'\n");
! 352: printw
! 353: (" j toggle latency(LS NABWV)/jitter(DR AGJMXI) stats\n");
! 354: printw(" c <n> report cycle n, default n=infinite\n");
! 355: printw
! 356: (" i <n> set the ping interval to n seconds, default n=1\n");
! 357: printw
! 358: (" f <n> set the initial time-to-live(ttl), default n=1\n");
! 359: printw
! 360: (" m <n> set the max time-to-live, default n= # of hops\n");
! 361: printw(" s <n> set the packet size to n or random(n<0)\n");
! 362: printw
! 363: (" b <c> set ping bit pattern to c(0..255) or random(c<0)\n");
! 364: printw(" Q <t> set ping packet's TOS to t\n");
! 365: printw(" u switch between ICMP ECHO and UDP datagrams\n");
! 366: #ifdef HAVE_IPINFO
! 367: printw(" y switching IP info\n");
! 368: printw(" z toggle ASN info on/off\n");
! 369: #endif
! 370: printw("\n");
! 371: printw(" press any key to go back...");
! 372: getch(); /* read and ignore 'any key' */
! 373: return ActionNone;
! 374: default: /* ignore unknown input */
! 375: return ActionNone;
! 376: }
! 377: }
! 378:
! 379:
! 380: static void format_field(
! 381: char *dst,
! 382: int dst_length,
! 383: const char *format,
! 384: int n)
! 385: {
! 386: if (index(format, 'N')) {
! 387: *dst++ = ' ';
! 388: format_number(n, 5, dst);
! 389: } else if (strchr(format, 'f')) {
! 390: /* this is for fields where we measure integer microseconds but
! 391: display floating point miliseconds. Convert to float here. */
! 392: snprintf(dst, dst_length, format, n / 1000.0);
! 393: /* this was marked as a temporary hack over 10 years ago. -- REW */
! 394: } else {
! 395: snprintf(dst, dst_length, format, n);
! 396: }
! 397: }
! 398:
! 399: static void mtr_curses_hosts(
! 400: struct mtr_ctl *ctl,
! 401: int startstat)
! 402: {
! 403: int max;
! 404: int at;
! 405: struct mplslen *mpls, *mplss;
! 406: ip_t *addr, *addrs;
! 407: int y;
! 408: char *name;
! 409:
! 410: int i, j, k;
! 411: int hd_len;
! 412: char buf[1024];
! 413: int __unused_int ATTRIBUTE_UNUSED;
! 414:
! 415: max = net_max(ctl);
! 416:
! 417: for (at = net_min(ctl) + ctl->display_offset; at < max; at++) {
! 418: printw("%2d. ", at + 1);
! 419: addr = net_addr(at);
! 420: mpls = net_mpls(at);
! 421:
! 422: if (addrcmp((void *) addr, (void *) &ctl->unspec_addr, ctl->af) !=
! 423: 0) {
! 424: name = dns_lookup(ctl, addr);
! 425: if (!net_up(at))
! 426: attron(A_BOLD);
! 427: #ifdef HAVE_IPINFO
! 428: if (is_printii(ctl))
! 429: printw(fmt_ipinfo(ctl, addr));
! 430: #endif
! 431: if (name != NULL) {
! 432: if (ctl->show_ips)
! 433: printw("%s (%s)", name, strlongip(ctl, addr));
! 434: else
! 435: printw("%s", name);
! 436: } else {
! 437: printw("%s", strlongip(ctl, addr));
! 438: }
! 439: attroff(A_BOLD);
! 440:
! 441: getyx(stdscr, y, __unused_int);
! 442: move(y, startstat);
! 443:
! 444: /* net_xxx returns times in usecs. Just display millisecs */
! 445: hd_len = 0;
! 446: for (i = 0; i < MAXFLD; i++) {
! 447: /* Ignore options that don't exist */
! 448: /* On the other hand, we now check the input side. Shouldn't happen,
! 449: can't be careful enough. */
! 450: j = ctl->fld_index[ctl->fld_active[i]];
! 451: if (j == -1)
! 452: continue;
! 453: format_field(buf + hd_len, sizeof(buf) - hd_len,
! 454: data_fields[j].format,
! 455: data_fields[j].net_xxx(at));
! 456: hd_len += data_fields[j].length;
! 457: }
! 458: buf[hd_len] = 0;
! 459: printw("%s", buf);
! 460:
! 461: for (k = 0; k < mpls->labels && ctl->enablempls; k++) {
! 462: printw("\n [MPLS: Lbl %lu Exp %u S %u TTL %u]",
! 463: mpls->label[k], mpls->exp[k], mpls->s[k],
! 464: mpls->ttl[k]);
! 465: }
! 466:
! 467: /* Multi path */
! 468: for (i = 0; i < MAXPATH; i++) {
! 469: addrs = net_addrs(at, i);
! 470: mplss = net_mplss(at, i);
! 471: if (addrcmp((void *) addrs, (void *) addr, ctl->af) == 0)
! 472: continue;
! 473: if (addrcmp
! 474: ((void *) addrs, (void *) &ctl->unspec_addr,
! 475: ctl->af) == 0)
! 476: break;
! 477:
! 478: name = dns_lookup(ctl, addrs);
! 479: if (!net_up(at))
! 480: attron(A_BOLD);
! 481: printw("\n ");
! 482: #ifdef HAVE_IPINFO
! 483: if (is_printii(ctl))
! 484: printw(fmt_ipinfo(ctl, addrs));
! 485: #endif
! 486: if (name != NULL) {
! 487: if (ctl->show_ips)
! 488: printw("%s (%s)", name, strlongip(ctl, addrs));
! 489: else
! 490: printw("%s", name);
! 491: } else {
! 492: printw("%s", strlongip(ctl, addrs));
! 493: }
! 494: for (k = 0; k < mplss->labels && ctl->enablempls; k++) {
! 495: printw("\n [MPLS: Lbl %lu Exp %u S %u TTL %u]",
! 496: mplss->label[k], mplss->exp[k], mplss->s[k],
! 497: mplss->ttl[k]);
! 498: }
! 499: attroff(A_BOLD);
! 500: }
! 501:
! 502: } else {
! 503: printw("???");
! 504: }
! 505:
! 506: printw("\n");
! 507: }
! 508: move(2, 0);
! 509: }
! 510:
! 511: static void mtr_gen_scale(
! 512: struct mtr_ctl *ctl)
! 513: {
! 514: int *saved, i, max, at;
! 515: int range;
! 516: static int low_ms, high_ms;
! 517:
! 518: low_ms = 1000000;
! 519: high_ms = -1;
! 520:
! 521: for (i = 0; i < NUM_FACTORS; i++) {
! 522: scale[i] = 0;
! 523: }
! 524: max = net_max(ctl);
! 525: for (at = ctl->display_offset; at < max; at++) {
! 526: saved = net_saved_pings(at);
! 527: for (i = 0; i < SAVED_PINGS; i++) {
! 528: if (saved[i] < 0)
! 529: continue;
! 530: if (saved[i] < low_ms) {
! 531: low_ms = saved[i];
! 532: }
! 533: if (saved[i] > high_ms) {
! 534: high_ms = saved[i];
! 535: }
! 536: }
! 537: }
! 538: range = high_ms - low_ms;
! 539: for (i = 0; i < NUM_FACTORS; i++) {
! 540: scale[i] = low_ms + ((double) range * factors[i]);
! 541: }
! 542: }
! 543:
! 544: static void mtr_curses_init(
! 545: void)
! 546: {
! 547: int i;
! 548: int block_split;
! 549:
! 550: /* Initialize factors to a log scale. */
! 551: for (i = 0; i < NUM_FACTORS; i++) {
! 552: factors[i] = ((double) 1 / NUM_FACTORS) * (i + 1);
! 553: factors[i] *= factors[i]; /* Squared. */
! 554: }
! 555:
! 556: /* Initialize block_map. The block_split is always smaller than 9 */
! 557: block_split = (NUM_FACTORS - 2) / 2;
! 558: for (i = 1; i <= block_split; i++) {
! 559: block_map[i] = '0' + i;
! 560: }
! 561: for (i = block_split + 1; i < NUM_FACTORS - 1; i++) {
! 562: block_map[i] = 'a' + i - block_split - 1;
! 563: }
! 564: block_map[0] = '.';
! 565: block_map[NUM_FACTORS - 1] = '>';
! 566: }
! 567:
! 568: static void mtr_print_scaled(
! 569: int ms)
! 570: {
! 571: int i;
! 572:
! 573: for (i = 0; i < NUM_FACTORS; i++) {
! 574: if (ms <= scale[i]) {
! 575: attrset(block_col[i + 1]);
! 576: printw("%c", block_map[i]);
! 577: attrset(A_NORMAL);
! 578: return;
! 579: }
! 580: }
! 581: printw(">");
! 582: }
! 583:
! 584:
! 585: static void mtr_fill_graph(
! 586: struct mtr_ctl *ctl,
! 587: int at,
! 588: int cols)
! 589: {
! 590: int *saved;
! 591: int i;
! 592:
! 593: saved = net_saved_pings(at);
! 594: for (i = SAVED_PINGS - cols; i < SAVED_PINGS; i++) {
! 595: if (saved[i] == -2) {
! 596: printw(" ");
! 597: } else if (saved[i] == -1) {
! 598: attrset(block_col[0]);
! 599: printw("%c", '?');
! 600: attrset(A_NORMAL);
! 601: } else {
! 602: if (ctl->display_mode == DisplayModeBlockmap) {
! 603: if (saved[i] > scale[6]) {
! 604: printw("%c", block_map[NUM_FACTORS - 1]);
! 605: } else {
! 606: printw(".");
! 607: }
! 608: } else {
! 609: mtr_print_scaled(saved[i]);
! 610: }
! 611: }
! 612: }
! 613: }
! 614:
! 615:
! 616: static void mtr_curses_graph(
! 617: struct mtr_ctl *ctl,
! 618: int startstat,
! 619: int cols)
! 620: {
! 621: int max, at, y;
! 622: ip_t *addr;
! 623: char *name;
! 624: int __unused_int ATTRIBUTE_UNUSED;
! 625:
! 626: max = net_max(ctl);
! 627:
! 628: for (at = ctl->display_offset; at < max; at++) {
! 629: printw("%2d. ", at + 1);
! 630:
! 631: addr = net_addr(at);
! 632: if (!addr) {
! 633: printw("???\n");
! 634: continue;
! 635: }
! 636:
! 637: if (!net_up(at))
! 638: attron(A_BOLD);
! 639: if (addrcmp((void *) addr, (void *) &ctl->unspec_addr, ctl->af)) {
! 640: #ifdef HAVE_IPINFO
! 641: if (is_printii(ctl))
! 642: printw(fmt_ipinfo(ctl, addr));
! 643: #endif
! 644: name = dns_lookup(ctl, addr);
! 645: printw("%s", name ? name : strlongip(ctl, addr));
! 646: } else
! 647: printw("???");
! 648: attroff(A_BOLD);
! 649:
! 650: getyx(stdscr, y, __unused_int);
! 651: move(y, startstat);
! 652:
! 653: printw(" ");
! 654: mtr_fill_graph(ctl, at, cols);
! 655: printw("\n");
! 656: }
! 657: }
! 658:
! 659:
! 660: void mtr_curses_redraw(
! 661: struct mtr_ctl *ctl)
! 662: {
! 663: int maxx;
! 664: int startstat;
! 665: int rowstat;
! 666: time_t t;
! 667: int __unused_int ATTRIBUTE_UNUSED;
! 668:
! 669: int i, j;
! 670: int hd_len = 0;
! 671: char buf[1024];
! 672: char fmt[16];
! 673:
! 674:
! 675: erase();
! 676: getmaxyx(stdscr, __unused_int, maxx);
! 677:
! 678: rowstat = 5;
! 679:
! 680: move(0, 0);
! 681: attron(A_BOLD);
! 682: snprintf(buf, sizeof(buf), "%s%s%s", "My traceroute [v",
! 683: PACKAGE_VERSION, "]");
! 684: pwcenter(buf);
! 685: attroff(A_BOLD);
! 686:
! 687: mvprintw(1, 0, "%s (%s)", ctl->LocalHostname, net_localaddr());
! 688: t = time(NULL);
! 689: mvprintw(1, maxx - 25, iso_time(&t));
! 690: printw("\n");
! 691:
! 692: printw("Keys: ");
! 693: attron(A_BOLD);
! 694: printw("H");
! 695: attroff(A_BOLD);
! 696: printw("elp ");
! 697: attron(A_BOLD);
! 698: printw("D");
! 699: attroff(A_BOLD);
! 700: printw("isplay mode ");
! 701: attron(A_BOLD);
! 702: printw("R");
! 703: attroff(A_BOLD);
! 704: printw("estart statistics ");
! 705: attron(A_BOLD);
! 706: printw("O");
! 707: attroff(A_BOLD);
! 708: printw("rder of fields ");
! 709: attron(A_BOLD);
! 710: printw("q");
! 711: attroff(A_BOLD);
! 712: printw("uit\n");
! 713:
! 714: if (ctl->display_mode == DisplayModeDefault) {
! 715: for (i = 0; i < MAXFLD; i++) {
! 716: j = ctl->fld_index[ctl->fld_active[i]];
! 717: if (j < 0)
! 718: continue;
! 719:
! 720: snprintf(fmt, sizeof(fmt), "%%%ds", data_fields[j].length);
! 721: snprintf(buf + hd_len, sizeof(buf) - hd_len, fmt,
! 722: data_fields[j].title);
! 723: hd_len += data_fields[j].length;
! 724: }
! 725: attron(A_BOLD);
! 726: mvprintw(rowstat - 1, 0, " Host");
! 727: mvprintw(rowstat - 1, maxx - hd_len - 1, "%s", buf);
! 728: mvprintw(rowstat - 2, maxx - hd_len - 1,
! 729: " Packets Pings");
! 730: attroff(A_BOLD);
! 731:
! 732: move(rowstat, 0);
! 733: mtr_curses_hosts(ctl, maxx - hd_len - 1);
! 734:
! 735: } else {
! 736: char msg[80];
! 737: int padding = 30;
! 738: int max_cols;
! 739:
! 740: #ifdef HAVE_IPINFO
! 741: if (is_printii(ctl))
! 742: padding += get_iiwidth(ctl->ipinfo_no);
! 743: #endif
! 744: max_cols =
! 745: maxx <= SAVED_PINGS + padding ? maxx - padding : SAVED_PINGS;
! 746: startstat = padding - 2;
! 747:
! 748: snprintf(msg, sizeof(msg), " Last %3d pings", max_cols);
! 749: mvprintw(rowstat - 1, startstat, msg);
! 750:
! 751: attroff(A_BOLD);
! 752: move(rowstat, 0);
! 753:
! 754: mtr_gen_scale(ctl);
! 755: mtr_curses_graph(ctl, startstat, max_cols);
! 756:
! 757: printw("\n");
! 758: attron(A_BOLD);
! 759: printw("Scale:");
! 760: attroff(A_BOLD);
! 761:
! 762: for (i = 0; i < NUM_FACTORS - 1; i++) {
! 763: printw(" ");
! 764: attrset(block_col[i + 1]);
! 765: printw("%c", block_map[i]);
! 766: attrset(A_NORMAL);
! 767: printw(":%d ms", scale[i] / 1000);
! 768: }
! 769: printw(" ");
! 770: attrset(block_col[NUM_FACTORS]);
! 771: printw("%c", block_map[NUM_FACTORS - 1]);
! 772: attrset(A_NORMAL);
! 773: }
! 774:
! 775: refresh();
! 776: }
! 777:
! 778:
! 779: void mtr_curses_open(
! 780: struct mtr_ctl *ctl)
! 781: {
! 782: int bg_col = 0;
! 783: int i;
! 784:
! 785: initscr();
! 786: raw();
! 787: noecho();
! 788: start_color();
! 789: if (use_default_colors() == OK)
! 790: bg_col = -1;
! 791: for (i = 0; i < NUM_FACTORS; i++)
! 792: init_pair(i + 1, i, bg_col);
! 793:
! 794: mtr_curses_init();
! 795: mtr_curses_redraw(ctl);
! 796: }
! 797:
! 798:
! 799: void mtr_curses_close(
! 800: void)
! 801: {
! 802: printw("\n");
! 803: endwin();
! 804: }
! 805:
! 806:
! 807: void mtr_curses_clear(
! 808: struct mtr_ctl *ctl)
! 809: {
! 810: mtr_curses_close();
! 811: mtr_curses_open(ctl);
! 812: }
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>