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>