diff options
author | Lorenz Kästle <12514511+RincewindsHat@users.noreply.github.com> | 2024-12-29 20:10:05 +0100 |
---|---|---|
committer | Lorenz Kästle <12514511+RincewindsHat@users.noreply.github.com> | 2024-12-29 20:10:05 +0100 |
commit | 54529b601385ebf3bd0fca855e8def62ec5b6527 (patch) | |
tree | 51bf00a9e8a9ad9d1b9bdd59504b2b1ad2b0583b /plugins-root/check_icmp.c | |
parent | bb88e38a345c90465386d9eb746ea704c7343e80 (diff) | |
parent | e301a9410d7f5f9f864086696eb2003d08512b2b (diff) | |
download | monitoring-plugins-54529b601385ebf3bd0fca855e8def62ec5b6527.tar.gz |
Merge branch 'master' into check_swap_again
Diffstat (limited to 'plugins-root/check_icmp.c')
-rw-r--r-- | plugins-root/check_icmp.c | 307 |
1 files changed, 200 insertions, 107 deletions
diff --git a/plugins-root/check_icmp.c b/plugins-root/check_icmp.c index 960ab8f1..dcaceddb 100644 --- a/plugins-root/check_icmp.c +++ b/plugins-root/check_icmp.c | |||
@@ -64,6 +64,7 @@ const char *email = "devel@monitoring-plugins.org"; | |||
64 | #include <netinet/ip_icmp.h> | 64 | #include <netinet/ip_icmp.h> |
65 | #include <netinet/icmp6.h> | 65 | #include <netinet/icmp6.h> |
66 | #include <arpa/inet.h> | 66 | #include <arpa/inet.h> |
67 | #include <math.h> | ||
67 | 68 | ||
68 | /** sometimes undefined system macros (quite a few, actually) **/ | 69 | /** sometimes undefined system macros (quite a few, actually) **/ |
69 | #ifndef MAXTTL | 70 | #ifndef MAXTTL |
@@ -272,8 +273,9 @@ static void crash(const char *fmt, ...) { | |||
272 | vprintf(fmt, ap); | 273 | vprintf(fmt, ap); |
273 | va_end(ap); | 274 | va_end(ap); |
274 | 275 | ||
275 | if (errno) | 276 | if (errno) { |
276 | printf(": %s", strerror(errno)); | 277 | printf(": %s", strerror(errno)); |
278 | } | ||
277 | puts(""); | 279 | puts(""); |
278 | 280 | ||
279 | exit(3); | 281 | exit(3); |
@@ -282,8 +284,9 @@ static void crash(const char *fmt, ...) { | |||
282 | static const char *get_icmp_error_msg(unsigned char icmp_type, unsigned char icmp_code) { | 284 | static const char *get_icmp_error_msg(unsigned char icmp_type, unsigned char icmp_code) { |
283 | const char *msg = "unreachable"; | 285 | const char *msg = "unreachable"; |
284 | 286 | ||
285 | if (debug > 1) | 287 | if (debug > 1) { |
286 | printf("get_icmp_error_msg(%u, %u)\n", icmp_type, icmp_code); | 288 | printf("get_icmp_error_msg(%u, %u)\n", icmp_type, icmp_code); |
289 | } | ||
287 | switch (icmp_type) { | 290 | switch (icmp_type) { |
288 | case ICMP_UNREACH: | 291 | case ICMP_UNREACH: |
289 | switch (icmp_code) { | 292 | switch (icmp_code) { |
@@ -392,8 +395,9 @@ static int handle_random_icmp(unsigned char *packet, struct sockaddr_storage *ad | |||
392 | return 0; | 395 | return 0; |
393 | } | 396 | } |
394 | 397 | ||
395 | if (debug) | 398 | if (debug) { |
396 | printf("handle_random_icmp(%p, %p)\n", (void *)&p, (void *)addr); | 399 | printf("handle_random_icmp(%p, %p)\n", (void *)&p, (void *)addr); |
400 | } | ||
397 | 401 | ||
398 | /* only handle a few types, since others can't possibly be replies to | 402 | /* only handle a few types, since others can't possibly be replies to |
399 | * us in a sane network (if it is anyway, it will be counted as lost | 403 | * us in a sane network (if it is anyway, it will be counted as lost |
@@ -413,8 +417,9 @@ static int handle_random_icmp(unsigned char *packet, struct sockaddr_storage *ad | |||
413 | * to RFC 792). If it isn't, just ignore it */ | 417 | * to RFC 792). If it isn't, just ignore it */ |
414 | memcpy(&sent_icmp, packet + 28, sizeof(sent_icmp)); | 418 | memcpy(&sent_icmp, packet + 28, sizeof(sent_icmp)); |
415 | if (sent_icmp.icmp_type != ICMP_ECHO || ntohs(sent_icmp.icmp_id) != pid || ntohs(sent_icmp.icmp_seq) >= targets * packets) { | 419 | if (sent_icmp.icmp_type != ICMP_ECHO || ntohs(sent_icmp.icmp_id) != pid || ntohs(sent_icmp.icmp_seq) >= targets * packets) { |
416 | if (debug) | 420 | if (debug) { |
417 | printf("Packet is no response to a packet we sent\n"); | 421 | printf("Packet is no response to a packet we sent\n"); |
422 | } | ||
418 | return 0; | 423 | return 0; |
419 | } | 424 | } |
420 | 425 | ||
@@ -429,8 +434,9 @@ static int handle_random_icmp(unsigned char *packet, struct sockaddr_storage *ad | |||
429 | icmp_lost++; | 434 | icmp_lost++; |
430 | host->icmp_lost++; | 435 | host->icmp_lost++; |
431 | /* don't spend time on lost hosts any more */ | 436 | /* don't spend time on lost hosts any more */ |
432 | if (host->flags & FLAG_LOST_CAUSE) | 437 | if (host->flags & FLAG_LOST_CAUSE) { |
433 | return 0; | 438 | return 0; |
439 | } | ||
434 | 440 | ||
435 | /* source quench means we're sending too fast, so increase the | 441 | /* source quench means we're sending too fast, so increase the |
436 | * interval and mark this packet lost */ | 442 | * interval and mark this packet lost */ |
@@ -488,10 +494,11 @@ int main(int argc, char **argv) { | |||
488 | /* get calling name the old-fashioned way for portability instead | 494 | /* get calling name the old-fashioned way for portability instead |
489 | * of relying on the glibc-ism __progname */ | 495 | * of relying on the glibc-ism __progname */ |
490 | ptr = strrchr(argv[0], '/'); | 496 | ptr = strrchr(argv[0], '/'); |
491 | if (ptr) | 497 | if (ptr) { |
492 | progname = &ptr[1]; | 498 | progname = &ptr[1]; |
493 | else | 499 | } else { |
494 | progname = argv[0]; | 500 | progname = argv[0]; |
501 | } | ||
495 | 502 | ||
496 | /* now set defaults. Use progname to set them initially (allows for | 503 | /* now set defaults. Use progname to set them initially (allows for |
497 | * superfast check_host program when target host is up */ | 504 | * superfast check_host program when target host is up */ |
@@ -534,10 +541,12 @@ int main(int argc, char **argv) { | |||
534 | 541 | ||
535 | /* support "--help" and "--version" */ | 542 | /* support "--help" and "--version" */ |
536 | if (argc == 2) { | 543 | if (argc == 2) { |
537 | if (!strcmp(argv[1], "--help")) | 544 | if (!strcmp(argv[1], "--help")) { |
538 | strcpy(argv[1], "-h"); | 545 | strcpy(argv[1], "-h"); |
539 | if (!strcmp(argv[1], "--version")) | 546 | } |
547 | if (!strcmp(argv[1], "--version")) { | ||
540 | strcpy(argv[1], "-V"); | 548 | strcpy(argv[1], "-V"); |
549 | } | ||
541 | } | 550 | } |
542 | 551 | ||
543 | /* Parse protocol arguments first */ | 552 | /* Parse protocol arguments first */ |
@@ -545,14 +554,16 @@ int main(int argc, char **argv) { | |||
545 | while ((arg = getopt(argc, argv, opts_str)) != EOF) { | 554 | while ((arg = getopt(argc, argv, opts_str)) != EOF) { |
546 | switch (arg) { | 555 | switch (arg) { |
547 | case '4': | 556 | case '4': |
548 | if (address_family != -1) | 557 | if (address_family != -1) { |
549 | crash("Multiple protocol versions not supported"); | 558 | crash("Multiple protocol versions not supported"); |
559 | } | ||
550 | address_family = AF_INET; | 560 | address_family = AF_INET; |
551 | break; | 561 | break; |
552 | case '6': | 562 | case '6': |
553 | #ifdef USE_IPV6 | 563 | #ifdef USE_IPV6 |
554 | if (address_family != -1) | 564 | if (address_family != -1) { |
555 | crash("Multiple protocol versions not supported"); | 565 | crash("Multiple protocol versions not supported"); |
566 | } | ||
556 | address_family = AF_INET6; | 567 | address_family = AF_INET6; |
557 | #else | 568 | #else |
558 | usage(_("IPv6 support not available\n")); | 569 | usage(_("IPv6 support not available\n")); |
@@ -579,9 +590,10 @@ int main(int argc, char **argv) { | |||
579 | if (size >= (sizeof(struct icmp) + sizeof(struct icmp_ping_data)) && size < MAX_PING_DATA) { | 590 | if (size >= (sizeof(struct icmp) + sizeof(struct icmp_ping_data)) && size < MAX_PING_DATA) { |
580 | icmp_data_size = size; | 591 | icmp_data_size = size; |
581 | icmp_pkt_size = size + ICMP_MINLEN; | 592 | icmp_pkt_size = size + ICMP_MINLEN; |
582 | } else | 593 | } else { |
583 | usage_va("ICMP data length must be between: %lu and %lu", sizeof(struct icmp) + sizeof(struct icmp_ping_data), | 594 | usage_va("ICMP data length must be between: %lu and %lu", sizeof(struct icmp) + sizeof(struct icmp_ping_data), |
584 | MAX_PING_DATA - 1); | 595 | MAX_PING_DATA - 1); |
596 | } | ||
585 | break; | 597 | break; |
586 | case 'i': | 598 | case 'i': |
587 | pkt_interval = get_timevar(optarg); | 599 | pkt_interval = get_timevar(optarg); |
@@ -601,8 +613,9 @@ int main(int argc, char **argv) { | |||
601 | break; | 613 | break; |
602 | case 't': | 614 | case 't': |
603 | timeout = strtoul(optarg, NULL, 0); | 615 | timeout = strtoul(optarg, NULL, 0); |
604 | if (!timeout) | 616 | if (!timeout) { |
605 | timeout = 10; | 617 | timeout = 10; |
618 | } | ||
606 | break; | 619 | break; |
607 | case 'H': | 620 | case 'H': |
608 | add_target(optarg); | 621 | add_target(optarg); |
@@ -709,18 +722,22 @@ int main(int argc, char **argv) { | |||
709 | default: | 722 | default: |
710 | crash("Address family not supported"); | 723 | crash("Address family not supported"); |
711 | } | 724 | } |
712 | if ((icmp_sock = socket(address_family, SOCK_RAW, icmp_proto)) != -1) | 725 | if ((icmp_sock = socket(address_family, SOCK_RAW, icmp_proto)) != -1) { |
713 | sockets |= HAVE_ICMP; | 726 | sockets |= HAVE_ICMP; |
714 | else | 727 | } else { |
715 | icmp_sockerrno = errno; | 728 | icmp_sockerrno = errno; |
729 | } | ||
716 | 730 | ||
717 | if (source_ip) | 731 | if (source_ip) { |
718 | set_source_ip(source_ip); | 732 | set_source_ip(source_ip); |
733 | } | ||
719 | 734 | ||
720 | #ifdef SO_TIMESTAMP | 735 | #ifdef SO_TIMESTAMP |
721 | if (setsockopt(icmp_sock, SOL_SOCKET, SO_TIMESTAMP, &on, sizeof(on))) | 736 | if (setsockopt(icmp_sock, SOL_SOCKET, SO_TIMESTAMP, &on, sizeof(on))) { |
722 | if (debug) | 737 | if (debug) { |
723 | printf("Warning: no SO_TIMESTAMP support\n"); | 738 | printf("Warning: no SO_TIMESTAMP support\n"); |
739 | } | ||
740 | } | ||
724 | #endif // SO_TIMESTAMP | 741 | #endif // SO_TIMESTAMP |
725 | 742 | ||
726 | /* now drop privileges (no effect if not setsuid or geteuid() == 0) */ | 743 | /* now drop privileges (no effect if not setsuid or geteuid() == 0) */ |
@@ -746,16 +763,18 @@ int main(int argc, char **argv) { | |||
746 | /* return -1; */ | 763 | /* return -1; */ |
747 | /* } */ | 764 | /* } */ |
748 | } | 765 | } |
749 | if (!ttl) | 766 | if (!ttl) { |
750 | ttl = 64; | 767 | ttl = 64; |
768 | } | ||
751 | 769 | ||
752 | if (icmp_sock) { | 770 | if (icmp_sock) { |
753 | result = setsockopt(icmp_sock, SOL_IP, IP_TTL, &ttl, sizeof(ttl)); | 771 | result = setsockopt(icmp_sock, SOL_IP, IP_TTL, &ttl, sizeof(ttl)); |
754 | if (debug) { | 772 | if (debug) { |
755 | if (result == -1) | 773 | if (result == -1) { |
756 | printf("setsockopt failed\n"); | 774 | printf("setsockopt failed\n"); |
757 | else | 775 | } else { |
758 | printf("ttl set to %u\n", ttl); | 776 | printf("ttl set to %u\n", ttl); |
777 | } | ||
759 | } | 778 | } |
760 | } | 779 | } |
761 | 780 | ||
@@ -763,18 +782,24 @@ int main(int argc, char **argv) { | |||
763 | * (nothing will break if they do), but some anal plugin maintainer | 782 | * (nothing will break if they do), but some anal plugin maintainer |
764 | * will probably add some printf() thing here later, so it might be | 783 | * will probably add some printf() thing here later, so it might be |
765 | * best to at least show them where to do it. ;) */ | 784 | * best to at least show them where to do it. ;) */ |
766 | if (warn.pl > crit.pl) | 785 | if (warn.pl > crit.pl) { |
767 | warn.pl = crit.pl; | 786 | warn.pl = crit.pl; |
768 | if (warn.rta > crit.rta) | 787 | } |
788 | if (warn.rta > crit.rta) { | ||
769 | warn.rta = crit.rta; | 789 | warn.rta = crit.rta; |
770 | if (warn_down > crit_down) | 790 | } |
791 | if (warn_down > crit_down) { | ||
771 | crit_down = warn_down; | 792 | crit_down = warn_down; |
772 | if (warn.jitter > crit.jitter) | 793 | } |
794 | if (warn.jitter > crit.jitter) { | ||
773 | crit.jitter = warn.jitter; | 795 | crit.jitter = warn.jitter; |
774 | if (warn.mos < crit.mos) | 796 | } |
797 | if (warn.mos < crit.mos) { | ||
775 | warn.mos = crit.mos; | 798 | warn.mos = crit.mos; |
776 | if (warn.score < crit.score) | 799 | } |
800 | if (warn.score < crit.score) { | ||
777 | warn.score = crit.score; | 801 | warn.score = crit.score; |
802 | } | ||
778 | 803 | ||
779 | #ifdef HAVE_SIGACTION | 804 | #ifdef HAVE_SIGACTION |
780 | sig_action.sa_sigaction = NULL; | 805 | sig_action.sa_sigaction = NULL; |
@@ -791,8 +816,9 @@ int main(int argc, char **argv) { | |||
791 | signal(SIGTERM, finish); | 816 | signal(SIGTERM, finish); |
792 | signal(SIGALRM, finish); | 817 | signal(SIGALRM, finish); |
793 | #endif /* HAVE_SIGACTION */ | 818 | #endif /* HAVE_SIGACTION */ |
794 | if (debug) | 819 | if (debug) { |
795 | printf("Setting alarm timeout to %u seconds\n", timeout); | 820 | printf("Setting alarm timeout to %u seconds\n", timeout); |
821 | } | ||
796 | alarm(timeout); | 822 | alarm(timeout); |
797 | 823 | ||
798 | /* make sure we don't wait any longer than necessary */ | 824 | /* make sure we don't wait any longer than necessary */ |
@@ -863,11 +889,13 @@ static void run_checks(void) { | |||
863 | for (i = 0; i < packets; i++) { | 889 | for (i = 0; i < packets; i++) { |
864 | for (t = 0; t < targets; t++) { | 890 | for (t = 0; t < targets; t++) { |
865 | /* don't send useless packets */ | 891 | /* don't send useless packets */ |
866 | if (!targets_alive) | 892 | if (!targets_alive) { |
867 | finish(0); | 893 | finish(0); |
894 | } | ||
868 | if (table[t]->flags & FLAG_LOST_CAUSE) { | 895 | if (table[t]->flags & FLAG_LOST_CAUSE) { |
869 | if (debug) | 896 | if (debug) { |
870 | printf("%s is a lost cause. not sending any more\n", table[t]->name); | 897 | printf("%s is a lost cause. not sending any more\n", table[t]->name); |
898 | } | ||
871 | continue; | 899 | continue; |
872 | } | 900 | } |
873 | 901 | ||
@@ -886,15 +914,17 @@ static void run_checks(void) { | |||
886 | printf("time_passed: %u final_wait: %u max_completion_time: %llu\n", time_passed, final_wait, max_completion_time); | 914 | printf("time_passed: %u final_wait: %u max_completion_time: %llu\n", time_passed, final_wait, max_completion_time); |
887 | } | 915 | } |
888 | if (time_passed > max_completion_time) { | 916 | if (time_passed > max_completion_time) { |
889 | if (debug) | 917 | if (debug) { |
890 | printf("Time passed. Finishing up\n"); | 918 | printf("Time passed. Finishing up\n"); |
919 | } | ||
891 | finish(0); | 920 | finish(0); |
892 | } | 921 | } |
893 | 922 | ||
894 | /* catch the packets that might come in within the timeframe, but | 923 | /* catch the packets that might come in within the timeframe, but |
895 | * haven't yet */ | 924 | * haven't yet */ |
896 | if (debug) | 925 | if (debug) { |
897 | printf("Waiting for %u micro-seconds (%0.3f msecs)\n", final_wait, (float)final_wait / 1000); | 926 | printf("Waiting for %u micro-seconds (%0.3f msecs)\n", final_wait, (float)final_wait / 1000); |
927 | } | ||
898 | wait_for_reply(icmp_sock, final_wait); | 928 | wait_for_reply(icmp_sock, final_wait); |
899 | } | 929 | } |
900 | } | 930 | } |
@@ -955,8 +985,9 @@ static int wait_for_reply(int sock, u_int t) { | |||
955 | continue; /* timeout for this one, so keep trying */ | 985 | continue; /* timeout for this one, so keep trying */ |
956 | } | 986 | } |
957 | if (n < 0) { | 987 | if (n < 0) { |
958 | if (debug) | 988 | if (debug) { |
959 | printf("recvfrom_wto() returned errors\n"); | 989 | printf("recvfrom_wto() returned errors\n"); |
990 | } | ||
960 | free(packet.buf); | 991 | free(packet.buf); |
961 | return n; | 992 | return n; |
962 | } | 993 | } |
@@ -1003,8 +1034,9 @@ static int wait_for_reply(int sock, u_int t) { | |||
1003 | ntohs(packet.icp->icmp_seq) >= targets * packets)) || | 1034 | ntohs(packet.icp->icmp_seq) >= targets * packets)) || |
1004 | (address_family == PF_INET6 && (ntohs(packet.icp6->icmp6_id) != pid || packet.icp6->icmp6_type != ICMP6_ECHO_REPLY || | 1035 | (address_family == PF_INET6 && (ntohs(packet.icp6->icmp6_id) != pid || packet.icp6->icmp6_type != ICMP6_ECHO_REPLY || |
1005 | ntohs(packet.icp6->icmp6_seq) >= targets * packets))) { | 1036 | ntohs(packet.icp6->icmp6_seq) >= targets * packets))) { |
1006 | if (debug > 2) | 1037 | if (debug > 2) { |
1007 | printf("not a proper ICMP_ECHOREPLY\n"); | 1038 | printf("not a proper ICMP_ECHOREPLY\n"); |
1039 | } | ||
1008 | handle_random_icmp(buf + hlen, &resp_addr); | 1040 | handle_random_icmp(buf + hlen, &resp_addr); |
1009 | continue; | 1041 | continue; |
1010 | } | 1042 | } |
@@ -1012,15 +1044,17 @@ static int wait_for_reply(int sock, u_int t) { | |||
1012 | /* this is indeed a valid response */ | 1044 | /* this is indeed a valid response */ |
1013 | if (address_family == PF_INET) { | 1045 | if (address_family == PF_INET) { |
1014 | memcpy(&data, packet.icp->icmp_data, sizeof(data)); | 1046 | memcpy(&data, packet.icp->icmp_data, sizeof(data)); |
1015 | if (debug > 2) | 1047 | if (debug > 2) { |
1016 | printf("ICMP echo-reply of len %lu, id %u, seq %u, cksum 0x%X\n", (unsigned long)sizeof(data), ntohs(packet.icp->icmp_id), | 1048 | printf("ICMP echo-reply of len %lu, id %u, seq %u, cksum 0x%X\n", (unsigned long)sizeof(data), ntohs(packet.icp->icmp_id), |
1017 | ntohs(packet.icp->icmp_seq), packet.icp->icmp_cksum); | 1049 | ntohs(packet.icp->icmp_seq), packet.icp->icmp_cksum); |
1050 | } | ||
1018 | host = table[ntohs(packet.icp->icmp_seq) / packets]; | 1051 | host = table[ntohs(packet.icp->icmp_seq) / packets]; |
1019 | } else { | 1052 | } else { |
1020 | memcpy(&data, &packet.icp6->icmp6_dataun.icmp6_un_data8[4], sizeof(data)); | 1053 | memcpy(&data, &packet.icp6->icmp6_dataun.icmp6_un_data8[4], sizeof(data)); |
1021 | if (debug > 2) | 1054 | if (debug > 2) { |
1022 | printf("ICMP echo-reply of len %lu, id %u, seq %u, cksum 0x%X\n", (unsigned long)sizeof(data), ntohs(packet.icp6->icmp6_id), | 1055 | printf("ICMP echo-reply of len %lu, id %u, seq %u, cksum 0x%X\n", (unsigned long)sizeof(data), ntohs(packet.icp6->icmp6_id), |
1023 | ntohs(packet.icp6->icmp6_seq), packet.icp6->icmp6_cksum); | 1056 | ntohs(packet.icp6->icmp6_seq), packet.icp6->icmp6_cksum); |
1057 | } | ||
1024 | host = table[ntohs(packet.icp6->icmp6_seq) / packets]; | 1058 | host = table[ntohs(packet.icp6->icmp6_seq) / packets]; |
1025 | } | 1059 | } |
1026 | 1060 | ||
@@ -1051,8 +1085,9 @@ static int wait_for_reply(int sock, u_int t) { | |||
1051 | } | 1085 | } |
1052 | 1086 | ||
1053 | /* Check if packets in order */ | 1087 | /* Check if packets in order */ |
1054 | if (host->last_icmp_seq >= packet.icp->icmp_seq) | 1088 | if (host->last_icmp_seq >= packet.icp->icmp_seq) { |
1055 | host->order_status = STATE_CRITICAL; | 1089 | host->order_status = STATE_CRITICAL; |
1090 | } | ||
1056 | } | 1091 | } |
1057 | host->last_tdiff = tdiff; | 1092 | host->last_tdiff = tdiff; |
1058 | 1093 | ||
@@ -1061,10 +1096,14 @@ static int wait_for_reply(int sock, u_int t) { | |||
1061 | host->time_waited += tdiff; | 1096 | host->time_waited += tdiff; |
1062 | host->icmp_recv++; | 1097 | host->icmp_recv++; |
1063 | icmp_recv++; | 1098 | icmp_recv++; |
1064 | if (tdiff > (unsigned int)host->rtmax) | 1099 | |
1100 | if (tdiff > (unsigned int)host->rtmax) { | ||
1065 | host->rtmax = tdiff; | 1101 | host->rtmax = tdiff; |
1066 | if (tdiff < (unsigned int)host->rtmin) | 1102 | } |
1103 | |||
1104 | if ((host->rtmin == INFINITY) || (tdiff < (unsigned int)host->rtmin)) { | ||
1067 | host->rtmin = tdiff; | 1105 | host->rtmin = tdiff; |
1106 | } | ||
1068 | 1107 | ||
1069 | if (debug) { | 1108 | if (debug) { |
1070 | char address[INET6_ADDRSTRLEN]; | 1109 | char address[INET6_ADDRSTRLEN]; |
@@ -1142,9 +1181,10 @@ static int send_icmp_ping(int sock, struct rta_host *host) { | |||
1142 | icp->icmp_seq = htons(host->id++); | 1181 | icp->icmp_seq = htons(host->id++); |
1143 | icp->icmp_cksum = icmp_checksum((uint16_t *)buf, (size_t)icmp_pkt_size); | 1182 | icp->icmp_cksum = icmp_checksum((uint16_t *)buf, (size_t)icmp_pkt_size); |
1144 | 1183 | ||
1145 | if (debug > 2) | 1184 | if (debug > 2) { |
1146 | printf("Sending ICMP echo-request of len %lu, id %u, seq %u, cksum 0x%X to host %s\n", (unsigned long)sizeof(data), | 1185 | printf("Sending ICMP echo-request of len %lu, id %u, seq %u, cksum 0x%X to host %s\n", (unsigned long)sizeof(data), |
1147 | ntohs(icp->icmp_id), ntohs(icp->icmp_seq), icp->icmp_cksum, host->name); | 1186 | ntohs(icp->icmp_id), ntohs(icp->icmp_seq), icp->icmp_cksum, host->name); |
1187 | } | ||
1148 | } else { | 1188 | } else { |
1149 | struct icmp6_hdr *icp6 = (struct icmp6_hdr *)buf; | 1189 | struct icmp6_hdr *icp6 = (struct icmp6_hdr *)buf; |
1150 | addrlen = sizeof(struct sockaddr_in6); | 1190 | addrlen = sizeof(struct sockaddr_in6); |
@@ -1216,8 +1256,9 @@ static int recvfrom_wto(int sock, void *buf, unsigned int len, struct sockaddr * | |||
1216 | #endif | 1256 | #endif |
1217 | 1257 | ||
1218 | if (!*timo) { | 1258 | if (!*timo) { |
1219 | if (debug) | 1259 | if (debug) { |
1220 | printf("*timo is not\n"); | 1260 | printf("*timo is not\n"); |
1261 | } | ||
1221 | return 0; | 1262 | return 0; |
1222 | } | 1263 | } |
1223 | 1264 | ||
@@ -1230,13 +1271,15 @@ static int recvfrom_wto(int sock, void *buf, unsigned int len, struct sockaddr * | |||
1230 | errno = 0; | 1271 | errno = 0; |
1231 | gettimeofday(&then, &tz); | 1272 | gettimeofday(&then, &tz); |
1232 | n = select(sock + 1, &rd, &wr, NULL, &to); | 1273 | n = select(sock + 1, &rd, &wr, NULL, &to); |
1233 | if (n < 0) | 1274 | if (n < 0) { |
1234 | crash("select() in recvfrom_wto"); | 1275 | crash("select() in recvfrom_wto"); |
1276 | } | ||
1235 | gettimeofday(&now, &tz); | 1277 | gettimeofday(&now, &tz); |
1236 | *timo = get_timevaldiff(&then, &now); | 1278 | *timo = get_timevaldiff(&then, &now); |
1237 | 1279 | ||
1238 | if (!n) | 1280 | if (!n) { |
1239 | return 0; /* timeout */ | 1281 | return 0; /* timeout */ |
1282 | } | ||
1240 | 1283 | ||
1241 | slen = sizeof(struct sockaddr_storage); | 1284 | slen = sizeof(struct sockaddr_storage); |
1242 | 1285 | ||
@@ -1281,15 +1324,19 @@ static void finish(int sig) { | |||
1281 | double R; | 1324 | double R; |
1282 | 1325 | ||
1283 | alarm(0); | 1326 | alarm(0); |
1284 | if (debug > 1) | 1327 | if (debug > 1) { |
1285 | printf("finish(%d) called\n", sig); | 1328 | printf("finish(%d) called\n", sig); |
1329 | } | ||
1286 | 1330 | ||
1287 | if (icmp_sock != -1) | 1331 | if (icmp_sock != -1) { |
1288 | close(icmp_sock); | 1332 | close(icmp_sock); |
1289 | if (udp_sock != -1) | 1333 | } |
1334 | if (udp_sock != -1) { | ||
1290 | close(udp_sock); | 1335 | close(udp_sock); |
1291 | if (tcp_sock != -1) | 1336 | } |
1337 | if (tcp_sock != -1) { | ||
1292 | close(tcp_sock); | 1338 | close(tcp_sock); |
1339 | } | ||
1293 | 1340 | ||
1294 | if (debug) { | 1341 | if (debug) { |
1295 | printf("icmp_sent: %u icmp_recv: %u icmp_lost: %u\n", icmp_sent, icmp_recv, icmp_lost); | 1342 | printf("icmp_sent: %u icmp_recv: %u icmp_lost: %u\n", icmp_sent, icmp_recv, icmp_lost); |
@@ -1310,8 +1357,9 @@ static void finish(int sig) { | |||
1310 | rta = 0; | 1357 | rta = 0; |
1311 | status = STATE_CRITICAL; | 1358 | status = STATE_CRITICAL; |
1312 | /* up the down counter if not already counted */ | 1359 | /* up the down counter if not already counted */ |
1313 | if (!(host->flags & FLAG_LOST_CAUSE) && targets_alive) | 1360 | if (!(host->flags & FLAG_LOST_CAUSE) && targets_alive) { |
1314 | targets_down++; | 1361 | targets_down++; |
1362 | } | ||
1315 | } else { | 1363 | } else { |
1316 | pl = ((host->icmp_sent - host->icmp_recv) * 100) / host->icmp_sent; | 1364 | pl = ((host->icmp_sent - host->icmp_recv) * 100) / host->icmp_sent; |
1317 | rta = (double)host->time_waited / host->icmp_recv; | 1365 | rta = (double)host->time_waited / host->icmp_recv; |
@@ -1444,32 +1492,39 @@ static void finish(int sig) { | |||
1444 | } | 1492 | } |
1445 | 1493 | ||
1446 | /* this is inevitable */ | 1494 | /* this is inevitable */ |
1447 | if (!targets_alive) | 1495 | if (!targets_alive) { |
1448 | status = STATE_CRITICAL; | 1496 | status = STATE_CRITICAL; |
1497 | } | ||
1449 | if (min_hosts_alive > -1) { | 1498 | if (min_hosts_alive > -1) { |
1450 | if (hosts_ok >= min_hosts_alive) | 1499 | if (hosts_ok >= min_hosts_alive) { |
1451 | status = STATE_OK; | 1500 | status = STATE_OK; |
1452 | else if ((hosts_ok + hosts_warn) >= min_hosts_alive) | 1501 | } else if ((hosts_ok + hosts_warn) >= min_hosts_alive) { |
1453 | status = STATE_WARNING; | 1502 | status = STATE_WARNING; |
1503 | } | ||
1454 | } | 1504 | } |
1455 | printf("%s - ", status_string[status]); | 1505 | printf("%s - ", status_string[status]); |
1456 | 1506 | ||
1457 | host = list; | 1507 | host = list; |
1458 | while (host) { | 1508 | while (host) { |
1459 | 1509 | if (debug) { | |
1460 | if (debug) | ||
1461 | puts(""); | 1510 | puts(""); |
1511 | } | ||
1512 | |||
1462 | if (i) { | 1513 | if (i) { |
1463 | if (i < targets) | 1514 | if (i < targets) { |
1464 | printf(" :: "); | 1515 | printf(" :: "); |
1465 | else | 1516 | } else { |
1466 | printf("\n"); | 1517 | printf("\n"); |
1518 | } | ||
1467 | } | 1519 | } |
1520 | |||
1468 | i++; | 1521 | i++; |
1522 | |||
1469 | if (!host->icmp_recv) { | 1523 | if (!host->icmp_recv) { |
1470 | status = STATE_CRITICAL; | 1524 | status = STATE_CRITICAL; |
1471 | host->rtmin = 0; | 1525 | host->rtmin = 0; |
1472 | host->jitter_min = 0; | 1526 | host->jitter_min = 0; |
1527 | |||
1473 | if (host->flags & FLAG_LOST_CAUSE) { | 1528 | if (host->flags & FLAG_LOST_CAUSE) { |
1474 | char address[INET6_ADDRSTRLEN]; | 1529 | char address[INET6_ADDRSTRLEN]; |
1475 | parse_address(&host->error_addr, address, sizeof(address)); | 1530 | parse_address(&host->error_addr, address, sizeof(address)); |
@@ -1481,55 +1536,66 @@ static void finish(int sig) { | |||
1481 | printf("%s", host->name); | 1536 | printf("%s", host->name); |
1482 | /* rta text output */ | 1537 | /* rta text output */ |
1483 | if (rta_mode) { | 1538 | if (rta_mode) { |
1484 | if (status == STATE_OK) | 1539 | if (status == STATE_OK) { |
1485 | printf(" rta %0.3fms", host->rta / 1000); | 1540 | printf(" rta %0.3fms", host->rta / 1000); |
1486 | else if (status == STATE_WARNING && host->rta_status == status) | 1541 | } else if (status == STATE_WARNING && host->rta_status == status) { |
1487 | printf(" rta %0.3fms > %0.3fms", (float)host->rta / 1000, (float)warn.rta / 1000); | 1542 | printf(" rta %0.3fms > %0.3fms", (float)host->rta / 1000, (float)warn.rta / 1000); |
1488 | else if (status == STATE_CRITICAL && host->rta_status == status) | 1543 | } else if (status == STATE_CRITICAL && host->rta_status == status) { |
1489 | printf(" rta %0.3fms > %0.3fms", (float)host->rta / 1000, (float)crit.rta / 1000); | 1544 | printf(" rta %0.3fms > %0.3fms", (float)host->rta / 1000, (float)crit.rta / 1000); |
1545 | } | ||
1490 | } | 1546 | } |
1547 | |||
1491 | /* pl text output */ | 1548 | /* pl text output */ |
1492 | if (pl_mode) { | 1549 | if (pl_mode) { |
1493 | if (status == STATE_OK) | 1550 | if (status == STATE_OK) { |
1494 | printf(" lost %u%%", host->pl); | 1551 | printf(" lost %u%%", host->pl); |
1495 | else if (status == STATE_WARNING && host->pl_status == status) | 1552 | } else if (status == STATE_WARNING && host->pl_status == status) { |
1496 | printf(" lost %u%% > %u%%", host->pl, warn.pl); | 1553 | printf(" lost %u%% > %u%%", host->pl, warn.pl); |
1497 | else if (status == STATE_CRITICAL && host->pl_status == status) | 1554 | } else if (status == STATE_CRITICAL && host->pl_status == status) { |
1498 | printf(" lost %u%% > %u%%", host->pl, crit.pl); | 1555 | printf(" lost %u%% > %u%%", host->pl, crit.pl); |
1556 | } | ||
1499 | } | 1557 | } |
1558 | |||
1500 | /* jitter text output */ | 1559 | /* jitter text output */ |
1501 | if (jitter_mode) { | 1560 | if (jitter_mode) { |
1502 | if (status == STATE_OK) | 1561 | if (status == STATE_OK) { |
1503 | printf(" jitter %0.3fms", (float)host->jitter); | 1562 | printf(" jitter %0.3fms", (float)host->jitter); |
1504 | else if (status == STATE_WARNING && host->jitter_status == status) | 1563 | } else if (status == STATE_WARNING && host->jitter_status == status) { |
1505 | printf(" jitter %0.3fms > %0.3fms", (float)host->jitter, warn.jitter); | 1564 | printf(" jitter %0.3fms > %0.3fms", (float)host->jitter, warn.jitter); |
1506 | else if (status == STATE_CRITICAL && host->jitter_status == status) | 1565 | } else if (status == STATE_CRITICAL && host->jitter_status == status) { |
1507 | printf(" jitter %0.3fms > %0.3fms", (float)host->jitter, crit.jitter); | 1566 | printf(" jitter %0.3fms > %0.3fms", (float)host->jitter, crit.jitter); |
1567 | } | ||
1508 | } | 1568 | } |
1569 | |||
1509 | /* mos text output */ | 1570 | /* mos text output */ |
1510 | if (mos_mode) { | 1571 | if (mos_mode) { |
1511 | if (status == STATE_OK) | 1572 | if (status == STATE_OK) { |
1512 | printf(" MOS %0.1f", (float)host->mos); | 1573 | printf(" MOS %0.1f", (float)host->mos); |
1513 | else if (status == STATE_WARNING && host->mos_status == status) | 1574 | } else if (status == STATE_WARNING && host->mos_status == status) { |
1514 | printf(" MOS %0.1f < %0.1f", (float)host->mos, (float)warn.mos); | 1575 | printf(" MOS %0.1f < %0.1f", (float)host->mos, (float)warn.mos); |
1515 | else if (status == STATE_CRITICAL && host->mos_status == status) | 1576 | } else if (status == STATE_CRITICAL && host->mos_status == status) { |
1516 | printf(" MOS %0.1f < %0.1f", (float)host->mos, (float)crit.mos); | 1577 | printf(" MOS %0.1f < %0.1f", (float)host->mos, (float)crit.mos); |
1578 | } | ||
1517 | } | 1579 | } |
1580 | |||
1518 | /* score text output */ | 1581 | /* score text output */ |
1519 | if (score_mode) { | 1582 | if (score_mode) { |
1520 | if (status == STATE_OK) | 1583 | if (status == STATE_OK) { |
1521 | printf(" Score %u", (int)host->score); | 1584 | printf(" Score %u", (int)host->score); |
1522 | else if (status == STATE_WARNING && host->score_status == status) | 1585 | } else if (status == STATE_WARNING && host->score_status == status) { |
1523 | printf(" Score %u < %u", (int)host->score, (int)warn.score); | 1586 | printf(" Score %u < %u", (int)host->score, (int)warn.score); |
1524 | else if (status == STATE_CRITICAL && host->score_status == status) | 1587 | } else if (status == STATE_CRITICAL && host->score_status == status) { |
1525 | printf(" Score %u < %u", (int)host->score, (int)crit.score); | 1588 | printf(" Score %u < %u", (int)host->score, (int)crit.score); |
1589 | } | ||
1526 | } | 1590 | } |
1591 | |||
1527 | /* order statis text output */ | 1592 | /* order statis text output */ |
1528 | if (order_mode) { | 1593 | if (order_mode) { |
1529 | if (status == STATE_OK) | 1594 | if (status == STATE_OK) { |
1530 | printf(" Packets in order"); | 1595 | printf(" Packets in order"); |
1531 | else if (status == STATE_CRITICAL && host->order_status == status) | 1596 | } else if (status == STATE_CRITICAL && host->order_status == status) { |
1532 | printf(" Packets out of order"); | 1597 | printf(" Packets out of order"); |
1598 | } | ||
1533 | } | 1599 | } |
1534 | } | 1600 | } |
1535 | host = host->next; | 1601 | host = host->next; |
@@ -1542,8 +1608,9 @@ static void finish(int sig) { | |||
1542 | i = 0; | 1608 | i = 0; |
1543 | host = list; | 1609 | host = list; |
1544 | while (host) { | 1610 | while (host) { |
1545 | if (debug) | 1611 | if (debug) { |
1546 | puts(""); | 1612 | puts(""); |
1613 | } | ||
1547 | 1614 | ||
1548 | if (rta_mode) { | 1615 | if (rta_mode) { |
1549 | if (host->pl < 100) { | 1616 | if (host->pl < 100) { |
@@ -1593,17 +1660,19 @@ static void finish(int sig) { | |||
1593 | } | 1660 | } |
1594 | 1661 | ||
1595 | if (min_hosts_alive > -1) { | 1662 | if (min_hosts_alive > -1) { |
1596 | if (hosts_ok >= min_hosts_alive) | 1663 | if (hosts_ok >= min_hosts_alive) { |
1597 | status = STATE_OK; | 1664 | status = STATE_OK; |
1598 | else if ((hosts_ok + hosts_warn) >= min_hosts_alive) | 1665 | } else if ((hosts_ok + hosts_warn) >= min_hosts_alive) { |
1599 | status = STATE_WARNING; | 1666 | status = STATE_WARNING; |
1667 | } | ||
1600 | } | 1668 | } |
1601 | 1669 | ||
1602 | /* finish with an empty line */ | 1670 | /* finish with an empty line */ |
1603 | puts(""); | 1671 | puts(""); |
1604 | if (debug) | 1672 | if (debug) { |
1605 | printf("targets: %u, targets_alive: %u, hosts_ok: %u, hosts_warn: %u, min_hosts_alive: %i\n", targets, targets_alive, hosts_ok, | 1673 | printf("targets: %u, targets_alive: %u, hosts_ok: %u, hosts_warn: %u, min_hosts_alive: %i\n", targets, targets_alive, hosts_ok, |
1606 | hosts_warn, min_hosts_alive); | 1674 | hosts_warn, min_hosts_alive); |
1675 | } | ||
1607 | 1676 | ||
1608 | exit(status); | 1677 | exit(status); |
1609 | } | 1678 | } |
@@ -1616,8 +1685,9 @@ static u_int get_timevaldiff(struct timeval *early, struct timeval *later) { | |||
1616 | gettimeofday(&now, &tz); | 1685 | gettimeofday(&now, &tz); |
1617 | later = &now; | 1686 | later = &now; |
1618 | } | 1687 | } |
1619 | if (!early) | 1688 | if (!early) { |
1620 | early = &prog_start; | 1689 | early = &prog_start; |
1690 | } | ||
1621 | 1691 | ||
1622 | /* if early > later we return 0 so as to indicate a timeout */ | 1692 | /* if early > later we return 0 so as to indicate a timeout */ |
1623 | if (early->tv_sec > later->tv_sec || (early->tv_sec == later->tv_sec && early->tv_usec > later->tv_usec)) { | 1693 | if (early->tv_sec > later->tv_sec || (early->tv_sec == later->tv_sec && early->tv_usec > later->tv_usec)) { |
@@ -1634,10 +1704,11 @@ static int add_target_ip(char *arg, struct sockaddr_storage *in) { | |||
1634 | struct sockaddr_in *sin, *host_sin; | 1704 | struct sockaddr_in *sin, *host_sin; |
1635 | struct sockaddr_in6 *sin6, *host_sin6; | 1705 | struct sockaddr_in6 *sin6, *host_sin6; |
1636 | 1706 | ||
1637 | if (address_family == AF_INET) | 1707 | if (address_family == AF_INET) { |
1638 | sin = (struct sockaddr_in *)in; | 1708 | sin = (struct sockaddr_in *)in; |
1639 | else | 1709 | } else { |
1640 | sin6 = (struct sockaddr_in6 *)in; | 1710 | sin6 = (struct sockaddr_in6 *)in; |
1711 | } | ||
1641 | 1712 | ||
1642 | /* disregard obviously stupid addresses | 1713 | /* disregard obviously stupid addresses |
1643 | * (I didn't find an ipv6 equivalent to INADDR_NONE) */ | 1714 | * (I didn't find an ipv6 equivalent to INADDR_NONE) */ |
@@ -1654,8 +1725,9 @@ static int add_target_ip(char *arg, struct sockaddr_storage *in) { | |||
1654 | 1725 | ||
1655 | if ((address_family == AF_INET && host_sin->sin_addr.s_addr == sin->sin_addr.s_addr) || | 1726 | if ((address_family == AF_INET && host_sin->sin_addr.s_addr == sin->sin_addr.s_addr) || |
1656 | (address_family == AF_INET6 && host_sin6->sin6_addr.s6_addr == sin6->sin6_addr.s6_addr)) { | 1727 | (address_family == AF_INET6 && host_sin6->sin6_addr.s6_addr == sin6->sin6_addr.s6_addr)) { |
1657 | if (debug) | 1728 | if (debug) { |
1658 | printf("Identical IP already exists. Not adding %s\n", arg); | 1729 | printf("Identical IP already exists. Not adding %s\n", arg); |
1730 | } | ||
1659 | return -1; | 1731 | return -1; |
1660 | } | 1732 | } |
1661 | host = host->next; | 1733 | host = host->next; |
@@ -1700,10 +1772,11 @@ static int add_target_ip(char *arg, struct sockaddr_storage *in) { | |||
1700 | host->score_status = 0; | 1772 | host->score_status = 0; |
1701 | host->pl_status = 0; | 1773 | host->pl_status = 0; |
1702 | 1774 | ||
1703 | if (!list) | 1775 | if (!list) { |
1704 | list = cursor = host; | 1776 | list = cursor = host; |
1705 | else | 1777 | } else { |
1706 | cursor->next = host; | 1778 | cursor->next = host; |
1779 | } | ||
1707 | 1780 | ||
1708 | cursor = host; | 1781 | cursor = host; |
1709 | targets++; | 1782 | targets++; |
@@ -1777,8 +1850,9 @@ static int add_target(char *arg) { | |||
1777 | 1850 | ||
1778 | /* this is silly, but it works */ | 1851 | /* this is silly, but it works */ |
1779 | if (mode == MODE_HOSTCHECK || mode == MODE_ALL) { | 1852 | if (mode == MODE_HOSTCHECK || mode == MODE_ALL) { |
1780 | if (debug > 2) | 1853 | if (debug > 2) { |
1781 | printf("mode: %d\n", mode); | 1854 | printf("mode: %d\n", mode); |
1855 | } | ||
1782 | continue; | 1856 | continue; |
1783 | } | 1857 | } |
1784 | break; | 1858 | break; |
@@ -1793,10 +1867,12 @@ static void set_source_ip(char *arg) { | |||
1793 | 1867 | ||
1794 | memset(&src, 0, sizeof(src)); | 1868 | memset(&src, 0, sizeof(src)); |
1795 | src.sin_family = address_family; | 1869 | src.sin_family = address_family; |
1796 | if ((src.sin_addr.s_addr = inet_addr(arg)) == INADDR_NONE) | 1870 | if ((src.sin_addr.s_addr = inet_addr(arg)) == INADDR_NONE) { |
1797 | src.sin_addr.s_addr = get_ip_address(arg); | 1871 | src.sin_addr.s_addr = get_ip_address(arg); |
1798 | if (bind(icmp_sock, (struct sockaddr *)&src, sizeof(src)) == -1) | 1872 | } |
1873 | if (bind(icmp_sock, (struct sockaddr *)&src, sizeof(src)) == -1) { | ||
1799 | crash("Cannot bind to IP address %s", arg); | 1874 | crash("Cannot bind to IP address %s", arg); |
1875 | } | ||
1800 | } | 1876 | } |
1801 | 1877 | ||
1802 | /* TODO: Move this to netutils.c and also change check_dhcp to use that. */ | 1878 | /* TODO: Move this to netutils.c and also change check_dhcp to use that. */ |
@@ -1811,8 +1887,9 @@ static in_addr_t get_ip_address(const char *ifname) { | |||
1811 | 1887 | ||
1812 | ifr.ifr_name[sizeof(ifr.ifr_name) - 1] = '\0'; | 1888 | ifr.ifr_name[sizeof(ifr.ifr_name) - 1] = '\0'; |
1813 | 1889 | ||
1814 | if (ioctl(icmp_sock, SIOCGIFADDR, &ifr) == -1) | 1890 | if (ioctl(icmp_sock, SIOCGIFADDR, &ifr) == -1) { |
1815 | crash("Cannot determine IP address of interface %s", ifname); | 1891 | crash("Cannot determine IP address of interface %s", ifname); |
1892 | } | ||
1816 | 1893 | ||
1817 | memcpy(&ip, &ifr.ifr_addr, sizeof(ip)); | 1894 | memcpy(&ip, &ifr.ifr_addr, sizeof(ip)); |
1818 | #else | 1895 | #else |
@@ -1835,47 +1912,57 @@ static u_int get_timevar(const char *str) { | |||
1835 | u_int i, d; /* integer and decimal, respectively */ | 1912 | u_int i, d; /* integer and decimal, respectively */ |
1836 | u_int factor = 1000; /* default to milliseconds */ | 1913 | u_int factor = 1000; /* default to milliseconds */ |
1837 | 1914 | ||
1838 | if (!str) | 1915 | if (!str) { |
1839 | return 0; | 1916 | return 0; |
1917 | } | ||
1840 | len = strlen(str); | 1918 | len = strlen(str); |
1841 | if (!len) | 1919 | if (!len) { |
1842 | return 0; | 1920 | return 0; |
1921 | } | ||
1843 | 1922 | ||
1844 | /* unit might be given as ms|m (millisec), | 1923 | /* unit might be given as ms|m (millisec), |
1845 | * us|u (microsec) or just plain s, for seconds */ | 1924 | * us|u (microsec) or just plain s, for seconds */ |
1846 | p = '\0'; | 1925 | p = '\0'; |
1847 | u = str[len - 1]; | 1926 | u = str[len - 1]; |
1848 | if (len >= 2 && !isdigit((int)str[len - 2])) | 1927 | if (len >= 2 && !isdigit((int)str[len - 2])) { |
1849 | p = str[len - 2]; | 1928 | p = str[len - 2]; |
1850 | if (p && u == 's') | 1929 | } |
1930 | if (p && u == 's') { | ||
1851 | u = p; | 1931 | u = p; |
1852 | else if (!p) | 1932 | } else if (!p) { |
1853 | p = u; | 1933 | p = u; |
1854 | if (debug > 2) | 1934 | } |
1935 | if (debug > 2) { | ||
1855 | printf("evaluating %s, u: %c, p: %c\n", str, u, p); | 1936 | printf("evaluating %s, u: %c, p: %c\n", str, u, p); |
1937 | } | ||
1856 | 1938 | ||
1857 | if (u == 'u') | 1939 | if (u == 'u') { |
1858 | factor = 1; /* microseconds */ | 1940 | factor = 1; /* microseconds */ |
1859 | else if (u == 'm') | 1941 | } else if (u == 'm') { |
1860 | factor = 1000; /* milliseconds */ | 1942 | factor = 1000; /* milliseconds */ |
1861 | else if (u == 's') | 1943 | } else if (u == 's') { |
1862 | factor = 1000000; /* seconds */ | 1944 | factor = 1000000; /* seconds */ |
1863 | if (debug > 2) | 1945 | } |
1946 | if (debug > 2) { | ||
1864 | printf("factor is %u\n", factor); | 1947 | printf("factor is %u\n", factor); |
1948 | } | ||
1865 | 1949 | ||
1866 | i = strtoul(str, &ptr, 0); | 1950 | i = strtoul(str, &ptr, 0); |
1867 | if (!ptr || *ptr != '.' || strlen(ptr) < 2 || factor == 1) | 1951 | if (!ptr || *ptr != '.' || strlen(ptr) < 2 || factor == 1) { |
1868 | return i * factor; | 1952 | return i * factor; |
1953 | } | ||
1869 | 1954 | ||
1870 | /* time specified in usecs can't have decimal points, so ignore them */ | 1955 | /* time specified in usecs can't have decimal points, so ignore them */ |
1871 | if (factor == 1) | 1956 | if (factor == 1) { |
1872 | return i; | 1957 | return i; |
1958 | } | ||
1873 | 1959 | ||
1874 | d = strtoul(ptr + 1, NULL, 0); | 1960 | d = strtoul(ptr + 1, NULL, 0); |
1875 | 1961 | ||
1876 | /* d is decimal, so get rid of excess digits */ | 1962 | /* d is decimal, so get rid of excess digits */ |
1877 | while (d >= factor) | 1963 | while (d >= factor) { |
1878 | d /= 10; | 1964 | d /= 10; |
1965 | } | ||
1879 | 1966 | ||
1880 | /* the last parenthesis avoids floating point exceptions. */ | 1967 | /* the last parenthesis avoids floating point exceptions. */ |
1881 | return ((i * factor) + (d * (factor / 10))); | 1968 | return ((i * factor) + (d * (factor / 10))); |
@@ -1885,15 +1972,16 @@ static u_int get_timevar(const char *str) { | |||
1885 | static int get_threshold(char *str, threshold *th) { | 1972 | static int get_threshold(char *str, threshold *th) { |
1886 | char *p = NULL, i = 0; | 1973 | char *p = NULL, i = 0; |
1887 | 1974 | ||
1888 | if (!str || !strlen(str) || !th) | 1975 | if (!str || !strlen(str) || !th) { |
1889 | return -1; | 1976 | return -1; |
1977 | } | ||
1890 | 1978 | ||
1891 | /* pointer magic slims code by 10 lines. i is bof-stop on stupid libc's */ | 1979 | /* pointer magic slims code by 10 lines. i is bof-stop on stupid libc's */ |
1892 | p = &str[strlen(str) - 1]; | 1980 | p = &str[strlen(str) - 1]; |
1893 | while (p != &str[1]) { | 1981 | while (p != &str[1]) { |
1894 | if (*p == '%') | 1982 | if (*p == '%') { |
1895 | *p = '\0'; | 1983 | *p = '\0'; |
1896 | else if (*p == ',' && i) { | 1984 | } else if (*p == ',' && i) { |
1897 | *p = '\0'; /* reset it so get_timevar(str) works nicely later */ | 1985 | *p = '\0'; /* reset it so get_timevar(str) works nicely later */ |
1898 | th->pl = (unsigned char)strtoul(p + 1, NULL, 0); | 1986 | th->pl = (unsigned char)strtoul(p + 1, NULL, 0); |
1899 | break; | 1987 | break; |
@@ -1903,13 +1991,16 @@ static int get_threshold(char *str, threshold *th) { | |||
1903 | } | 1991 | } |
1904 | th->rta = get_timevar(str); | 1992 | th->rta = get_timevar(str); |
1905 | 1993 | ||
1906 | if (!th->rta) | 1994 | if (!th->rta) { |
1907 | return -1; | 1995 | return -1; |
1996 | } | ||
1908 | 1997 | ||
1909 | if (th->rta > MAXTTL * 1000000) | 1998 | if (th->rta > MAXTTL * 1000000) { |
1910 | th->rta = MAXTTL * 1000000; | 1999 | th->rta = MAXTTL * 1000000; |
1911 | if (th->pl > 100) | 2000 | } |
2001 | if (th->pl > 100) { | ||
1912 | th->pl = 100; | 2002 | th->pl = 100; |
2003 | } | ||
1913 | 2004 | ||
1914 | return 0; | 2005 | return 0; |
1915 | } | 2006 | } |
@@ -1925,8 +2016,9 @@ static int get_threshold(char *str, threshold *th) { | |||
1925 | * @param[in] mode Determines whether this a threshold for rta, packet_loss, jitter, mos or score (exclusively) | 2016 | * @param[in] mode Determines whether this a threshold for rta, packet_loss, jitter, mos or score (exclusively) |
1926 | */ | 2017 | */ |
1927 | static bool get_threshold2(char *str, size_t length, threshold *warn, threshold *crit, threshold_mode mode) { | 2018 | static bool get_threshold2(char *str, size_t length, threshold *warn, threshold *crit, threshold_mode mode) { |
1928 | if (!str || !length || !warn || !crit) | 2019 | if (!str || !length || !warn || !crit) { |
1929 | return false; | 2020 | return false; |
2021 | } | ||
1930 | 2022 | ||
1931 | // p points to the last char in str | 2023 | // p points to the last char in str |
1932 | char *p = &str[length - 1]; | 2024 | char *p = &str[length - 1]; |
@@ -1999,8 +2091,9 @@ unsigned short icmp_checksum(uint16_t *p, size_t n) { | |||
1999 | } | 2091 | } |
2000 | 2092 | ||
2001 | /* mop up the occasional odd byte */ | 2093 | /* mop up the occasional odd byte */ |
2002 | if (n == 1) | 2094 | if (n == 1) { |
2003 | sum += *((uint8_t *)p - 1); | 2095 | sum += *((uint8_t *)p - 1); |
2096 | } | ||
2004 | 2097 | ||
2005 | sum = (sum >> 16) + (sum & 0xffff); /* add hi 16 to low 16 */ | 2098 | sum = (sum >> 16) + (sum & 0xffff); /* add hi 16 to low 16 */ |
2006 | sum += (sum >> 16); /* add carry */ | 2099 | sum += (sum >> 16); /* add carry */ |