From 873bc1df2dd5a3d7e13b0383e80b2c066da5dc92 Mon Sep 17 00:00:00 2001 From: David Schweikert Date: Fri, 9 Dec 2011 20:06:12 -0500 Subject: [PATCH] Imported Debian patches: 2.4b2-to-ipv6-16.1 --- fping.8 | 19 +++--- fping.c | 194 ++++++++++++++++++++++++++++++++++++++++++-------------- 2 files changed, 159 insertions(+), 54 deletions(-) diff --git a/fping.8 b/fping.8 index 254804e..a1e8411 100755 --- a/fping.8 +++ b/fping.8 @@ -1,4 +1,4 @@ -.TH fping l +.TH fping 8 .SH NAME fping \- send ICMP ECHO_REQUEST packets to network hosts .SH SYNOPSIS @@ -12,9 +12,9 @@ fping \- send ICMP ECHO_REQUEST packets to network hosts .B fping -is a -.MS ping 8 -like program which uses the Internet Control +is a program like +.B ping(8) +which uses the Internet Control Message Protocol (ICMP) echo request to determine if a target host is responding. .B fping @@ -37,8 +37,7 @@ looping indefinitely (as in ). .PP Unlike -.B ping -, +.B ping, .B fping is meant to be used in scripts, so its output is designed to be easy to parse. @@ -130,12 +129,16 @@ Retry limit (default 3). This is the number of times an attempt at pinging a target will be made, not including the first try. .IP \fB-s\fR 5 Print cumulative statistics upon exit. +.IP \fB-S\fIaddr\fR 5 +Set source address. .IP \fB-t\fIn\fR 5 Initial target timeout in milliseconds (default 500). In the default mode, this is the amount of time that .B fping waits for a response to its first request. Successive timeouts are multiplied by the backoff factor. +.IP \fB-T\fIn\fR 5 +Select timeout in seconds (default 10). .IP \fB-u\fR 5 Show targets that are unreachable. .IP \fB-v\fR 5 @@ -152,7 +155,7 @@ Sure the open2 usage is not needed in this example, but it's a good open2 example none the less. .nf -#!/usr/local/bin/perl +#!/usr/bin/perl require 'open2.pl'; $MAILTO = "root"; @@ -178,7 +181,7 @@ Another good example is when you want to perform an action only on hosts that are currently reachable. .nf -#!/usr/local/bin/perl +#!/usr/bin/perl $hosts_to_backup = `cat /etc/hosts.backup | fping -a`; diff --git a/fping.c b/fping.c index 7d0c525..ff37c7e 100755 --- a/fping.c +++ b/fping.c @@ -24,7 +24,7 @@ * * Original author: Roland Schemers * IPv6 Support: Jeroen Massar - * + * Bugfixes, byte order & senseful seq.-numbers: Stephan Fuhrmann (stephan.fuhrmann AT 1und1.de) * * * RCS header information no longer used. It has been moved to the @@ -42,7 +42,6 @@ * IMPLIED WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED * WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR A PARTICULAR PURPOSE. */ -#define IPV6 1 /* This should be a compiler option, or even better be done from the Makefile... ;) */ #ifndef _NO_PROTO #if !__STDC__ && !defined( __cplusplus ) && !defined( FUNCPROTO ) \ @@ -101,13 +100,8 @@ extern "C" #endif #include -/* Linux has bizarre ip.h and ip_icmp.h */ -#if defined( __linux__ ) -#include "linux.h" -#else #include #include -#endif /* defined(__linux__) */ #include #include @@ -150,7 +144,11 @@ typedef struct ping_data #define MIN_PING_DATA sizeof( PING_DATA ) #define MAX_IP_PACKET 65536 /* (theoretical) max IP packet size */ #define SIZE_IP_HDR 20 +#ifndef IPV6 #define SIZE_ICMP_HDR ICMP_MINLEN /* from ip_icmp.h */ +#else +#define SIZE_ICMP_HDR sizeof(FPING_ICMPHDR) +#endif #define MAX_PING_DATA ( MAX_IP_PACKET - SIZE_IP_HDR - SIZE_ICMP_HDR ) /* sized so as to be like traditional ping */ @@ -192,7 +190,7 @@ char *icmp_type_str[19] = "", "", "ICMP Time Exceeded", /* 11 */ - "ICMP Paramter Problem", /* 12 */ + "ICMP Parameter Problem", /* 12 */ "ICMP Timestamp Request", /* 13 */ "ICMP Timestamp Reply", /* 14 */ "ICMP Information Request", /* 15 */ @@ -283,6 +281,12 @@ u_int ping_pkt_size; u_int count = 1; u_int trials; u_int report_interval = 0; +int src_addr_present = 0; +#ifndef IPV6 +struct in_addr src_addr; +#else +struct in6_addr src_addr; +#endif /* global stats */ long max_reply = 0; @@ -292,6 +296,7 @@ double sum_replies = 0; int max_hostname_len = 0; int num_jobs = 0; /* number of hosts still to do */ int num_hosts; /* total number of hosts */ +int max_seq_sent = 0; /* maximum sequence number sent so far */ int num_alive = 0, /* total number alive */ num_unreachable = 0, /* total number unreachable */ num_noaddress = 0; /* total number of addresses not found */ @@ -408,6 +413,11 @@ int main( int argc, char **argv ) struct protoent *proto; char *buf; uid_t uid; +#ifndef IPV6 + struct sockaddr_in sa; +#else + struct sockaddr_in6 sa; +#endif /* check if we are root */ if( geteuid() ) @@ -439,7 +449,7 @@ int main( int argc, char **argv ) #ifdef IPV6 /* - * let the kerel pass extension headers of incoming packets, + * let the kernel pass extension headers of incoming packets, * for privileged socket options */ #ifdef IPV6_RECVHOPOPTS @@ -474,6 +484,35 @@ int main( int argc, char **argv ) sizeof(opton))) err(1, "setsockopt(IPV6_RTHDR)"); #endif +#ifndef USE_SIN6_SCOPE_ID +#ifdef IPV6_RECVPKTINFO + if (setsockopt(s, IPPROTO_IPV6, IPV6_RECVPKTINFO, &opton, + sizeof(opton))) + err(1, "setsockopt(IPV6_RECVPKTINFO)"); +#else /* old adv. API */ + if (setsockopt(s, IPPROTO_IPV6, IPV6_PKTINFO, &opton, + sizeof(opton))) + err(1, "setsockopt(IPV6_PKTINFO)"); +#endif +#endif /* USE_SIN6_SCOPE_ID */ +#ifdef IPV6_RECVHOPLIMIT + if (setsockopt(s, IPPROTO_IPV6, IPV6_RECVHOPLIMIT, &opton, + sizeof(opton))) + err(1, "setsockopt(IPV6_RECVHOPLIMIT)"); +#else /* old adv. API */ + if (setsockopt(s, IPPROTO_IPV6, IPV6_HOPLIMIT, &opton, + sizeof(opton))) + err(1, "setsockopt(IPV6_HOPLIMIT)"); +#endif +#ifdef IPV6_CHECKSUM +#ifndef SOL_RAW +#define SOL_RAW IPPROTO_IPV6 +#endif + opton = 2; + if (setsockopt(s, SOL_RAW, IPV6_CHECKSUM, &opton, + sizeof(opton))) + err(1, "setsockopt(SOL_RAW,IPV6_CHECKSUM)"); +#endif #endif if( ( uid = getuid() ) ) @@ -491,7 +530,7 @@ int main( int argc, char **argv ) /* get command line options */ - while( ( c = getopt( argc, argv, "gedhlmnqusaAvz:t:i:p:f:r:c:b:C:Q:B:" ) ) != EOF ) + while( ( c = getopt( argc, argv, "gedhlmnqusaAvz:t:i:p:f:r:c:b:C:Q:B:S:I:T:" ) ) != EOF ) { switch( c ) { @@ -502,7 +541,7 @@ int main( int argc, char **argv ) break; case 'r': - if( !( retry = ( u_int )atoi( optarg ) ) ) + if( ( retry = ( u_int )atoi( optarg ) ) < 0 ) usage(); break; @@ -639,6 +678,33 @@ int main( int argc, char **argv ) generate_flag = 1; break; + case 'S': +#ifndef IPV6 + if( ! inet_pton( AF_INET, optarg, &src_addr ) ) +#else + if( ! inet_pton( AF_INET6, optarg, &src_addr ) ) +#endif + usage(); + src_addr_present = 1; + break; + + case 'I': +#ifdef SO_BINDTODEVICE + if (setsockopt(s, SOL_SOCKET, SO_BINDTODEVICE, optarg, + strlen(optarg))) + err(1, "setsockopt(AF_INET, SO_BINDTODEVICE)"); +#else + fprintf( stderr, + "Warning: SO_BINDTODEVICE not supported, argument -I %s ignored\n", + optarg ); +#endif + break; + + case 'T': + if ( ! ( select_time = ( u_int )atoi( optarg ) * 100 ) ) + usage(); + break; + default: usage(); break; @@ -821,7 +887,7 @@ int main( int argc, char **argv ) errno_crash_and_burn( "fopen" ); - while( fgets( line, 132, ping_file ) ) + while( fgets( line, sizeof(line), ping_file ) ) { if( sscanf( line, "%s", host ) != 1 ) continue; @@ -962,6 +1028,22 @@ int main( int argc, char **argv ) if( !num_hosts ) exit( 2 ); + /* set the source address */ + + if( src_addr_present ) + { + memset( &sa, 0, sizeof( sa ) ); +#ifndef IPV6 + sa.sin_family = AF_INET; + sa.sin_addr = src_addr; +#else + sa.sin6_family = AF_INET6; + sa.sin6_addr = src_addr; +#endif + if ( bind( s, (struct sockaddr *)&sa, sizeof( sa ) ) < 0 ) + errno_crash_and_burn( "cannot bind source address" ); + } + /* allocate array to hold outstanding ping requests */ table = ( HOST_ENTRY** )malloc( sizeof( HOST_ENTRY* ) * num_hosts ); @@ -1112,7 +1194,7 @@ int main( int argc, char **argv ) /* but allow time for the last one to come in */ if( count_flag ) { - if( ( cursor->num_sent >= count ) && ( ht > cursor->timeout ) ) + if( ( cursor->num_sent >= count ) && ( cursor->num_recv >= count || ht > cursor->timeout ) ) { remove_job( cursor ); continue; @@ -1153,6 +1235,7 @@ int main( int argc, char **argv ) finish(); + return 0; } /* main() */ @@ -1382,15 +1465,15 @@ void print_per_system_splits( void ) if( h->num_recv_i <= h->num_sent_i ) { fprintf( stderr, " xmt/rcv/%%loss = %d/%d/%d%%", - h->num_sent_i, h->num_recv_i, - ( ( h->num_sent_i - h->num_recv_i ) * 100 ) / h->num_sent_i ); + h->num_sent_i, h->num_recv_i, h->num_sent_i > 0 ? + ( ( h->num_sent_i - h->num_recv_i ) * 100 ) / h->num_sent_i : 0 ); }/* IF */ else { fprintf( stderr, " xmt/rcv/%%return = %d/%d/%d%%", - h->num_sent_i, h->num_recv_i, - ( ( h->num_recv_i * 100 ) / h->num_sent_i ) ); + h->num_sent_i, h->num_recv_i, h->num_sent_i > 0 ? + ( ( h->num_recv_i * 100 ) / h->num_sent_i ) : 0 ); }/* ELSE */ @@ -1504,12 +1587,15 @@ void send_ping( int s, HOST_ENTRY *h ) icp = ( FPING_ICMPHDR* )buffer; gettimeofday( &h->last_send_time, &tz ); + int myseq = h->num_sent * num_hosts + h->i; + max_seq_sent = myseq > max_seq_sent ? myseq : max_seq_sent; + #ifndef IPV6 icp->icmp_type = ICMP_ECHO; icp->icmp_code = 0; icp->icmp_cksum = 0; - icp->icmp_seq = h->i; - icp->icmp_id = ident; + icp->icmp_seq = htons(myseq); + icp->icmp_id = htons(ident); pdp = ( PING_DATA* )( buffer + SIZE_ICMP_HDR ); pdp->ping_ts = h->last_send_time; @@ -1519,8 +1605,8 @@ void send_ping( int s, HOST_ENTRY *h ) #else icp->icmp6_type = ICMP6_ECHO_REQUEST; icp->icmp6_code = 0; - icp->icmp6_seq = h->i; - icp->icmp6_id = ident; + icp->icmp6_seq = htons(myseq); + icp->icmp6_id = htons(ident); pdp = ( PING_DATA* )( buffer + SIZE_ICMP_HDR ); pdp->ping_ts = h->last_send_time; @@ -1667,25 +1753,25 @@ int wait_for_reply( void ) }/* IF */ #ifndef IPV6 - if( icp->icmp_id != ident ) + if( ntohs(icp->icmp_id) != ident ) #else - if( icp->icmp6_id != ident ) + if( ntohs(icp->icmp6_id) != ident ) #endif return 1; /* packet received, but not the one we are looking for! */ num_pingreceived++; #ifndef IPV6 - if( icp->icmp_seq >= ( n_short )num_hosts ) + if( ntohs(icp->icmp_seq) > max_seq_sent ) #else - if( icp->icmp6_seq >= ( n_short )num_hosts ) + if( ntohs(icp->icmp6_seq) > max_seq_sent ) #endif return( 1 ); /* packet received, don't worry about it anymore */ #ifndef IPV6 - n = icp->icmp_seq; + n = ntohs(icp->icmp_seq) % num_hosts; #else - n = icp->icmp6_seq; + n = ntohs(icp->icmp6_seq) % num_hosts; #endif h = table[n]; @@ -1824,6 +1910,7 @@ int wait_for_reply( void ) }/* IF */ + fflush( stdout ); return num_jobs; } /* wait_for_reply() */ @@ -1875,11 +1962,11 @@ int handle_random_icmp( FPING_ICMPHDR *p, int psize, FPING_SOCKADDR *addr ) sent_icmp = ( struct icmp* )( c + 28 ); if( ( sent_icmp->icmp_type == ICMP_ECHO ) && - ( sent_icmp->icmp_id == ident ) && - ( sent_icmp->icmp_seq < ( n_short )num_hosts ) ) + ( ntohs(sent_icmp->icmp_id) == ident ) && + ( ntohs(sent_icmp->icmp_seq) <= ( n_short )max_seq_sent ) ) { /* this is a response to a ping we sent */ - h = table[sent_icmp->icmp_seq]; + h = table[ntohs(sent_icmp->icmp_seq) % num_hosts]; if( p->icmp_code > ICMP_UNREACH_MAXTYPE ) { @@ -1888,11 +1975,11 @@ int handle_random_icmp( FPING_ICMPHDR *p, int psize, FPING_SOCKADDR *addr ) #else if( ( sent_icmp->icmp6_type == ICMP_ECHO ) && - ( sent_icmp->icmp6_id == ident ) && - ( sent_icmp->icmp6_seq < ( n_short )num_hosts ) ) + ( ntohs(sent_icmp->icmp6_id) == ident ) && + ( ntohs(sent_icmp->icmp6_seq) <= ( n_short )max_seq_sent ) ) { /* this is a response to a ping we sent */ - h = table[sent_icmp->icmp6_seq]; + h = table[ntohs(sent_icmp->icmp6_seq) % num_hosts]; if( p->icmp6_code > ICMP_UNREACH_MAXTYPE ) { @@ -1930,24 +2017,24 @@ int handle_random_icmp( FPING_ICMPHDR *p, int psize, FPING_SOCKADDR *addr ) case ICMP_PARAMPROB: sent_icmp = ( FPING_ICMPHDR* )( c + 28 ); #ifndef IPV6 - if( ( sent_icmp->icmp_type = ICMP_ECHO ) && - ( sent_icmp->icmp_id = ident ) && - ( sent_icmp->icmp_seq < ( n_short )num_hosts ) ) + if( ( sent_icmp->icmp_type == ICMP_ECHO ) && + ( ntohs(sent_icmp->icmp_id) == ident ) && + ( ntohs(sent_icmp->icmp_seq) <= ( n_short )max_seq_sent ) ) { /* this is a response to a ping we sent */ - h = table[sent_icmp->icmp_seq]; + h = table[ntohs(sent_icmp->icmp_seq) % num_hosts]; fprintf( stderr, "%s from %s for ICMP Echo sent to %s", icmp_type_str[p->icmp_type], inet_ntoa( addr->sin_addr ), h->host ); if( inet_addr( h->host ) == -1 ) fprintf( stderr, " (%s)", inet_ntoa( h->saddr.sin_addr ) ); #else - if( ( sent_icmp->icmp6_type = ICMP_ECHO ) && - ( sent_icmp->icmp6_id = ident ) && - ( sent_icmp->icmp6_seq < ( n_short )num_hosts ) ) + if( ( sent_icmp->icmp6_type == ICMP_ECHO ) && + ( ntohs(sent_icmp->icmp6_id) == ident ) && + ( ntohs(sent_icmp->icmp6_seq) <= ( n_short )max_seq_sent ) ) { /* this is a response to a ping we sent */ - h = table[sent_icmp->icmp6_seq]; + h = table[ntohs(sent_icmp->icmp6_seq) % num_hosts]; fprintf( stderr, "%s from %s for ICMP Echo sent to %s", icmp_type_str[p->icmp6_type], addr_ascii, h->host ); @@ -2165,20 +2252,33 @@ void add_name( char *name ) struct addrinfo *res, hints; int ret_ga; char *hostname; + size_t len; /* getaddrinfo */ bzero(&hints, sizeof(struct addrinfo)); - hints.ai_flags = AI_CANONNAME; + hints.ai_flags = name_flag ? AI_CANONNAME : 0; hints.ai_family = AF_INET6; hints.ai_socktype = SOCK_RAW; hints.ai_protocol = IPPROTO_ICMPV6; ret_ga = getaddrinfo(name, NULL, &hints, &res); - if (ret_ga) errx(1, "%s", gai_strerror(ret_ga)); + if (ret_ga) { + if(!quiet_flag) + warnx("%s", gai_strerror(ret_ga)); + num_noaddress++; + return; + } if (res->ai_canonname) hostname = res->ai_canonname; else hostname = name; - if (!res->ai_addr) errx(1, "getaddrinfo failed"); - (void)memcpy(&dst, res->ai_addr, sizeof(FPING_SOCKADDR)); /*res->ai_addrlen);*/ + if (!res->ai_addr) { + if(!quiet_flag) + warnx("getaddrinfo failed"); + num_noaddress++; + return; + } + len = res->ai_addrlen; + if (len > sizeof(FPING_SOCKADDR)) len = sizeof(FPING_SOCKADDR); + (void)memcpy(&dst, res->ai_addr, len); add_addr(name, name, &dst); #endif } /* add_name() */ @@ -2730,9 +2830,11 @@ void usage( void ) fprintf( stderr, " (in looping and counting modes, default %d)\n", perhost_interval / 100 ); fprintf( stderr, " -q quiet (don't show per-target/per-ping results)\n" ); fprintf( stderr, " -Q n same as -q, but show summary every n seconds\n" ); - fprintf( stderr, " -r n number of retries (default %d)\n", retry ); + fprintf( stderr, " -r n number of retries (default %d)\n", DEFAULT_RETRY ); fprintf( stderr, " -s print final stats\n" ); + fprintf( stderr, " -S addr set source address\n" ); fprintf( stderr, " -t n individual target initial timeout (in millisec) (default %d)\n", timeout / 100 ); + fprintf( stderr, " -T n set select timeout (default %d)\n", select_time / 100 ); fprintf( stderr, " -u show targets that are unreachable\n" ); fprintf( stderr, " -v show version\n" ); fprintf( stderr, " targets list of targets to check (if no -f specified)\n" ); -- 2.43.0