| /*--------------------------------------------------------------- |
| * Copyright (c) 1999,2000,2001,2002,2003 |
| * The Board of Trustees of the University of Illinois |
| * All Rights Reserved. |
| *--------------------------------------------------------------- |
| * Permission is hereby granted, free of charge, to any person |
| * obtaining a copy of this software (Iperf) and associated |
| * documentation files (the "Software"), to deal in the Software |
| * without restriction, including without limitation the |
| * rights to use, copy, modify, merge, publish, distribute, |
| * sublicense, and/or sell copies of the Software, and to permit |
| * persons to whom the Software is furnished to do |
| * so, subject to the following conditions: |
| * |
| * |
| * Redistributions of source code must retain the above |
| * copyright notice, this list of conditions and |
| * the following disclaimers. |
| * |
| * |
| * Redistributions in binary form must reproduce the above |
| * copyright notice, this list of conditions and the following |
| * disclaimers in the documentation and/or other materials |
| * provided with the distribution. |
| * |
| * |
| * Neither the names of the University of Illinois, NCSA, |
| * nor the names of its contributors may be used to endorse |
| * or promote products derived from this Software without |
| * specific prior written permission. |
| * |
| * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, |
| * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES |
| * OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND |
| * NONINFRINGEMENT. IN NO EVENT SHALL THE CONTIBUTORS OR COPYRIGHT |
| * HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, |
| * WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, |
| * ARISING FROM, OUT OF OR IN CONNECTION WITH THE |
| * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. |
| * ________________________________________________________________ |
| * National Laboratory for Applied Network Research |
| * National Center for Supercomputing Applications |
| * University of Illinois at Urbana-Champaign |
| * http://www.ncsa.uiuc.edu |
| * ________________________________________________________________ |
| * |
| * ReportDefault.c |
| * by Kevin Gibbs <kgibbs@nlanr.net> |
| * |
| * ________________________________________________________________ */ |
| |
| #include "headers.h" |
| #include "Settings.hpp" |
| #include "util.h" |
| #include "Reporter.h" |
| #include "report_default.h" |
| #include "Thread.h" |
| #include "Locale.h" |
| #include "PerfSocket.hpp" |
| #include "SocketAddr.h" |
| |
| #ifdef __cplusplus |
| extern "C" { |
| #endif |
| |
| /* |
| * Prints transfer reports in default style |
| */ |
| void reporter_printstats( Transfer_Info *stats ) { |
| static char header_printed = 0; |
| |
| byte_snprintf( buffer, sizeof(buffer)/2, (double) stats->TotalLen, |
| toupper( stats->mFormat)); |
| byte_snprintf( &buffer[sizeof(buffer)/2], sizeof(buffer)/2, |
| stats->TotalLen / (stats->endTime - stats->startTime), |
| stats->mFormat); |
| |
| if ( stats->mUDP != (char)kMode_Server ) { |
| // TCP Reporting |
| if( !header_printed ) { |
| printf( report_bw_header); |
| header_printed = 1; |
| } |
| printf( report_bw_format, stats->transferID, |
| stats->startTime, stats->endTime, |
| buffer, &buffer[sizeof(buffer)/2] ); |
| } else { |
| // UDP Reporting |
| if( !header_printed ) { |
| printf( report_bw_jitter_loss_header); |
| header_printed = 1; |
| } |
| printf( report_bw_jitter_loss_format, stats->transferID, |
| stats->startTime, stats->endTime, |
| buffer, &buffer[sizeof(buffer)/2], |
| stats->jitter*1000.0, stats->cntError, stats->cntDatagrams, |
| (100.0 * stats->cntError) / stats->cntDatagrams ); |
| if ( stats->cntOutofOrder > 0 ) { |
| printf( report_outoforder, |
| stats->transferID, stats->startTime, |
| stats->endTime, stats->cntOutofOrder ); |
| } |
| } |
| if ( stats->free == 1 && stats->mUDP == (char)kMode_Client ) { |
| printf( report_datagrams, stats->transferID, stats->cntDatagrams ); |
| } |
| } |
| |
| |
| /* |
| * Prints multiple transfer reports in default style |
| */ |
| void reporter_multistats( Transfer_Info *stats ) { |
| |
| byte_snprintf( buffer, sizeof(buffer)/2, (double) stats->TotalLen, |
| toupper( stats->mFormat)); |
| byte_snprintf( &buffer[sizeof(buffer)/2], sizeof(buffer)/2, |
| stats->TotalLen / (stats->endTime - stats->startTime), |
| stats->mFormat); |
| |
| if ( stats->mUDP != (char)kMode_Server ) { |
| // TCP Reporting |
| printf( report_sum_bw_format, |
| stats->startTime, stats->endTime, |
| buffer, &buffer[sizeof(buffer)/2] ); |
| } else { |
| // UDP Reporting |
| printf( report_sum_bw_jitter_loss_format, |
| stats->startTime, stats->endTime, |
| buffer, &buffer[sizeof(buffer)/2], |
| stats->jitter*1000.0, stats->cntError, stats->cntDatagrams, |
| (100.0 * stats->cntError) / stats->cntDatagrams ); |
| if ( stats->cntOutofOrder > 0 ) { |
| printf( report_sum_outoforder, |
| stats->startTime, |
| stats->endTime, stats->cntOutofOrder ); |
| } |
| } |
| if ( stats->free == 1 && stats->mUDP == (char)kMode_Client ) { |
| printf( report_sum_datagrams, stats->cntDatagrams ); |
| } |
| } |
| |
| /* |
| * Prints server transfer reports in default style |
| */ |
| void reporter_serverstats( Connection_Info *nused, Transfer_Info *stats ) { |
| printf( server_reporting, stats->transferID ); |
| reporter_printstats( stats ); |
| } |
| |
| /* |
| * Report the client or listener Settings in default style |
| */ |
| void reporter_reportsettings( ReporterData *data ) { |
| int win, win_requested; |
| |
| win = getsock_tcp_windowsize( data->info.transferID, |
| (data->mThreadMode == kMode_Listener ? 0 : 1) ); |
| win_requested = data->mTCPWin; |
| |
| printf( separator_line ); |
| if ( data->mThreadMode == kMode_Listener ) { |
| printf( server_port, |
| (isUDP( data ) ? "UDP" : "TCP"), |
| data->mPort ); |
| } else { |
| printf( client_port, |
| data->mHost, |
| (isUDP( data ) ? "UDP" : "TCP"), |
| data->mPort ); |
| } |
| if ( data->mLocalhost != NULL ) { |
| printf( bind_address, data->mLocalhost ); |
| if ( SockAddr_isMulticast( &data->connection.local ) ) { |
| printf( join_multicast, data->mLocalhost ); |
| } |
| } |
| |
| if ( isUDP( data ) ) { |
| printf( (data->mThreadMode == kMode_Listener ? |
| server_datagram_size : client_datagram_size), |
| data->mBufLen ); |
| if ( SockAddr_isMulticast( &data->connection.peer ) ) { |
| printf( multicast_ttl, data->info.mTTL); |
| } |
| } |
| byte_snprintf( buffer, sizeof(buffer), win, |
| toupper( data->info.mFormat)); |
| printf( "%s: %s", (isUDP( data ) ? |
| udp_buffer_size : tcp_window_size), buffer ); |
| |
| if ( win_requested == 0 ) { |
| printf( " %s", window_default ); |
| } else if ( win != win_requested ) { |
| byte_snprintf( buffer, sizeof(buffer), win_requested, |
| toupper( data->info.mFormat)); |
| printf( warn_window_requested, buffer ); |
| } |
| printf( "\n" ); |
| printf( separator_line ); |
| } |
| |
| /* |
| * Report a socket's peer IP address in default style |
| */ |
| void *reporter_reportpeer( Connection_Info *stats, int ID ) { |
| if ( ID > 0 ) { |
| // copy the inet_ntop into temp buffers, to avoid overwriting |
| char local_addr[ REPORT_ADDRLEN ]; |
| char remote_addr[ REPORT_ADDRLEN ]; |
| struct sockaddr *local = ((struct sockaddr*)&stats->local); |
| struct sockaddr *peer = ((struct sockaddr*)&stats->peer); |
| |
| if ( local->sa_family == AF_INET ) { |
| inet_ntop( AF_INET, &((struct sockaddr_in*)local)->sin_addr, |
| local_addr, REPORT_ADDRLEN); |
| } |
| #ifdef HAVE_IPV6 |
| else { |
| inet_ntop( AF_INET6, &((struct sockaddr_in6*)local)->sin6_addr, |
| local_addr, REPORT_ADDRLEN); |
| } |
| #endif |
| |
| if ( peer->sa_family == AF_INET ) { |
| inet_ntop( AF_INET, &((struct sockaddr_in*)peer)->sin_addr, |
| remote_addr, REPORT_ADDRLEN); |
| } |
| #ifdef HAVE_IPV6 |
| else { |
| inet_ntop( AF_INET6, &((struct sockaddr_in6*)peer)->sin6_addr, |
| remote_addr, REPORT_ADDRLEN); |
| } |
| #endif |
| |
| printf( report_peer, |
| ID, |
| local_addr, ( local->sa_family == AF_INET ? |
| ntohs(((struct sockaddr_in*)local)->sin_port) : |
| #ifdef HAVE_IPV6 |
| ntohs(((struct sockaddr_in6*)local)->sin6_port)), |
| #else |
| 0), |
| #endif |
| remote_addr, ( peer->sa_family == AF_INET ? |
| ntohs(((struct sockaddr_in*)peer)->sin_port) : |
| #ifdef HAVE_IPV6 |
| ntohs(((struct sockaddr_in6*)peer)->sin6_port))); |
| #else |
| 0)); |
| #endif |
| } |
| return NULL; |
| } |
| // end ReportPeer |
| |
| /* ------------------------------------------------------------------- |
| * Report the MSS and MTU, given the MSS (or a guess thereof) |
| * ------------------------------------------------------------------- */ |
| |
| // compare the MSS against the (MTU - 40) to (MTU - 80) bytes. |
| // 40 byte IP header and somewhat arbitrarily, 40 more bytes of IP options. |
| |
| #define checkMSS_MTU( inMSS, inMTU ) (inMTU-40) >= inMSS && inMSS >= (inMTU-80) |
| |
| void reporter_reportMSS( int inMSS, thread_Settings *inSettings ) { |
| if ( inMSS <= 0 ) { |
| printf( report_mss_unsupported, inSettings->mSock ); |
| } else { |
| char* net; |
| int mtu = 0; |
| |
| if ( checkMSS_MTU( inMSS, 1500 ) ) { |
| net = "ethernet"; |
| mtu = 1500; |
| } else if ( checkMSS_MTU( inMSS, 4352 ) ) { |
| net = "FDDI"; |
| mtu = 4352; |
| } else if ( checkMSS_MTU( inMSS, 9180 ) ) { |
| net = "ATM"; |
| mtu = 9180; |
| } else if ( checkMSS_MTU( inMSS, 65280 ) ) { |
| net = "HIPPI"; |
| mtu = 65280; |
| } else if ( checkMSS_MTU( inMSS, 576 ) ) { |
| net = "minimum"; |
| mtu = 576; |
| printf( warn_no_pathmtu ); |
| } else { |
| mtu = inMSS + 40; |
| net = "unknown interface"; |
| } |
| |
| printf( report_mss, |
| inSettings->mSock, inMSS, mtu, net ); |
| } |
| } |
| // end ReportMSS |
| |
| #ifdef __cplusplus |
| } /* end extern "C" */ |
| #endif |