2009-10-24 21:15:04 +00:00
|
|
|
|
|
|
|
/*
|
2009-11-10 04:41:42 +00:00
|
|
|
* Copyright (c) 2009, The Regents of the University of California, through
|
2009-10-24 21:15:04 +00:00
|
|
|
* Lawrence Berkeley National Laboratory (subject to receipt of any required
|
|
|
|
* approvals from the U.S. Dept. of Energy). All rights reserved.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <getopt.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <signal.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <sys/socket.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <netinet/in.h>
|
|
|
|
#include <arpa/inet.h>
|
|
|
|
#include <netdb.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <netinet/tcp.h>
|
|
|
|
|
|
|
|
|
2009-11-02 22:43:19 +00:00
|
|
|
#include "iperf.h"
|
2009-10-24 21:15:04 +00:00
|
|
|
#include "iperf_api.h"
|
2009-11-02 22:43:19 +00:00
|
|
|
#include "iperf_server_api.h"
|
2009-10-24 21:15:04 +00:00
|
|
|
#include "units.h"
|
|
|
|
#include "locale.h"
|
|
|
|
|
2009-11-02 22:43:19 +00:00
|
|
|
int iperf_run(struct iperf_test * test);
|
2009-10-24 21:15:04 +00:00
|
|
|
|
|
|
|
/**************************************************************************/
|
|
|
|
|
|
|
|
static struct option longopts[] =
|
|
|
|
{
|
|
|
|
{"client", required_argument, NULL, 'c'},
|
|
|
|
{"server", no_argument, NULL, 's'},
|
|
|
|
{"time", required_argument, NULL, 't'},
|
|
|
|
{"port", required_argument, NULL, 'p'},
|
|
|
|
{"parallel", required_argument, NULL, 'P'},
|
|
|
|
{"udp", no_argument, NULL, 'u'},
|
|
|
|
{"tcpInfo", no_argument, NULL, 'T'},
|
|
|
|
{"bandwidth", required_argument, NULL, 'b'},
|
|
|
|
{"length", required_argument, NULL, 'l'},
|
|
|
|
{"window", required_argument, NULL, 'w'},
|
|
|
|
{"interval", required_argument, NULL, 'i'},
|
|
|
|
{"bytes", required_argument, NULL, 'n'},
|
|
|
|
{"NoDelay", no_argument, NULL, 'N'},
|
|
|
|
{"Print-mss", no_argument, NULL, 'm'},
|
|
|
|
{"Set-mss", required_argument, NULL, 'M'},
|
2009-11-13 13:59:07 +00:00
|
|
|
{"version", no_argument, NULL, 'v'},
|
2009-11-16 01:58:22 +00:00
|
|
|
{"verbose", no_argument, NULL, 'V'},
|
|
|
|
{"debug", no_argument, NULL, 'd'},
|
2009-10-24 21:15:04 +00:00
|
|
|
{"help", no_argument, NULL, 'h'},
|
2009-11-13 18:29:10 +00:00
|
|
|
#ifdef ADD_WHEN_SUPPORTED
|
|
|
|
{"tos", required_argument, NULL, 'S'},
|
|
|
|
{"linux-congestion", required_argument, NULL, 'Z'},
|
|
|
|
{"daemon", no_argument, NULL, 'D'},
|
|
|
|
#endif
|
2009-10-24 21:15:04 +00:00
|
|
|
{NULL, 0, NULL, 0}
|
|
|
|
};
|
|
|
|
|
|
|
|
/**************************************************************************/
|
|
|
|
|
|
|
|
int
|
|
|
|
main(int argc, char **argv)
|
|
|
|
{
|
|
|
|
|
|
|
|
char ch, role;
|
|
|
|
struct iperf_test *test;
|
2009-11-06 07:25:10 +00:00
|
|
|
int port = PORT;
|
2009-10-24 21:15:04 +00:00
|
|
|
|
|
|
|
#ifdef TEST_PROC_AFFINITY
|
|
|
|
/* didnt seem to work.... */
|
|
|
|
/*
|
|
|
|
* increasing the priority of the process to minimise packet generation
|
|
|
|
* delay
|
|
|
|
*/
|
|
|
|
int rc = setpriority(PRIO_PROCESS, 0, -15);
|
|
|
|
|
|
|
|
if (rc < 0)
|
|
|
|
{
|
|
|
|
perror("setpriority:");
|
|
|
|
printf("setting priority to valid level\n");
|
|
|
|
rc = setpriority(PRIO_PROCESS, 0, 0);
|
|
|
|
}
|
|
|
|
/* setting the affinity of the process */
|
|
|
|
cpu_set_t cpu_set;
|
|
|
|
int affinity = -1;
|
|
|
|
int ncores = 1;
|
|
|
|
|
|
|
|
sched_getaffinity(0, sizeof(cpu_set_t), &cpu_set);
|
|
|
|
if (errno)
|
|
|
|
perror("couldn't get affinity:");
|
|
|
|
|
|
|
|
if ((ncores = sysconf(_SC_NPROCESSORS_CONF)) <= 0)
|
|
|
|
err("sysconf: couldn't get _SC_NPROCESSORS_CONF");
|
|
|
|
|
|
|
|
CPU_ZERO(&cpu_set);
|
|
|
|
CPU_SET(affinity, &cpu_set);
|
|
|
|
if (sched_setaffinity(0, sizeof(cpu_set_t), &cpu_set) != 0)
|
|
|
|
err("couldn't change CPU affinity");
|
|
|
|
#endif
|
|
|
|
|
|
|
|
test = iperf_new_test();
|
|
|
|
iperf_defaults(test); /* sets defaults */
|
|
|
|
|
2009-11-16 01:58:22 +00:00
|
|
|
while ((ch = getopt_long(argc, argv, "c:p:st:uP:b:l:w:i:n:mNTvhVdM:f:", longopts, NULL)) != -1)
|
2009-10-24 21:15:04 +00:00
|
|
|
{
|
|
|
|
switch (ch)
|
|
|
|
{
|
|
|
|
case 'c':
|
|
|
|
test->role = 'c';
|
|
|
|
role = test->role;
|
2009-11-16 20:26:20 +00:00
|
|
|
test->server_hostname = (char *) malloc(strlen(optarg)+1);
|
2009-10-24 21:15:04 +00:00
|
|
|
strncpy(test->server_hostname, optarg, strlen(optarg));
|
|
|
|
break;
|
|
|
|
case 'p':
|
|
|
|
test->server_port = atoi(optarg);
|
|
|
|
port = test->server_port;
|
|
|
|
break;
|
|
|
|
case 's':
|
|
|
|
test->role = 's';
|
|
|
|
role = test->role;
|
|
|
|
break;
|
|
|
|
case 't':
|
|
|
|
test->duration = atoi(optarg);
|
2009-11-10 16:03:17 +00:00
|
|
|
if (test->duration > MAX_TIME)
|
|
|
|
{
|
2009-11-16 01:58:22 +00:00
|
|
|
fprintf(stderr, "\n Error: test duration too long. Maximum value = %d \n\n", (int)MAX_TIME);
|
2009-11-10 16:03:17 +00:00
|
|
|
fprintf(stderr, usage_long1);
|
|
|
|
fprintf(stderr, usage_long2);
|
|
|
|
exit(1);
|
|
|
|
}
|
2009-10-24 21:15:04 +00:00
|
|
|
break;
|
|
|
|
case 'u':
|
|
|
|
test->protocol = Pudp;
|
|
|
|
test->default_settings->blksize = DEFAULT_UDP_BLKSIZE;
|
|
|
|
test->new_stream = iperf_new_udp_stream;
|
|
|
|
break;
|
|
|
|
case 'P':
|
|
|
|
test->num_streams = atoi(optarg);
|
2009-11-10 16:03:17 +00:00
|
|
|
if (test->num_streams > MAX_STREAMS)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "\n Error: Number of parallel streams too large. Maximum value = %d \n\n", MAX_STREAMS);
|
|
|
|
fprintf(stderr, usage_long1);
|
|
|
|
fprintf(stderr, usage_long2);
|
|
|
|
exit(1);
|
|
|
|
}
|
2009-10-24 21:15:04 +00:00
|
|
|
break;
|
|
|
|
case 'b':
|
|
|
|
test->default_settings->rate = unit_atof(optarg);
|
|
|
|
break;
|
|
|
|
case 'l':
|
|
|
|
test->default_settings->blksize = unit_atoi(optarg);
|
2009-11-10 16:03:17 +00:00
|
|
|
if (test->default_settings->blksize > MAX_BLOCKSIZE)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "\n Error: Block size too large. Maximum value = %d \n\n", MAX_BLOCKSIZE);
|
|
|
|
fprintf(stderr, usage_long1);
|
|
|
|
fprintf(stderr, usage_long2);
|
|
|
|
exit(1);
|
|
|
|
}
|
2009-10-24 21:15:04 +00:00
|
|
|
break;
|
|
|
|
case 'w':
|
|
|
|
test->default_settings->socket_bufsize = unit_atof(optarg);
|
2009-11-10 16:03:17 +00:00
|
|
|
if (test->default_settings->socket_bufsize > MAX_TCP_BUFFER)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "\n Error: TCP buffer too large. Maximum value = %d \n\n", MAX_TCP_BUFFER);
|
|
|
|
fprintf(stderr, usage_long1);
|
|
|
|
fprintf(stderr, usage_long2);
|
|
|
|
exit(1);
|
|
|
|
}
|
2009-10-24 21:15:04 +00:00
|
|
|
break;
|
|
|
|
case 'i':
|
2009-11-13 20:14:06 +00:00
|
|
|
/* could potentially want separate stat collection and reporting intervals,
|
|
|
|
but just set them to be the same for now */
|
2009-10-24 21:15:04 +00:00
|
|
|
test->stats_interval = atoi(optarg);
|
|
|
|
test->reporter_interval = atoi(optarg);
|
2009-11-10 16:03:17 +00:00
|
|
|
if (test->stats_interval > MAX_INTERVAL)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "\n Error: Report interval too large. Maximum value = %d \n\n", MAX_INTERVAL);
|
|
|
|
fprintf(stderr, usage_long1);
|
|
|
|
fprintf(stderr, usage_long2);
|
|
|
|
exit(1);
|
|
|
|
}
|
2009-10-24 21:15:04 +00:00
|
|
|
break;
|
|
|
|
case 'n':
|
|
|
|
test->default_settings->bytes = unit_atoi(optarg);
|
|
|
|
printf("total bytes to be transferred = %llu\n", test->default_settings->bytes);
|
|
|
|
break;
|
|
|
|
case 'm':
|
|
|
|
test->print_mss = 1;
|
|
|
|
break;
|
|
|
|
case 'N':
|
|
|
|
test->no_delay = 1;
|
|
|
|
break;
|
|
|
|
case 'M':
|
|
|
|
test->default_settings->mss = atoi(optarg);
|
2009-11-10 16:03:17 +00:00
|
|
|
if (test->default_settings->mss > MAX_MSS)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "\n Error: MSS too large. Maximum value = %d \n\n", MAX_MSS);
|
|
|
|
fprintf(stderr, usage_long1);
|
|
|
|
fprintf(stderr, usage_long2);
|
|
|
|
exit(1);
|
|
|
|
}
|
2009-10-24 21:15:04 +00:00
|
|
|
break;
|
|
|
|
case 'f':
|
|
|
|
test->default_settings->unit_format = *optarg;
|
|
|
|
break;
|
|
|
|
case 'T':
|
|
|
|
test->tcp_info = 1;
|
|
|
|
break;
|
2009-11-16 01:58:22 +00:00
|
|
|
case 'V':
|
|
|
|
test->verbose = 1;
|
|
|
|
break;
|
|
|
|
case 'd':
|
|
|
|
test->debug = 1;
|
|
|
|
break;
|
2009-11-10 17:05:44 +00:00
|
|
|
case 'v': // print version and exit
|
|
|
|
fprintf( stderr, version );
|
|
|
|
exit(1);
|
2009-10-24 21:15:04 +00:00
|
|
|
case 'h':
|
|
|
|
default:
|
|
|
|
fprintf(stderr, usage_long1);
|
|
|
|
fprintf(stderr, usage_long2);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-11-10 16:03:17 +00:00
|
|
|
/* exit until this is done.... */
|
2009-11-02 22:43:19 +00:00
|
|
|
if (test->protocol == Pudp) {
|
|
|
|
printf("UDP mode not yet supported. Exiting. \n");
|
|
|
|
exit(0);
|
|
|
|
}
|
2009-10-24 21:15:04 +00:00
|
|
|
|
|
|
|
//printf("in main: calling iperf_init_test \n");
|
2009-10-27 03:51:04 +00:00
|
|
|
if (test->role == 'c')
|
|
|
|
printf("Connecting to port %d on host %s \n", test->server_port, test->server_hostname);
|
2009-10-24 21:15:04 +00:00
|
|
|
iperf_init_test(test);
|
|
|
|
|
|
|
|
if (test->role == 'c') /* if client, send params to server */
|
|
|
|
{
|
|
|
|
exchange_parameters(test);
|
|
|
|
test->streams->settings->state = STREAM_BEGIN;
|
|
|
|
}
|
2009-11-06 02:19:20 +00:00
|
|
|
|
2009-10-24 21:15:04 +00:00
|
|
|
//printf("in main: calling iperf_run \n");
|
|
|
|
iperf_run(test);
|
|
|
|
iperf_free_test(test);
|
|
|
|
|
|
|
|
printf("\niperf Done.\n");
|
|
|
|
exit(0);
|
|
|
|
}
|
2009-11-02 22:43:19 +00:00
|
|
|
|
|
|
|
/**************************************************************************/
|
|
|
|
int
|
|
|
|
iperf_run(struct iperf_test * test)
|
|
|
|
{
|
2009-11-06 02:19:20 +00:00
|
|
|
|
2009-11-02 22:43:19 +00:00
|
|
|
test->default_settings->state = TEST_RUNNING;
|
|
|
|
|
|
|
|
switch (test->role)
|
|
|
|
{
|
|
|
|
case 's':
|
2009-11-02 23:56:55 +00:00
|
|
|
while (1)
|
|
|
|
{
|
|
|
|
iperf_run_server(test);
|
2009-11-06 02:19:20 +00:00
|
|
|
test->streams = NULL;
|
2009-11-02 23:56:55 +00:00
|
|
|
sleep(1);
|
|
|
|
}
|
2009-11-02 22:43:19 +00:00
|
|
|
return 0;
|
|
|
|
case 'c':
|
|
|
|
iperf_run_client(test);
|
|
|
|
return 0;
|
|
|
|
default:
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|