diff options
author | Denys Vlasenko <vda.linux@googlemail.com> | 2018-02-13 23:21:33 +0100 |
---|---|---|
committer | Denys Vlasenko <vda.linux@googlemail.com> | 2018-02-13 23:21:33 +0100 |
commit | 7189af847fd8a54a85f08e9bf51cd103f0d87393 (patch) | |
tree | 2329c5eb95e97de80ed0efe071539fb23a3054f8 | |
parent | 4892f3a4c77fb5051d64cfc5f9056a7d6ec6cdb2 (diff) | |
download | busybox-w32-7189af847fd8a54a85f08e9bf51cd103f0d87393.tar.gz busybox-w32-7189af847fd8a54a85f08e9bf51cd103f0d87393.tar.bz2 busybox-w32-7189af847fd8a54a85f08e9bf51cd103f0d87393.zip |
ping: implement -A "adaptive ping"
function old new delta
common_ping_main 1757 1862 +105
packed_usage 32367 32427 +60
sendping_tail 236 209 -27
------------------------------------------------------------------------------
(add/remove: 0/0 grow/shrink: 2/1 up/down: 165/-27) Total: 138 bytes
Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
-rw-r--r-- | networking/ping.c | 100 |
1 files changed, 60 insertions, 40 deletions
diff --git a/networking/ping.c b/networking/ping.c index d1d59d545..bf750d032 100644 --- a/networking/ping.c +++ b/networking/ping.c | |||
@@ -74,6 +74,7 @@ | |||
74 | //usage: ) | 74 | //usage: ) |
75 | //usage: "\n -c CNT Send only CNT pings" | 75 | //usage: "\n -c CNT Send only CNT pings" |
76 | //usage: "\n -s SIZE Send SIZE data bytes in packets (default 56)" | 76 | //usage: "\n -s SIZE Send SIZE data bytes in packets (default 56)" |
77 | //usage: "\n -A Ping as soon as reply is recevied" | ||
77 | //usage: "\n -t TTL Set TTL" | 78 | //usage: "\n -t TTL Set TTL" |
78 | //usage: "\n -I IFACE/IP Source interface or IP address" | 79 | //usage: "\n -I IFACE/IP Source interface or IP address" |
79 | //usage: "\n -W SEC Seconds to wait for the first response (default 10)" | 80 | //usage: "\n -W SEC Seconds to wait for the first response (default 10)" |
@@ -90,6 +91,7 @@ | |||
90 | //usage: "Send ICMP ECHO_REQUEST packets to network hosts\n" | 91 | //usage: "Send ICMP ECHO_REQUEST packets to network hosts\n" |
91 | //usage: "\n -c CNT Send only CNT pings" | 92 | //usage: "\n -c CNT Send only CNT pings" |
92 | //usage: "\n -s SIZE Send SIZE data bytes in packets (default 56)" | 93 | //usage: "\n -s SIZE Send SIZE data bytes in packets (default 56)" |
94 | //usage: "\n -A Ping as soon as reply is recevied" | ||
93 | //usage: "\n -I IFACE/IP Source interface or IP address" | 95 | //usage: "\n -I IFACE/IP Source interface or IP address" |
94 | //usage: "\n -q Quiet, only display output at start" | 96 | //usage: "\n -q Quiet, only display output at start" |
95 | //usage: "\n and when finished" | 97 | //usage: "\n and when finished" |
@@ -348,20 +350,21 @@ static int common_ping_main(sa_family_t af, char **argv) | |||
348 | /* Full(er) version */ | 350 | /* Full(er) version */ |
349 | 351 | ||
350 | /* -c NUM, -t NUM, -w NUM, -W NUM */ | 352 | /* -c NUM, -t NUM, -w NUM, -W NUM */ |
351 | #define OPT_STRING "qvc:+s:t:+w:+W:+I:np:4"IF_PING6("6") | 353 | #define OPT_STRING "qvAc:+s:t:+w:+W:+I:np:4"IF_PING6("6") |
352 | enum { | 354 | enum { |
353 | OPT_QUIET = 1 << 0, | 355 | OPT_QUIET = 1 << 0, |
354 | OPT_VERBOSE = 1 << 1, | 356 | OPT_VERBOSE = 1 << 1, |
355 | OPT_c = 1 << 2, | 357 | OPT_A = 1 << 2, |
356 | OPT_s = 1 << 3, | 358 | OPT_c = 1 << 3, |
357 | OPT_t = 1 << 4, | 359 | OPT_s = 1 << 4, |
358 | OPT_w = 1 << 5, | 360 | OPT_t = 1 << 5, |
359 | OPT_W = 1 << 6, | 361 | OPT_w = 1 << 6, |
360 | OPT_I = 1 << 7, | 362 | OPT_W = 1 << 7, |
361 | /*OPT_n = 1 << 8, - ignored */ | 363 | OPT_I = 1 << 8, |
362 | OPT_p = 1 << 9, | 364 | /*OPT_n = 1 << 9, - ignored */ |
363 | OPT_IPV4 = 1 << 10, | 365 | OPT_p = 1 << 10, |
364 | OPT_IPV6 = (1 << 11) * ENABLE_PING6, | 366 | OPT_IPV4 = 1 << 11, |
367 | OPT_IPV6 = (1 << 12) * ENABLE_PING6, | ||
365 | }; | 368 | }; |
366 | 369 | ||
367 | 370 | ||
@@ -377,9 +380,8 @@ struct globals { | |||
377 | uint8_t pattern; | 380 | uint8_t pattern; |
378 | unsigned tmin, tmax; /* in us */ | 381 | unsigned tmin, tmax; /* in us */ |
379 | unsigned long long tsum; /* in us, sum of all times */ | 382 | unsigned long long tsum; /* in us, sum of all times */ |
380 | unsigned deadline; | 383 | unsigned deadline_ms; |
381 | unsigned timeout; | 384 | unsigned timeout; |
382 | unsigned total_secs; | ||
383 | unsigned sizeof_rcv_packet; | 385 | unsigned sizeof_rcv_packet; |
384 | char *rcv_packet; /* [datalen + MAXIPLEN + MAXICMPLEN] */ | 386 | char *rcv_packet; /* [datalen + MAXIPLEN + MAXICMPLEN] */ |
385 | void *snd_packet; /* [datalen + ipv4/ipv6_const] */ | 387 | void *snd_packet; /* [datalen + ipv4/ipv6_const] */ |
@@ -405,9 +407,7 @@ struct globals { | |||
405 | #define tmin (G.tmin ) | 407 | #define tmin (G.tmin ) |
406 | #define tmax (G.tmax ) | 408 | #define tmax (G.tmax ) |
407 | #define tsum (G.tsum ) | 409 | #define tsum (G.tsum ) |
408 | #define deadline (G.deadline ) | ||
409 | #define timeout (G.timeout ) | 410 | #define timeout (G.timeout ) |
410 | #define total_secs (G.total_secs ) | ||
411 | #define hostname (G.hostname ) | 411 | #define hostname (G.hostname ) |
412 | #define dotted (G.dotted ) | 412 | #define dotted (G.dotted ) |
413 | #define pingaddr (G.pingaddr ) | 413 | #define pingaddr (G.pingaddr ) |
@@ -455,7 +455,7 @@ static void print_stats_and_exit(int junk UNUSED_PARAM) | |||
455 | tmax / 1000, tmax % 1000); | 455 | tmax / 1000, tmax % 1000); |
456 | } | 456 | } |
457 | /* if condition is true, exit with 1 -- 'failure' */ | 457 | /* if condition is true, exit with 1 -- 'failure' */ |
458 | exit(nrecv == 0 || (deadline && nrecv < pingcount)); | 458 | exit(nrecv == 0 || (G.deadline_ms && nrecv < pingcount)); |
459 | } | 459 | } |
460 | 460 | ||
461 | static void sendping_tail(void (*sp)(int), int size_pkt) | 461 | static void sendping_tail(void (*sp)(int), int size_pkt) |
@@ -467,22 +467,23 @@ static void sendping_tail(void (*sp)(int), int size_pkt) | |||
467 | 467 | ||
468 | size_pkt += datalen; | 468 | size_pkt += datalen; |
469 | 469 | ||
470 | if (G.deadline_ms) { | ||
471 | unsigned n = ((unsigned)monotonic_ms()) - G.deadline_ms; | ||
472 | if ((int)n >= 0) | ||
473 | print_stats_and_exit(0); | ||
474 | } | ||
475 | |||
470 | /* sizeof(pingaddr) can be larger than real sa size, but I think | 476 | /* sizeof(pingaddr) can be larger than real sa size, but I think |
471 | * it doesn't matter */ | 477 | * it doesn't matter */ |
472 | sz = xsendto(pingsock, G.snd_packet, size_pkt, &pingaddr.sa, sizeof(pingaddr)); | 478 | sz = xsendto(pingsock, G.snd_packet, size_pkt, &pingaddr.sa, sizeof(pingaddr)); |
473 | if (sz != size_pkt) | 479 | if (sz != size_pkt) |
474 | bb_error_msg_and_die(bb_msg_write_error); | 480 | bb_error_msg_and_die(bb_msg_write_error); |
475 | 481 | ||
476 | if (pingcount == 0 || deadline || G.ntransmitted < pingcount) { | 482 | if (pingcount == 0 || G.ntransmitted < pingcount) { |
477 | /* Didn't send all pings yet - schedule next in 1s */ | 483 | /* Didn't send all pings yet - schedule next in 1s */ |
478 | signal(SIGALRM, sp); | 484 | signal(SIGALRM, sp); |
479 | if (deadline) { | ||
480 | total_secs += PINGINTERVAL; | ||
481 | if (total_secs >= deadline) | ||
482 | signal(SIGALRM, print_stats_and_exit); | ||
483 | } | ||
484 | alarm(PINGINTERVAL); | 485 | alarm(PINGINTERVAL); |
485 | } else { /* -c NN, and all NN are sent (and no deadline) */ | 486 | } else { /* -c NN, and all NN are sent */ |
486 | /* Wait for the last ping to come back. | 487 | /* Wait for the last ping to come back. |
487 | * -W timeout: wait for a response in seconds. | 488 | * -W timeout: wait for a response in seconds. |
488 | * Affects only timeout in absence of any responses, | 489 | * Affects only timeout in absence of any responses, |
@@ -632,7 +633,7 @@ static void unpack_tail(int sz, uint32_t *tp, | |||
632 | puts(dupmsg); | 633 | puts(dupmsg); |
633 | fflush_all(); | 634 | fflush_all(); |
634 | } | 635 | } |
635 | static void unpack4(char *buf, int sz, struct sockaddr_in *from) | 636 | static int unpack4(char *buf, int sz, struct sockaddr_in *from) |
636 | { | 637 | { |
637 | struct icmp *icmppkt; | 638 | struct icmp *icmppkt; |
638 | struct iphdr *iphdr; | 639 | struct iphdr *iphdr; |
@@ -640,7 +641,7 @@ static void unpack4(char *buf, int sz, struct sockaddr_in *from) | |||
640 | 641 | ||
641 | /* discard if too short */ | 642 | /* discard if too short */ |
642 | if (sz < (datalen + ICMP_MINLEN)) | 643 | if (sz < (datalen + ICMP_MINLEN)) |
643 | return; | 644 | return 0; |
644 | 645 | ||
645 | /* check IP header */ | 646 | /* check IP header */ |
646 | iphdr = (struct iphdr *) buf; | 647 | iphdr = (struct iphdr *) buf; |
@@ -648,7 +649,7 @@ static void unpack4(char *buf, int sz, struct sockaddr_in *from) | |||
648 | sz -= hlen; | 649 | sz -= hlen; |
649 | icmppkt = (struct icmp *) (buf + hlen); | 650 | icmppkt = (struct icmp *) (buf + hlen); |
650 | if (icmppkt->icmp_id != myid) | 651 | if (icmppkt->icmp_id != myid) |
651 | return; /* not our ping */ | 652 | return 0; /* not our ping */ |
652 | 653 | ||
653 | if (icmppkt->icmp_type == ICMP_ECHOREPLY) { | 654 | if (icmppkt->icmp_type == ICMP_ECHOREPLY) { |
654 | uint16_t recv_seq = ntohs(icmppkt->icmp_seq); | 655 | uint16_t recv_seq = ntohs(icmppkt->icmp_seq); |
@@ -659,25 +660,28 @@ static void unpack4(char *buf, int sz, struct sockaddr_in *from) | |||
659 | unpack_tail(sz, tp, | 660 | unpack_tail(sz, tp, |
660 | inet_ntoa(*(struct in_addr *) &from->sin_addr.s_addr), | 661 | inet_ntoa(*(struct in_addr *) &from->sin_addr.s_addr), |
661 | recv_seq, iphdr->ttl); | 662 | recv_seq, iphdr->ttl); |
662 | } else if (icmppkt->icmp_type != ICMP_ECHO) { | 663 | return 1; |
664 | } | ||
665 | if (icmppkt->icmp_type != ICMP_ECHO) { | ||
663 | bb_error_msg("warning: got ICMP %d (%s)", | 666 | bb_error_msg("warning: got ICMP %d (%s)", |
664 | icmppkt->icmp_type, | 667 | icmppkt->icmp_type, |
665 | icmp_type_name(icmppkt->icmp_type)); | 668 | icmp_type_name(icmppkt->icmp_type)); |
666 | } | 669 | } |
670 | return 0; | ||
667 | } | 671 | } |
668 | #if ENABLE_PING6 | 672 | #if ENABLE_PING6 |
669 | static void unpack6(char *packet, int sz, struct sockaddr_in6 *from, int hoplimit) | 673 | static int unpack6(char *packet, int sz, struct sockaddr_in6 *from, int hoplimit) |
670 | { | 674 | { |
671 | struct icmp6_hdr *icmppkt; | 675 | struct icmp6_hdr *icmppkt; |
672 | char buf[INET6_ADDRSTRLEN]; | 676 | char buf[INET6_ADDRSTRLEN]; |
673 | 677 | ||
674 | /* discard if too short */ | 678 | /* discard if too short */ |
675 | if (sz < (datalen + sizeof(struct icmp6_hdr))) | 679 | if (sz < (datalen + sizeof(struct icmp6_hdr))) |
676 | return; | 680 | return 0; |
677 | 681 | ||
678 | icmppkt = (struct icmp6_hdr *) packet; | 682 | icmppkt = (struct icmp6_hdr *) packet; |
679 | if (icmppkt->icmp6_id != myid) | 683 | if (icmppkt->icmp6_id != myid) |
680 | return; /* not our ping */ | 684 | return 0; /* not our ping */ |
681 | 685 | ||
682 | if (icmppkt->icmp6_type == ICMP6_ECHO_REPLY) { | 686 | if (icmppkt->icmp6_type == ICMP6_ECHO_REPLY) { |
683 | uint16_t recv_seq = ntohs(icmppkt->icmp6_seq); | 687 | uint16_t recv_seq = ntohs(icmppkt->icmp6_seq); |
@@ -689,11 +693,14 @@ static void unpack6(char *packet, int sz, struct sockaddr_in6 *from, int hoplimi | |||
689 | inet_ntop(AF_INET6, &from->sin6_addr, | 693 | inet_ntop(AF_INET6, &from->sin6_addr, |
690 | buf, sizeof(buf)), | 694 | buf, sizeof(buf)), |
691 | recv_seq, hoplimit); | 695 | recv_seq, hoplimit); |
692 | } else if (icmppkt->icmp6_type != ICMP6_ECHO_REQUEST) { | 696 | return 1; |
697 | } | ||
698 | if (icmppkt->icmp6_type != ICMP6_ECHO_REQUEST) { | ||
693 | bb_error_msg("warning: got ICMP %d (%s)", | 699 | bb_error_msg("warning: got ICMP %d (%s)", |
694 | icmppkt->icmp6_type, | 700 | icmppkt->icmp6_type, |
695 | icmp6_type_name(icmppkt->icmp6_type)); | 701 | icmp6_type_name(icmppkt->icmp6_type)); |
696 | } | 702 | } |
703 | return 0; | ||
697 | } | 704 | } |
698 | #endif | 705 | #endif |
699 | 706 | ||
@@ -726,6 +733,7 @@ static void ping4(len_and_sockaddr *lsa) | |||
726 | signal(SIGINT, print_stats_and_exit); | 733 | signal(SIGINT, print_stats_and_exit); |
727 | 734 | ||
728 | /* start the ping's going ... */ | 735 | /* start the ping's going ... */ |
736 | send_ping: | ||
729 | sendping4(0); | 737 | sendping4(0); |
730 | 738 | ||
731 | /* listen for replies */ | 739 | /* listen for replies */ |
@@ -741,9 +749,12 @@ static void ping4(len_and_sockaddr *lsa) | |||
741 | bb_perror_msg("recvfrom"); | 749 | bb_perror_msg("recvfrom"); |
742 | continue; | 750 | continue; |
743 | } | 751 | } |
744 | unpack4(G.rcv_packet, c, &from); | 752 | c = unpack4(G.rcv_packet, c, &from); |
745 | if (pingcount && G.nreceived >= pingcount) | 753 | if (pingcount && G.nreceived >= pingcount) |
746 | break; | 754 | break; |
755 | if (c && (option_mask32 & OPT_A)) { | ||
756 | goto send_ping; | ||
757 | } | ||
747 | } | 758 | } |
748 | } | 759 | } |
749 | #if ENABLE_PING6 | 760 | #if ENABLE_PING6 |
@@ -794,10 +805,6 @@ static void ping6(len_and_sockaddr *lsa) | |||
794 | 805 | ||
795 | signal(SIGINT, print_stats_and_exit); | 806 | signal(SIGINT, print_stats_and_exit); |
796 | 807 | ||
797 | /* start the ping's going ... */ | ||
798 | sendping6(0); | ||
799 | |||
800 | /* listen for replies */ | ||
801 | msg.msg_name = &from; | 808 | msg.msg_name = &from; |
802 | msg.msg_namelen = sizeof(from); | 809 | msg.msg_namelen = sizeof(from); |
803 | msg.msg_iov = &iov; | 810 | msg.msg_iov = &iov; |
@@ -805,12 +812,18 @@ static void ping6(len_and_sockaddr *lsa) | |||
805 | msg.msg_control = control_buf; | 812 | msg.msg_control = control_buf; |
806 | iov.iov_base = G.rcv_packet; | 813 | iov.iov_base = G.rcv_packet; |
807 | iov.iov_len = G.sizeof_rcv_packet; | 814 | iov.iov_len = G.sizeof_rcv_packet; |
815 | |||
816 | /* start the ping's going ... */ | ||
817 | send_ping: | ||
818 | sendping6(0); | ||
819 | |||
820 | /* listen for replies */ | ||
808 | while (1) { | 821 | while (1) { |
809 | int c; | 822 | int c; |
810 | struct cmsghdr *mp; | 823 | struct cmsghdr *mp; |
811 | int hoplimit = -1; | 824 | int hoplimit = -1; |
812 | msg.msg_controllen = sizeof(control_buf); | ||
813 | 825 | ||
826 | msg.msg_controllen = sizeof(control_buf); | ||
814 | c = recvmsg(pingsock, &msg, 0); | 827 | c = recvmsg(pingsock, &msg, 0); |
815 | if (c < 0) { | 828 | if (c < 0) { |
816 | if (errno != EINTR) | 829 | if (errno != EINTR) |
@@ -827,9 +840,12 @@ static void ping6(len_and_sockaddr *lsa) | |||
827 | move_from_unaligned_int(hoplimit, CMSG_DATA(mp)); | 840 | move_from_unaligned_int(hoplimit, CMSG_DATA(mp)); |
828 | } | 841 | } |
829 | } | 842 | } |
830 | unpack6(G.rcv_packet, c, &from, hoplimit); | 843 | c = unpack6(G.rcv_packet, c, &from, hoplimit); |
831 | if (pingcount && G.nreceived >= pingcount) | 844 | if (pingcount && G.nreceived >= pingcount) |
832 | break; | 845 | break; |
846 | if (c && (option_mask32 & OPT_A)) { | ||
847 | goto send_ping; | ||
848 | } | ||
833 | } | 849 | } |
834 | } | 850 | } |
835 | #endif | 851 | #endif |
@@ -875,7 +891,7 @@ static int common_ping_main(int opt, char **argv) | |||
875 | OPT_STRING | 891 | OPT_STRING |
876 | /* exactly one arg; -v and -q don't mix */ | 892 | /* exactly one arg; -v and -q don't mix */ |
877 | "\0" "=1:q--v:v--q", | 893 | "\0" "=1:q--v:v--q", |
878 | &pingcount, &str_s, &opt_ttl, &deadline, &timeout, &str_I, &str_p | 894 | &pingcount, &str_s, &opt_ttl, &G.deadline_ms, &timeout, &str_I, &str_p |
879 | ); | 895 | ); |
880 | if (opt & OPT_s) | 896 | if (opt & OPT_s) |
881 | datalen = xatou16(str_s); // -s | 897 | datalen = xatou16(str_s); // -s |
@@ -889,6 +905,10 @@ static int common_ping_main(int opt, char **argv) | |||
889 | } | 905 | } |
890 | if (opt & OPT_p) | 906 | if (opt & OPT_p) |
891 | G.pattern = xstrtou_range(str_p, 16, 0, 255); | 907 | G.pattern = xstrtou_range(str_p, 16, 0, 255); |
908 | if (G.deadline_ms) { | ||
909 | unsigned d = G.deadline_ms < INT_MAX/1000 ? G.deadline_ms : INT_MAX/1000; | ||
910 | G.deadline_ms = 1 | ((d * 1000) + monotonic_ms()); | ||
911 | } | ||
892 | 912 | ||
893 | myid = (uint16_t) getpid(); | 913 | myid = (uint16_t) getpid(); |
894 | hostname = argv[optind]; | 914 | hostname = argv[optind]; |
@@ -911,7 +931,7 @@ static int common_ping_main(int opt, char **argv) | |||
911 | 931 | ||
912 | dotted = xmalloc_sockaddr2dotted_noport(&lsa->u.sa); | 932 | dotted = xmalloc_sockaddr2dotted_noport(&lsa->u.sa); |
913 | ping(lsa); | 933 | ping(lsa); |
914 | print_stats_and_exit(EXIT_SUCCESS); | 934 | print_stats_and_exit(0); |
915 | /*return EXIT_SUCCESS;*/ | 935 | /*return EXIT_SUCCESS;*/ |
916 | } | 936 | } |
917 | #endif /* FEATURE_FANCY_PING */ | 937 | #endif /* FEATURE_FANCY_PING */ |