VirtualBox

source: vbox/trunk/src/VBox/ValidationKit/utils/network/NetPerf.cpp@ 68395

Last change on this file since 68395 was 64366, checked in by vboxsync, 8 years ago

ValidationKit: Doxygen

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 64.4 KB
Line 
1/* $Id: NetPerf.cpp 64366 2016-10-22 16:36:39Z vboxsync $ */
2/** @file
3 * NetPerf - Network Performance Benchmark.
4 */
5
6/*
7 * Copyright (C) 2010-2016 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * The contents of this file may alternatively be used under the terms
18 * of the Common Development and Distribution License Version 1.0
19 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
20 * VirtualBox OSE distribution, in which case the provisions of the
21 * CDDL are applicable instead of those of the GPL.
22 *
23 * You may elect to license modified versions of this file under the
24 * terms and conditions of either the GPL or the CDDL or both.
25 */
26
27
28/*********************************************************************************************************************************
29* Header Files *
30*********************************************************************************************************************************/
31#include <iprt/asm.h>
32#include <iprt/assert.h>
33#include <iprt/ctype.h>
34#include <iprt/err.h>
35#include <iprt/getopt.h>
36#include <iprt/initterm.h>
37#include <iprt/mem.h>
38#include <iprt/message.h>
39#include <iprt/path.h>
40#include <iprt/param.h>
41#include <iprt/process.h>
42#include <iprt/rand.h>
43#include <iprt/stream.h>
44#include <iprt/string.h>
45#include <iprt/tcp.h>
46#include <iprt/thread.h>
47#include <iprt/test.h>
48#include <iprt/time.h>
49#include <iprt/timer.h>
50
51
52/*********************************************************************************************************************************
53* Defined Constants And Macros *
54*********************************************************************************************************************************/
55/** Default TCP port (update help text if you change this) */
56#define NETPERF_DEFAULT_PORT 5002
57
58/** Default TCP packet size (bytes) */
59#define NETPERF_DEFAULT_PKT_SIZE_THROUGHPUT 8192
60/** Default TCP packet size (bytes) */
61#define NETPERF_DEFAULT_PKT_SIZE_LATENCY 1024
62/** Maximum packet size possible (bytes). */
63#define NETPERF_MAX_PKT_SIZE _1M
64/** Minimum packet size possible (bytes). */
65#define NETPERF_MIN_PKT_SIZE sizeof(NETPERFHDR)
66
67/** Default timeout in (seconds) */
68#define NETPERF_DEFAULT_TIMEOUT 10
69/** Maximum timeout possible (seconds). */
70#define NETPERF_MAX_TIMEOUT 3600 /* 1h */
71/** Minimum timeout possible (seconds). */
72#define NETPERF_MIN_TIMEOUT 1
73
74/** The default warmup time (ms). */
75#define NETPERF_DEFAULT_WARMUP 1000 /* 1s */
76/** The maxium warmup time (ms). */
77#define NETPERF_MAX_WARMUP 60000 /* 60s */
78/** The minimum warmup time (ms). */
79#define NETPERF_MIN_WARMUP 1000 /* 1s */
80
81/** The default cool down time (ms). */
82#define NETPERF_DEFAULT_COOL_DOWN 1000 /* 1s */
83/** The maxium cool down time (ms). */
84#define NETPERF_MAX_COOL_DOWN 60000 /* 60s */
85/** The minimum cool down time (ms). */
86#define NETPERF_MIN_COOL_DOWN 1000 /* 1s */
87
88/** The length of the length prefix used when submitting parameters and
89 * results. */
90#define NETPERF_LEN_PREFIX 4
91
92
93/*********************************************************************************************************************************
94* Structures and Typedefs *
95*********************************************************************************************************************************/
96typedef enum NETPERFPROTO
97{
98 NETPERFPROTO_INVALID = 0,
99 NETPERFPROTO_TCP
100 //NETPERFPROTO_UDP
101} NETPERFPROTO;
102
103/**
104 * What kind of test we're performing.
105 */
106typedef enum NETPERFMODE
107{
108 NETPERFMODE_INVALID = 0,
109 /** Latency of a symmetric packet exchange. */
110 NETPERFMODE_LATENCY,
111 /** Separate throughput measurements for each direction. */
112 NETPERFMODE_THROUGHPUT,
113 /** Transmit throughput. */
114 NETPERFMODE_THROUGHPUT_XMIT,
115 /** Transmit throughput. */
116 NETPERFMODE_THROUGHPUT_RECV
117} NETPERFMODE;
118
119/**
120 * Statistics.
121 */
122typedef struct NETPERFSTATS
123{
124 uint64_t cTx;
125 uint64_t cRx;
126 uint64_t cEchos;
127 uint64_t cErrors;
128 uint64_t cNsElapsed;
129} NETPERFSTATS;
130
131/**
132 * Settings & a little bit of state.
133 */
134typedef struct NETPERFPARAMS
135{
136 /** @name Static settings
137 * @{ */
138 /** The TCP port number. */
139 uint32_t uPort;
140 /** Client: Use server statistcs. */
141 bool fServerStats;
142 /** Server: Quit after the first client. */
143 bool fSingleClient;
144 /** @} */
145
146 /** @name Dynamic settings
147 * @{ */
148 /** Disable send packet coalescing. */
149 bool fNoDelay;
150 /** Detect broken payloads. */
151 bool fCheckData;
152 /** The test mode. */
153 NETPERFMODE enmMode;
154 /** The number of seconds to run each of the test steps. */
155 uint32_t cSecTimeout;
156 /** Number of millisecond to spend warning up before testing. */
157 uint32_t cMsWarmup;
158 /** Number of millisecond to spend cooling down after the testing. */
159 uint32_t cMsCoolDown;
160 /** The packet size. */
161 uint32_t cbPacket;
162 /** @} */
163
164 /** @name State
165 * @{ */
166 RTSOCKET hSocket;
167 /** @} */
168} NETPERFPARAMS;
169
170/**
171 * Packet header used in tests.
172 *
173 * Need to indicate when we've timed out and it's time to reverse the roles or
174 * stop testing.
175 */
176typedef struct NETPERFHDR
177{
178 /** Magic value (little endian). */
179 uint32_t u32Magic;
180 /** State value. */
181 uint32_t u32State;
182 /** Sequence number (little endian). */
183 uint32_t u32Seq;
184 /** Reserved, must be zero. */
185 uint32_t u32Reserved;
186} NETPERFHDR;
187
188/** Magic value for NETPERFHDR::u32Magic. */
189#define NETPERFHDR_MAGIC UINT32_C(0xfeedf00d)
190
191/** @name Packet State (NETPERF::u32Magic)
192 * @{ */
193/** Warm up. */
194#define NETPERFHDR_WARMUP UINT32_C(0x0c0ffe01)
195/** The clock is running. */
196#define NETPERFHDR_TESTING UINT32_C(0x0c0ffe02)
197/** Stop the clock but continue the package flow. */
198#define NETPERFHDR_COOL_DOWN UINT32_C(0x0c0ffe03)
199/** Done, stop the clock if not done already and reply with results. */
200#define NETPERFHDR_DONE UINT32_C(0x0c0ffe04)
201/** @} */
202
203
204/*********************************************************************************************************************************
205* Global Variables *
206*********************************************************************************************************************************/
207/** Connection start/identifier to make sure other end is NetPerf. */
208static const char g_ConnectStart[] = "yo! waaazzzzzaaaaup dude?";
209/** Start of parameters proposal made by the client. */
210static const char g_szStartParams[] = "deal?";
211/** All okay to start test */
212static const char g_szAck[] = "okay!";
213/** Negative. */
214static const char g_szNegative[] = "nope!";
215AssertCompile(sizeof(g_szAck) == sizeof(g_szNegative));
216/** Start of statistics. */
217static const char g_szStartStats[] = "dude, stats";
218
219/** Command line parameters */
220static const RTGETOPTDEF g_aCmdOptions[] =
221{
222 { "--server", 's', RTGETOPT_REQ_NOTHING },
223 { "--client", 'c', RTGETOPT_REQ_STRING },
224 { "--interval", 'i', RTGETOPT_REQ_UINT32 },
225 { "--port", 'p', RTGETOPT_REQ_UINT32 },
226 { "--len", 'l', RTGETOPT_REQ_UINT32 },
227 { "--nodelay", 'N', RTGETOPT_REQ_NOTHING },
228 { "--mode", 'm', RTGETOPT_REQ_STRING },
229 { "--warmpup", 'w', RTGETOPT_REQ_UINT32 },
230 { "--cool-down", 'W', RTGETOPT_REQ_UINT32 },
231 { "--server-stats", 'S', RTGETOPT_REQ_NOTHING },
232 { "--single-client", '1', RTGETOPT_REQ_NOTHING },
233 { "--daemonize", 'd', RTGETOPT_REQ_NOTHING },
234 { "--daemonized", 'D', RTGETOPT_REQ_NOTHING },
235 { "--check-data", 'C', RTGETOPT_REQ_NOTHING },
236 { "--help", 'h', RTGETOPT_REQ_NOTHING } /* for Usage() */
237};
238
239/** The test handle. */
240static RTTEST g_hTest;
241
242
243
244static void Usage(PRTSTREAM pStrm)
245{
246 char szExec[RTPATH_MAX];
247 RTStrmPrintf(pStrm, "usage: %s <-s|-c <host>> [options]\n",
248 RTPathFilename(RTProcGetExecutablePath(szExec, sizeof(szExec))));
249 RTStrmPrintf(pStrm, "\n");
250 RTStrmPrintf(pStrm, "options: \n");
251
252
253 for (unsigned i = 0; i < RT_ELEMENTS(g_aCmdOptions); i++)
254 {
255 const char *pszHelp;
256 switch (g_aCmdOptions[i].iShort)
257 {
258 case 'h':
259 pszHelp = "Displays this help and exit";
260 break;
261 case 's':
262 pszHelp = "Run in server mode, waiting for clients (default)";
263 break;
264 case 'c':
265 pszHelp = "Run in client mode, connecting to <host>";
266 break;
267 case 'i':
268 pszHelp = "Interval in seconds to run the test (default " RT_XSTR(NETPERF_DEFAULT_TIMEOUT) " s)";
269 break;
270 case 'p':
271 pszHelp = "Server port to listen/connect to (default " RT_XSTR(NETPERF_DEFAULT_PORT) ")";
272 break;
273 case 'l':
274 pszHelp = "Packet size in bytes (defaults to " RT_XSTR(NETPERF_DEFAULT_PKT_SIZE_LATENCY)
275 " for latency and " RT_XSTR(NETPERF_DEFAULT_PKT_SIZE_THROUGHPUT) " for throughput)";
276 break;
277 case 'm':
278 pszHelp = "Test mode: latency (default), throughput, throughput-xmit or throughput-recv";
279 break;
280 case 'N':
281 pszHelp = "Set TCP no delay, disabling Nagle's algorithm";
282 break;
283 case 'S':
284 pszHelp = "Report server stats, ignored if server";
285 break;
286 case '1':
287 pszHelp = "Stop the server after the first client";
288 break;
289 case 'd':
290 pszHelp = "Daemonize if server, ignored if client";
291 break;
292 case 'D':
293 continue; /* internal */
294 case 'w':
295 pszHelp = "Warmup time, in milliseconds (default " RT_XSTR(NETPERF_DEFAULT_WARMPUP) " ms)";
296 break;
297 case 'W':
298 pszHelp = "Cool down time, in milliseconds (default " RT_XSTR(NETPERF_DEFAULT_COOL_DOWN) " ms)";
299 break;
300 case 'C':
301 pszHelp = "Check payload data at the receiving end";
302 break;
303 default:
304 pszHelp = "Option undocumented";
305 break;
306 }
307 char szOpt[256];
308 RTStrPrintf(szOpt, sizeof(szOpt), "%s, -%c", g_aCmdOptions[i].pszLong, g_aCmdOptions[i].iShort);
309 RTStrmPrintf(pStrm, " %-20s%s\n", szOpt, pszHelp);
310 }
311}
312
313/**
314 * Timer callback employed to set the stop indicator.
315 *
316 * This is used both by the client and server side.
317 *
318 * @param hTimer The timer, ignored.
319 * @param pvUser Pointer to the stop variable.
320 * @param iTick The tick, ignored.
321 */
322static DECLCALLBACK(void) netperfStopTimerCallback(RTTIMERLR hTimer, void *pvUser, uint64_t iTick)
323{
324 bool volatile *pfStop = (bool volatile *)pvUser;
325/* RTPrintf("Time's Up!\n");*/
326 ASMAtomicWriteBool(pfStop, true);
327 NOREF(hTimer); NOREF(iTick);
328}
329
330/**
331 * Sends a statistics packet to our peer.
332 *
333 * @returns IPRT status code.
334 * @param pStats The stats to send.
335 * @param hSocket The TCP socket to send them to.
336 */
337static int netperfSendStats(NETPERFSTATS const *pStats, RTSOCKET hSocket)
338{
339 char szBuf[256 + NETPERF_LEN_PREFIX];
340 size_t cch = RTStrPrintf(&szBuf[NETPERF_LEN_PREFIX], sizeof(szBuf) - NETPERF_LEN_PREFIX,
341 "%s:%llu:%llu:%llu:%llu:%llu",
342 g_szStartStats,
343 pStats->cTx,
344 pStats->cRx,
345 pStats->cEchos,
346 pStats->cErrors,
347 pStats->cNsElapsed);
348
349 RTStrPrintf(szBuf, NETPERF_LEN_PREFIX + 1, "%0*u", NETPERF_LEN_PREFIX, cch);
350 szBuf[NETPERF_LEN_PREFIX] = g_szStartStats[0];
351 Assert(strlen(szBuf) == cch + NETPERF_LEN_PREFIX);
352
353 int rc = RTTcpWrite(hSocket, szBuf, cch + NETPERF_LEN_PREFIX);
354 if (RT_FAILURE(rc))
355 return RTTestIFailedRc(rc, "stats: Failed to send stats: %Rrc\n", rc);
356
357 /*
358 * Wait for ACK.
359 */
360 rc = RTTcpRead(hSocket, szBuf, sizeof(g_szAck) - 1, NULL);
361 if (RT_FAILURE(rc))
362 return RTTestIFailedRc(rc, "stats: failed to write stats: %Rrc\n", rc);
363 szBuf[sizeof(g_szAck) - 1] = '\0';
364 if (!strcmp(szBuf, g_szNegative))
365 return RTTestIFailedRc(rc, "stats: client failed to parse them\n");
366 if (strcmp(szBuf, g_szAck))
367 return RTTestIFailedRc(rc, "stats: got '%s' in instead of ack/nack\n", szBuf);
368
369 return VINF_SUCCESS;
370}
371
372/**
373 * Receives a statistics packet from our peer.
374 *
375 * @returns IPRT status code. Error signalled.
376 * @param pStats Where to receive the stats.
377 * @param hSocket The TCP socket to recevie them from.
378 */
379static int netperfRecvStats(NETPERFSTATS *pStats, RTSOCKET hSocket)
380{
381 /*
382 * Read the stats message.
383 */
384 /* the length prefix */
385 char szBuf[256 + NETPERF_LEN_PREFIX];
386 int rc = RTTcpRead(hSocket, szBuf, NETPERF_LEN_PREFIX, NULL);
387 if (RT_FAILURE(rc))
388 return RTTestIFailedRc(rc, "stats: failed to read stats prefix: %Rrc\n", rc);
389
390 szBuf[NETPERF_LEN_PREFIX] = '\0';
391 uint32_t cch;
392 rc = RTStrToUInt32Full(szBuf, 10, &cch);
393 if (rc != VINF_SUCCESS)
394 return RTTestIFailedRc(RT_SUCCESS(rc) ? -rc : rc, "stats: bad stat length prefix: '%s' - %Rrc\n", szBuf, rc);
395 if (cch >= sizeof(szBuf))
396 return RTTestIFailedRc(VERR_BUFFER_OVERFLOW, "stats: too large: %u bytes\n", cch);
397
398 /* the actual message */
399 rc = RTTcpRead(hSocket, szBuf, cch, NULL);
400 if (RT_FAILURE(rc))
401 return RTTestIFailedRc(rc, "failed to read stats: %Rrc\n", rc);
402 szBuf[cch] = '\0';
403
404 /*
405 * Validate the message header.
406 */
407 if ( strncmp(szBuf, g_szStartStats, sizeof(g_szStartStats) - 1)
408 || szBuf[sizeof(g_szStartStats) - 1] != ':')
409 return RTTestIFailedRc(VERR_NET_PROTOCOL_ERROR, "stats: invalid packet start: '%s'\n", szBuf);
410 char *pszCur = &szBuf[sizeof(g_szStartStats)];
411
412 /*
413 * Parse it.
414 */
415 static const char * const s_apszNames[] =
416 {
417 "cTx", "cRx", "cEchos", "cErrors", "cNsElapsed"
418 };
419 uint64_t *apu64[RT_ELEMENTS(s_apszNames)] =
420 {
421 &pStats->cTx,
422 &pStats->cRx,
423 &pStats->cEchos,
424 &pStats->cErrors,
425 &pStats->cNsElapsed
426 };
427
428 for (unsigned i = 0; i < RT_ELEMENTS(apu64); i++)
429 {
430 if (!pszCur)
431 return RTTestIFailedRc(VERR_PARSE_ERROR, "stats: missing %s\n", s_apszNames[i]);
432
433 char *pszNext = strchr(pszCur, ':');
434 if (pszNext)
435 *pszNext++ = '\0';
436 rc = RTStrToUInt64Full(pszCur, 10, apu64[i]);
437 if (rc != VINF_SUCCESS)
438 return RTTestIFailedRc(RT_SUCCESS(rc) ? -rc : rc, "stats: bad value for %s: '%s' - %Rrc\n",
439 s_apszNames[i], pszCur, rc);
440
441 pszCur = pszNext;
442 }
443
444 if (pszCur)
445 return RTTestIFailedRc(VERR_PARSE_ERROR, "stats: Unparsed data: '%s'\n", pszCur);
446
447 /*
448 * Send ACK.
449 */
450 rc = RTTcpWrite(hSocket, g_szAck, sizeof(g_szAck) - 1);
451 if (RT_FAILURE(rc))
452 return RTTestIFailedRc(rc, "stats: failed to write ack: %Rrc\n", rc);
453
454 return VINF_SUCCESS;
455}
456
457/**
458 * TCP Throughput: Print the statistics.
459 *
460 * @param pSendStats Send stats.
461 * @param pRecvStats Receive stats.
462 * @param cbPacket Packet size.
463 */
464static void netperfPrintThroughputStats(NETPERFSTATS const *pSendStats, NETPERFSTATS const *pRecvStats, uint32_t cbPacket)
465{
466 RTTestIValue("Packet size", cbPacket, RTTESTUNIT_BYTES);
467
468 if (pSendStats)
469 {
470 double rdSecElapsed = (double)pSendStats->cNsElapsed / 1000000000.0;
471 RTTestIValue("Sends", pSendStats->cTx, RTTESTUNIT_PACKETS);
472 RTTestIValue("Send Interval", pSendStats->cNsElapsed, RTTESTUNIT_NS);
473 RTTestIValue("Send Throughput", (uint64_t)(cbPacket * pSendStats->cTx / rdSecElapsed), RTTESTUNIT_BYTES_PER_SEC);
474 RTTestIValue("Send Rate", (uint64_t)(pSendStats->cTx / rdSecElapsed), RTTESTUNIT_PACKETS_PER_SEC);
475 RTTestIValue("Send Latency", (uint64_t)(rdSecElapsed / pSendStats->cTx * 1000000000.0), RTTESTUNIT_NS_PER_PACKET);
476 }
477
478 if (pRecvStats)
479 {
480 double rdSecElapsed = (double)pRecvStats->cNsElapsed / 1000000000.0;
481 RTTestIValue("Receives", pRecvStats->cRx, RTTESTUNIT_PACKETS);
482 RTTestIValue("Receive Interval", pRecvStats->cNsElapsed, RTTESTUNIT_NS);
483 RTTestIValue("Receive Throughput", (uint64_t)(cbPacket * pRecvStats->cRx / rdSecElapsed), RTTESTUNIT_BYTES_PER_SEC);
484 RTTestIValue("Receive Rate", (uint64_t)(pRecvStats->cRx / rdSecElapsed), RTTESTUNIT_PACKETS_PER_SEC);
485 RTTestIValue("Receive Latency", (uint64_t)(rdSecElapsed / pRecvStats->cRx * 1000000000.0), RTTESTUNIT_NS_PER_PACKET);
486 }
487}
488
489/**
490 * TCP Throughput: Send data to the other party.
491 *
492 * @returns IPRT status code.
493 * @param pParams The TCP parameters block.
494 * @param pBuf The buffer we're using when sending.
495 * @param pSendStats Where to return the statistics.
496 */
497static int netperfTCPThroughputSend(NETPERFPARAMS const *pParams, NETPERFHDR *pBuf, NETPERFSTATS *pSendStats)
498{
499 RT_ZERO(*pSendStats);
500
501 /*
502 * Create the timer
503 */
504 RTTIMERLR hTimer;
505 bool volatile fStop = false;
506 int rc = RTTimerLRCreateEx(&hTimer, 0 /* nsec */, RTTIMER_FLAGS_CPU_ANY, netperfStopTimerCallback, (void *)&fStop);
507 if (RT_SUCCESS(rc))
508 {
509 uint32_t u32Seq = 0;
510
511 RT_BZERO(pBuf, pParams->cbPacket);
512 pBuf->u32Magic = RT_H2LE_U32_C(NETPERFHDR_MAGIC);
513 pBuf->u32State = 0;
514 pBuf->u32Seq = 0;
515 pBuf->u32Reserved = 0;
516
517 /*
518 * Warm up.
519 */
520 pBuf->u32State = RT_H2LE_U32_C(NETPERFHDR_WARMUP);
521 rc = RTTimerLRStart(hTimer, pParams->cMsWarmup * UINT64_C(1000000) /* nsec */);
522 if (RT_SUCCESS(rc))
523 {
524 while (!fStop)
525 {
526 u32Seq++;
527 pBuf->u32Seq = RT_H2LE_U32(u32Seq);
528 rc = RTTcpWrite(pParams->hSocket, pBuf, pParams->cbPacket);
529 if (RT_FAILURE(rc))
530 {
531 RTTestIFailed("RTTcpWrite/warmup: %Rrc\n", rc);
532 break;
533 }
534 }
535 }
536 else
537 RTTestIFailed("RTTimerLRStart/warmup: %Rrc\n", rc);
538
539 /*
540 * The real thing.
541 */
542 if (RT_SUCCESS(rc))
543 {
544 pBuf->u32State = RT_H2LE_U32_C(NETPERFHDR_TESTING);
545 fStop = false;
546 rc = RTTimerLRStart(hTimer, pParams->cSecTimeout * UINT64_C(1000000000) /* nsec */);
547 if (RT_SUCCESS(rc))
548 {
549 uint64_t u64StartTS = RTTimeNanoTS();
550 while (!fStop)
551 {
552 u32Seq++;
553 pBuf->u32Seq = RT_H2LE_U32(u32Seq);
554 rc = RTTcpWrite(pParams->hSocket, pBuf, pParams->cbPacket);
555 if (RT_FAILURE(rc))
556 {
557 RTTestIFailed("RTTcpWrite/testing: %Rrc\n", rc);
558 break;
559 }
560 pSendStats->cTx++;
561 }
562 pSendStats->cNsElapsed = RTTimeNanoTS() - u64StartTS;
563 }
564 else
565 RTTestIFailed("RTTimerLRStart/testing: %Rrc\n", rc);
566 }
567
568 /*
569 * Cool down.
570 */
571 if (RT_SUCCESS(rc))
572 {
573 pBuf->u32State = RT_H2LE_U32_C(NETPERFHDR_COOL_DOWN);
574 fStop = false;
575 rc = RTTimerLRStart(hTimer, pParams->cMsCoolDown * UINT64_C(1000000) /* nsec */);
576 if (RT_SUCCESS(rc))
577 {
578 while (!fStop)
579 {
580 u32Seq++;
581 pBuf->u32Seq = RT_H2LE_U32(u32Seq);
582 rc = RTTcpWrite(pParams->hSocket, pBuf, pParams->cbPacket);
583 if (RT_FAILURE(rc))
584 {
585 RTTestIFailed("RTTcpWrite/cool down: %Rrc\n", rc);
586 break;
587 }
588 }
589 }
590 else
591 RTTestIFailed("RTTimerLRStart/testing: %Rrc\n", rc);
592 }
593
594 /*
595 * Send DONE packet.
596 */
597 if (RT_SUCCESS(rc))
598 {
599 u32Seq++;
600 pBuf->u32Seq = RT_H2LE_U32(u32Seq);
601 pBuf->u32State = RT_H2LE_U32_C(NETPERFHDR_DONE);
602 rc = RTTcpWrite(pParams->hSocket, pBuf, pParams->cbPacket);
603 if (RT_FAILURE(rc))
604 RTTestIFailed("RTTcpWrite/done: %Rrc\n", rc);
605 }
606
607 RTTimerLRDestroy(hTimer);
608 }
609 else
610 RTTestIFailed("Failed to create timer object: %Rrc\n", rc);
611 return rc;
612}
613
614
615/**
616 * TCP Throughput: Receive data from the other party.
617 *
618 * @returns IPRT status code.
619 * @param pParams The TCP parameters block.
620 * @param pBuf The buffer we're using when sending.
621 * @param pStats Where to return the statistics.
622 */
623static int netperfTCPThroughputRecv(NETPERFPARAMS const *pParams, NETPERFHDR *pBuf, NETPERFSTATS *pStats)
624{
625 RT_ZERO(*pStats);
626
627 int rc;
628 uint32_t u32Seq = 0;
629 uint64_t cRx = 0;
630 uint64_t u64StartTS = 0;
631 uint32_t uState = RT_H2LE_U32_C(NETPERFHDR_WARMUP);
632
633 for (;;)
634 {
635 rc = RTTcpRead(pParams->hSocket, pBuf, pParams->cbPacket, NULL);
636 if (RT_FAILURE(rc))
637 {
638 pStats->cErrors++;
639 RTTestIFailed("RTTcpRead failed: %Rrc\n", rc);
640 break;
641 }
642 if (RT_UNLIKELY( pBuf->u32Magic != RT_H2LE_U32_C(NETPERFHDR_MAGIC)
643 || pBuf->u32Reserved != 0))
644 {
645 pStats->cErrors++;
646 RTTestIFailed("Invalid magic or reserved field value: %#x %#x\n", RT_H2LE_U32(pBuf->u32Magic), RT_H2LE_U32(pBuf->u32Reserved));
647 rc = VERR_INVALID_MAGIC;
648 break;
649 }
650
651 u32Seq += 1;
652 if (RT_UNLIKELY(pBuf->u32Seq != RT_H2LE_U32(u32Seq)))
653 {
654 pStats->cErrors++;
655 RTTestIFailed("Out of sequence: got %#x, expected %#x\n", RT_H2LE_U32(pBuf->u32Seq), u32Seq);
656 rc = VERR_WRONG_ORDER;
657 break;
658 }
659
660 if (pParams->fCheckData && uState == RT_H2LE_U32_C(NETPERFHDR_TESTING))
661 {
662 unsigned i = sizeof(NETPERFHDR);
663 for (;i < pParams->cbPacket; ++i)
664 if (((unsigned char *)pBuf)[i])
665 break;
666 if (i != pParams->cbPacket)
667 {
668 pStats->cErrors++;
669 RTTestIFailed("Broken payload: at %#x got %#x, expected %#x\n", i, ((unsigned char *)pBuf)[i], 0);
670 rc = VERR_NOT_EQUAL;
671 break;
672 }
673 }
674 if (RT_LIKELY(pBuf->u32State == uState))
675 cRx++;
676 /*
677 * Validate and act on switch state.
678 */
679 else if ( uState == RT_H2LE_U32_C(NETPERFHDR_WARMUP)
680 && pBuf->u32State == RT_H2LE_U32_C(NETPERFHDR_TESTING))
681 {
682 cRx = 0;
683 u64StartTS = RTTimeNanoTS();
684 uState = pBuf->u32State;
685 }
686 else if ( uState == RT_H2LE_U32_C(NETPERFHDR_TESTING)
687 && ( pBuf->u32State == RT_H2LE_U32_C(NETPERFHDR_COOL_DOWN)
688 || pBuf->u32State == RT_H2LE_U32_C(NETPERFHDR_DONE)) )
689 {
690 pStats->cNsElapsed = RTTimeNanoTS() - u64StartTS;
691 pStats->cRx = cRx + 1;
692 uState = pBuf->u32State;
693 if (uState == RT_H2LE_U32_C(NETPERFHDR_DONE))
694 break;
695 }
696 else if ( uState == RT_H2LE_U32_C(NETPERFHDR_COOL_DOWN)
697 && pBuf->u32State == RT_H2LE_U32_C(NETPERFHDR_DONE))
698 {
699 uState = pBuf->u32State;
700 break;
701 }
702 else
703 {
704 pStats->cErrors++;
705 RTTestIFailed("Protocol error: invalid state transition %#x -> %#x\n",
706 RT_LE2H_U32(uState), RT_LE2H_U32(pBuf->u32State));
707 rc = VERR_INVALID_MAGIC;
708 break;
709 }
710 }
711
712 AssertReturn(uState == RT_H2LE_U32_C(NETPERFHDR_DONE) || RT_FAILURE(rc), VERR_INVALID_STATE);
713 return rc;
714}
715
716
717/**
718 * Prints the statistics for the latency test.
719 *
720 * @param pStats The statistics.
721 * @param cbPacket The packet size in bytes.
722 */
723static void netperfPrintLatencyStats(NETPERFSTATS const *pStats, uint32_t cbPacket)
724{
725 double rdSecElapsed = (double)pStats->cNsElapsed / 1000000000.0;
726 RTTestIValue("Transmitted", pStats->cTx, RTTESTUNIT_PACKETS);
727 RTTestIValue("Successful echos", pStats->cEchos, RTTESTUNIT_PACKETS);
728 RTTestIValue("Errors", pStats->cErrors, RTTESTUNIT_PACKETS);
729 RTTestIValue("Interval", pStats->cNsElapsed, RTTESTUNIT_NS);
730 RTTestIValue("Packet size", cbPacket, RTTESTUNIT_BYTES);
731 RTTestIValue("Average rate", (uint64_t)(pStats->cEchos / rdSecElapsed), RTTESTUNIT_PACKETS_PER_SEC);
732 RTTestIValue("Average throughput", (uint64_t)(cbPacket * pStats->cEchos / rdSecElapsed), RTTESTUNIT_BYTES_PER_SEC);
733 RTTestIValue("Average latency", (uint64_t)(rdSecElapsed / pStats->cEchos * 1000000000.0), RTTESTUNIT_NS_PER_ROUND_TRIP);
734 RTTestISubDone();
735}
736
737
738/**
739 * NETPERFMODE -> string.
740 *
741 * @returns readonly string.
742 * @param enmMode The mode.
743 */
744static const char *netperfModeToString(NETPERFMODE enmMode)
745{
746 switch (enmMode)
747 {
748 case NETPERFMODE_LATENCY: return "latency";
749 case NETPERFMODE_THROUGHPUT: return "throughput";
750 case NETPERFMODE_THROUGHPUT_XMIT: return "throughput-xmit";
751 case NETPERFMODE_THROUGHPUT_RECV: return "throughput-recv";
752 default: AssertFailed(); return "internal-error";
753 }
754}
755
756/**
757 * String -> NETPERFMODE.
758 *
759 * @returns The corresponding NETPERFMODE, NETPERFMODE_INVALID on failure.
760 * @param pszMode The mode string.
761 */
762static NETPERFMODE netperfModeFromString(const char *pszMode)
763{
764 if (!strcmp(pszMode, "latency"))
765 return NETPERFMODE_LATENCY;
766 if ( !strcmp(pszMode, "throughput")
767 || !strcmp(pszMode, "thruput") )
768 return NETPERFMODE_THROUGHPUT;
769 if ( !strcmp(pszMode, "throughput-xmit")
770 || !strcmp(pszMode, "thruput-xmit")
771 || !strcmp(pszMode, "xmit") )
772 return NETPERFMODE_THROUGHPUT_XMIT;
773 if ( !strcmp(pszMode, "throughput-recv")
774 || !strcmp(pszMode, "thruput-recv")
775 || !strcmp(pszMode, "recv") )
776 return NETPERFMODE_THROUGHPUT_RECV;
777 return NETPERFMODE_INVALID;
778}
779
780
781
782
783
784/**
785 * TCP Server: Throughput test.
786 *
787 * @returns IPRT status code.
788 * @param pParams The parameters to use for this test.
789 */
790static int netperfTCPServerDoThroughput(NETPERFPARAMS const *pParams)
791{
792 /*
793 * Allocate the buffer.
794 */
795 NETPERFHDR *pBuf = (NETPERFHDR *)RTMemAllocZ(pParams->cbPacket);
796 if (!pBuf)
797 return RTTestIFailedRc(VERR_NO_MEMORY, "Out of memory");
798
799 /*
800 * Receive first, then Send. The reverse of the client.
801 */
802 NETPERFSTATS RecvStats;
803 int rc = netperfTCPThroughputRecv(pParams, pBuf, &RecvStats);
804 if (RT_SUCCESS(rc))
805 {
806 rc = netperfSendStats(&RecvStats, pParams->hSocket);
807 if (RT_SUCCESS(rc))
808 {
809 NETPERFSTATS SendStats;
810 rc = netperfTCPThroughputSend(pParams, pBuf, &SendStats);
811 if (RT_SUCCESS(rc))
812 {
813 rc = netperfSendStats(&SendStats, pParams->hSocket);
814 netperfPrintThroughputStats(&SendStats, &RecvStats, pParams->cbPacket);
815 }
816 }
817 }
818
819 return rc;
820}
821
822/**
823 * TCP Server: Throughput xmit test (receive from client).
824 *
825 * @returns IPRT status code.
826 * @param pParams The parameters to use for this test.
827 */
828static int netperfTCPServerDoThroughputXmit(NETPERFPARAMS const *pParams)
829{
830 /*
831 * Allocate the buffer.
832 */
833 NETPERFHDR *pBuf = (NETPERFHDR *)RTMemAllocZ(pParams->cbPacket);
834 if (!pBuf)
835 return RTTestIFailedRc(VERR_NO_MEMORY, "Out of memory");
836
837 /*
838 * Receive the transmitted data (reverse of client).
839 */
840 NETPERFSTATS RecvStats;
841 int rc = netperfTCPThroughputRecv(pParams, pBuf, &RecvStats);
842 if (RT_SUCCESS(rc))
843 {
844 rc = netperfSendStats(&RecvStats, pParams->hSocket);
845 if (RT_SUCCESS(rc))
846 netperfPrintThroughputStats(NULL, &RecvStats, pParams->cbPacket);
847 }
848
849 return rc;
850}
851
852/**
853 * TCP Server: Throughput recv test (transmit to client).
854 *
855 * @returns IPRT status code.
856 * @param pParams The parameters to use for this test.
857 */
858static int netperfTCPServerDoThroughputRecv(NETPERFPARAMS const *pParams)
859{
860 /*
861 * Allocate the buffer.
862 */
863 NETPERFHDR *pBuf = (NETPERFHDR *)RTMemAllocZ(pParams->cbPacket);
864 if (!pBuf)
865 return RTTestIFailedRc(VERR_NO_MEMORY, "Out of memory");
866
867 /*
868 * Send data to the client (reverse of client).
869 */
870 NETPERFSTATS SendStats;
871 int rc = netperfTCPThroughputSend(pParams, pBuf, &SendStats);
872 if (RT_SUCCESS(rc))
873 {
874 rc = netperfSendStats(&SendStats, pParams->hSocket);
875 if (RT_SUCCESS(rc))
876 netperfPrintThroughputStats(&SendStats, NULL, pParams->cbPacket);
877 }
878
879 return rc;
880}
881
882/**
883 * TCP Server: Latency test.
884 *
885 * @returns IPRT status code.
886 * @param pParams The parameters to use for this test.
887 */
888static int netperfTCPServerDoLatency(NETPERFPARAMS const *pParams)
889{
890 NETPERFHDR *pBuf = (NETPERFHDR *)RTMemAllocZ(pParams->cbPacket);
891 if (!pBuf)
892 return RTTestIFailedRc(VERR_NO_MEMORY, "Failed to allocated packet buffer of %u bytes.\n", pParams->cbPacket);
893
894 /*
895 * Ping pong with client.
896 */
897 int rc;
898 uint32_t uState = RT_H2LE_U32_C(NETPERFHDR_WARMUP);
899 uint32_t u32Seq = 0;
900 uint64_t cTx = 0;
901 uint64_t cRx = 0;
902 uint64_t u64StartTS = 0;
903 NETPERFSTATS Stats;
904 RT_ZERO(Stats);
905 for (;;)
906 {
907 rc = RTTcpRead(pParams->hSocket, pBuf, pParams->cbPacket, NULL);
908 if (RT_FAILURE(rc))
909 {
910 RTTestIFailed("Failed to read data from client: %Rrc\n", rc);
911 break;
912 }
913
914 /*
915 * Validate the packet
916 */
917 if (RT_UNLIKELY( pBuf->u32Magic != RT_H2LE_U32_C(NETPERFHDR_MAGIC)
918 || pBuf->u32Reserved != 0))
919 {
920 RTTestIFailed("Invalid magic or reserved field value: %#x %#x\n", RT_H2LE_U32(pBuf->u32Magic), RT_H2LE_U32(pBuf->u32Reserved));
921 rc = VERR_INVALID_MAGIC;
922 break;
923 }
924
925 u32Seq += 1;
926 if (RT_UNLIKELY(pBuf->u32Seq != RT_H2LE_U32(u32Seq)))
927 {
928 RTTestIFailed("Out of sequence: got %#x, expected %#x\n", RT_H2LE_U32(pBuf->u32Seq), u32Seq);
929 rc = VERR_WRONG_ORDER;
930 break;
931 }
932
933 /*
934 * Count the packet if the state remains unchanged.
935 */
936 if (RT_LIKELY(pBuf->u32State == uState))
937 cRx++;
938 /*
939 * Validate and act on the state transition.
940 */
941 else if ( uState == RT_H2LE_U32_C(NETPERFHDR_WARMUP)
942 && pBuf->u32State == RT_H2LE_U32_C(NETPERFHDR_TESTING))
943 {
944 cRx = cTx = 0;
945 u64StartTS = RTTimeNanoTS();
946 uState = pBuf->u32State;
947 }
948 else if ( uState == RT_H2LE_U32_C(NETPERFHDR_TESTING)
949 && ( pBuf->u32State == RT_H2LE_U32_C(NETPERFHDR_COOL_DOWN)
950 || pBuf->u32State == RT_H2LE_U32_C(NETPERFHDR_DONE)) )
951 {
952 Stats.cNsElapsed = RTTimeNanoTS() - u64StartTS;
953 Stats.cEchos = cTx;
954 Stats.cTx = cTx;
955 Stats.cRx = cRx;
956 uState = pBuf->u32State;
957 if (uState == RT_H2LE_U32_C(NETPERFHDR_DONE))
958 break;
959 }
960 else if ( uState == RT_H2LE_U32_C(NETPERFHDR_COOL_DOWN)
961 && pBuf->u32State == RT_H2LE_U32_C(NETPERFHDR_DONE))
962 {
963 uState = pBuf->u32State;
964 break;
965 }
966 else
967 {
968 RTTestIFailed("Protocol error: invalid state transition %#x -> %#x\n",
969 RT_LE2H_U32(uState), RT_LE2H_U32(pBuf->u32State));
970 break;
971 }
972
973 /*
974 * Write same data back to client.
975 */
976 rc = RTTcpWrite(pParams->hSocket, pBuf, pParams->cbPacket);
977 if (RT_FAILURE(rc))
978 {
979 RTTestIFailed("Failed to write data to client: %Rrc\n", rc);
980 break;
981 }
982
983 cTx++;
984 }
985
986 /*
987 * Send stats to client and print them.
988 */
989 if (uState == RT_H2LE_U32_C(NETPERFHDR_DONE))
990 netperfSendStats(&Stats, pParams->hSocket);
991
992 if ( uState == RT_H2LE_U32_C(NETPERFHDR_DONE)
993 || uState == RT_H2LE_U32_C(NETPERFHDR_COOL_DOWN))
994 netperfPrintLatencyStats(&Stats, pParams->cbPacket);
995
996 RTMemFree(pBuf);
997 return rc;
998}
999
1000/**
1001 * Parses the parameters the client has sent us.
1002 *
1003 * @returns IPRT status code. Message has been shown on failure.
1004 * @param pParams The parameter structure to store the parameters
1005 * in.
1006 * @param pszParams The parameter string sent by the client.
1007 */
1008static int netperfTCPServerParseParams(NETPERFPARAMS *pParams, char *pszParams)
1009{
1010 /*
1011 * Set defaults for the dynamic settings.
1012 */
1013 pParams->fNoDelay = false;
1014 pParams->enmMode = NETPERFMODE_LATENCY;
1015 pParams->cSecTimeout = NETPERF_DEFAULT_WARMUP;
1016 pParams->cMsCoolDown = NETPERF_DEFAULT_COOL_DOWN;
1017 pParams->cMsWarmup = NETPERF_DEFAULT_WARMUP;
1018 pParams->cbPacket = NETPERF_DEFAULT_PKT_SIZE_LATENCY;
1019
1020 /*
1021 * Parse the client parameters.
1022 */
1023 /* first arg: transport type. [mandatory] */
1024 char *pszCur = strchr(pszParams, ':');
1025 if (!pszCur)
1026 return RTTestIFailedRc(VERR_PARSE_ERROR, "client params: No colon\n");
1027 char *pszNext = strchr(++pszCur, ':');
1028 if (pszNext)
1029 *pszNext++ = '\0';
1030 if (strcmp(pszCur, "TCP"))
1031 return RTTestIFailedRc(VERR_PARSE_ERROR, "client params: Invalid transport type: \"%s\"\n", pszCur);
1032 pszCur = pszNext;
1033
1034 /* second arg: mode. [mandatory] */
1035 if (!pszCur)
1036 return RTTestIFailedRc(VERR_PARSE_ERROR, "client params: Missing test mode\n");
1037 pszNext = strchr(pszCur, ':');
1038 if (pszNext)
1039 *pszNext++ = '\0';
1040 pParams->enmMode = netperfModeFromString(pszCur);
1041 if (pParams->enmMode == NETPERFMODE_INVALID)
1042 return RTTestIFailedRc(VERR_PARSE_ERROR, "client params: Invalid test mode: \"%s\"\n", pszCur);
1043 pszCur = pszNext;
1044
1045 /*
1046 * The remainder are uint32_t or bool.
1047 */
1048 struct
1049 {
1050 bool fBool;
1051 bool fMandatory;
1052 void *pvValue;
1053 uint32_t uMin;
1054 uint32_t uMax;
1055 const char *pszName;
1056 } aElements[] =
1057 {
1058 { false, true, &pParams->cSecTimeout, NETPERF_MIN_TIMEOUT, NETPERF_MAX_TIMEOUT, "timeout" },
1059 { false, true, &pParams->cbPacket, NETPERF_MIN_PKT_SIZE, NETPERF_MAX_PKT_SIZE, "packet size" },
1060 { false, true, &pParams->cMsWarmup, NETPERF_MIN_WARMUP, NETPERF_MAX_WARMUP, "warmup period" },
1061 { false, true, &pParams->cMsCoolDown, NETPERF_MIN_COOL_DOWN, NETPERF_MAX_COOL_DOWN, "cool down period" },
1062 { true, true, &pParams->fNoDelay, false, true, "no delay" },
1063 };
1064
1065 for (unsigned i = 0; i < RT_ELEMENTS(aElements); i++)
1066 {
1067 if (!pszCur)
1068 return aElements[i].fMandatory
1069 ? RTTestIFailedRc(VERR_PARSE_ERROR, "client params: missing %s\n", aElements[i].pszName)
1070 : VINF_SUCCESS;
1071
1072 pszNext = strchr(pszCur, ':');
1073 if (pszNext)
1074 *pszNext++ = '\0';
1075 uint32_t u32;
1076 int rc = RTStrToUInt32Full(pszCur, 10, &u32);
1077 if (rc != VINF_SUCCESS)
1078 return RTTestIFailedRc(VERR_PARSE_ERROR, "client params: bad %s value \"%s\": %Rrc\n",
1079 aElements[i].pszName, pszCur, rc);
1080
1081 if ( u32 < aElements[i].uMin
1082 || u32 > aElements[i].uMax)
1083 return RTTestIFailedRc(VERR_PARSE_ERROR, "client params: %s %u s is out of range (%u..%u)\n",
1084 aElements[i].pszName, u32, aElements[i].uMin, aElements[i].uMax);
1085 if (aElements[i].fBool)
1086 *(bool *)aElements[i].pvValue = u32 ? true : false;
1087 else
1088 *(uint32_t *)aElements[i].pvValue = u32;
1089
1090 pszCur = pszNext;
1091 }
1092
1093 /* Fail if too many elements. */
1094 if (pszCur)
1095 return RTTestIFailedRc(VERR_PARSE_ERROR, "client params: too many elements: \"%s\"\n",
1096 pszCur);
1097 return VINF_SUCCESS;
1098}
1099
1100
1101/**
1102 * TCP server callback that handles one client connection.
1103 *
1104 * @returns IPRT status code. VERR_TCP_SERVER_STOP is special.
1105 * @param hSocket The client socket.
1106 * @param pvUser Our parameters.
1107 */
1108static DECLCALLBACK(int) netperfTCPServerWorker(RTSOCKET hSocket, void *pvUser)
1109{
1110 NETPERFPARAMS *pParams = (NETPERFPARAMS *)pvUser;
1111 AssertReturn(pParams, VERR_INVALID_POINTER);
1112
1113 pParams->hSocket = hSocket;
1114
1115 RTNETADDR Addr;
1116 int rc = RTTcpGetPeerAddress(hSocket, &Addr);
1117 if (RT_SUCCESS(rc))
1118 RTTestIPrintf(RTTESTLVL_ALWAYS, "Client connected from %RTnaddr\n", &Addr);
1119 else
1120 {
1121 RTTestIPrintf(RTTESTLVL_ALWAYS, "Failed to get client details: %Rrc\n", rc);
1122 Addr.enmType = RTNETADDRTYPE_INVALID;
1123 }
1124
1125 /*
1126 * Greet the other dude.
1127 */
1128 rc = RTTcpWrite(hSocket, g_ConnectStart, sizeof(g_ConnectStart) - 1);
1129 if (RT_FAILURE(rc))
1130 return RTTestIFailedRc(rc, "Failed to send connection start Id: %Rrc\n", rc);
1131
1132 /*
1133 * Read connection parameters.
1134 */
1135 char szBuf[256];
1136 rc = RTTcpRead(hSocket, szBuf, NETPERF_LEN_PREFIX, NULL);
1137 if (RT_FAILURE(rc))
1138 return RTTestIFailedRc(rc, "Failed to read connection parameters: %Rrc\n", rc);
1139 szBuf[NETPERF_LEN_PREFIX] = '\0';
1140 uint32_t cchParams;
1141 rc = RTStrToUInt32Full(szBuf, 10, &cchParams);
1142 if (rc != VINF_SUCCESS)
1143 return RTTestIFailedRc(RT_SUCCESS(rc) ? VERR_INTERNAL_ERROR : rc,
1144 "Failed to read connection parameters: %Rrc\n", rc);
1145 if (cchParams >= sizeof(szBuf))
1146 return RTTestIFailedRc(VERR_TOO_MUCH_DATA, "parameter packet is too big (%u bytes)\n", cchParams);
1147 rc = RTTcpRead(hSocket, szBuf, cchParams, NULL);
1148 if (RT_FAILURE(rc))
1149 return RTTestIFailedRc(rc, "Failed to read connection parameters: %Rrc\n", rc);
1150 szBuf[cchParams] = '\0';
1151
1152 if (strncmp(szBuf, g_szStartParams, sizeof(g_szStartParams) - 1))
1153 return RTTestIFailedRc(VERR_NET_PROTOCOL_ERROR, "Invalid connection parameters '%s'\n", szBuf);
1154
1155 /*
1156 * Parse the parameters and signal whether we've got a deal or not.
1157 */
1158 rc = netperfTCPServerParseParams(pParams, szBuf);
1159 if (RT_FAILURE(rc))
1160 {
1161 int rc2 = RTTcpWrite(hSocket, g_szNegative, sizeof(g_szNegative) - 1);
1162 if (RT_FAILURE(rc2))
1163 RTTestIFailed("Failed to send negative ack: %Rrc\n", rc2);
1164 return rc;
1165 }
1166
1167 if (Addr.enmType != RTNETADDRTYPE_INVALID)
1168 RTTestISubF("%RTnaddr - %s, %u s, %u bytes", &Addr,
1169 netperfModeToString(pParams->enmMode), pParams->cSecTimeout, pParams->cbPacket);
1170 else
1171 RTTestISubF("Unknown - %s, %u s, %u bytes",
1172 netperfModeToString(pParams->enmMode), pParams->cSecTimeout, pParams->cbPacket);
1173
1174 rc = RTTcpSetSendCoalescing(hSocket, !pParams->fNoDelay);
1175 if (RT_FAILURE(rc))
1176 return RTTestIFailedRc(rc, "Failed to apply no-delay option (%RTbool): %Rrc\n", pParams->fNoDelay, rc);
1177
1178 rc = RTTcpWrite(hSocket, g_szAck, sizeof(g_szAck) - 1);
1179 if (RT_FAILURE(rc))
1180 return RTTestIFailedRc(rc, "Failed to send start test commend to client: %Rrc\n", rc);
1181
1182 /*
1183 * Take action according to our mode.
1184 */
1185 switch (pParams->enmMode)
1186 {
1187 case NETPERFMODE_LATENCY:
1188 rc = netperfTCPServerDoLatency(pParams);
1189 break;
1190
1191 case NETPERFMODE_THROUGHPUT:
1192 rc = netperfTCPServerDoThroughput(pParams);
1193 break;
1194
1195 case NETPERFMODE_THROUGHPUT_XMIT:
1196 rc = netperfTCPServerDoThroughputXmit(pParams);
1197 break;
1198
1199 case NETPERFMODE_THROUGHPUT_RECV:
1200 rc = netperfTCPServerDoThroughputRecv(pParams);
1201 break;
1202
1203 case NETPERFMODE_INVALID:
1204 rc = VERR_INTERNAL_ERROR;
1205 break;
1206
1207 /* no default! */
1208 }
1209 if (rc == VERR_NO_MEMORY)
1210 return VERR_TCP_SERVER_STOP;
1211
1212 /*
1213 * Wait for other clients or quit.
1214 */
1215 if (pParams->fSingleClient)
1216 return VERR_TCP_SERVER_STOP;
1217 return VINF_SUCCESS;
1218}
1219
1220
1221/**
1222 * TCP server.
1223 *
1224 * @returns IPRT status code.
1225 * @param pParams The TCP parameter block.
1226 */
1227static int netperfTCPServer(NETPERFPARAMS *pParams)
1228{
1229 /*
1230 * Spawn the TCP server thread & listen.
1231 */
1232 PRTTCPSERVER pServer;
1233 int rc = RTTcpServerCreateEx(NULL, pParams->uPort, &pServer);
1234 if (RT_SUCCESS(rc))
1235 {
1236 RTPrintf("Server listening on TCP port %d\n", pParams->uPort);
1237 rc = RTTcpServerListen(pServer, netperfTCPServerWorker, pParams);
1238 RTTcpServerDestroy(pServer);
1239 }
1240 else
1241 RTPrintf("Failed to create TCP server thread: %Rrc\n", rc);
1242
1243 return rc;
1244}
1245
1246/**
1247 * The server part.
1248 *
1249 * @returns Exit code.
1250 * @param enmProto The protocol.
1251 * @param pParams The parameter block.
1252 */
1253static RTEXITCODE netperfServer(NETPERFPROTO enmProto, NETPERFPARAMS *pParams)
1254{
1255
1256 switch (enmProto)
1257 {
1258 case NETPERFPROTO_TCP:
1259 {
1260 int rc = netperfTCPServer(pParams);
1261 return RT_SUCCESS(rc) ? RTEXITCODE_SUCCESS : RTEXITCODE_FAILURE;
1262 }
1263
1264 default:
1265 RTTestIFailed("Protocol not supported.\n");
1266 return RTEXITCODE_FAILURE;
1267 }
1268}
1269
1270
1271
1272
1273
1274/**
1275 * TCP client: Do the throughput test.
1276 *
1277 * @returns IPRT status code
1278 * @param pParams The parameters.
1279 */
1280static int netperfTCPClientDoThroughput(NETPERFPARAMS *pParams)
1281{
1282 /*
1283 * Allocate the buffer.
1284 */
1285 NETPERFHDR *pBuf = (NETPERFHDR *)RTMemAllocZ(pParams->cbPacket);
1286 if (!pBuf)
1287 return RTTestIFailedRc(VERR_NO_MEMORY, "Out of memory");
1288
1289 /*
1290 * Send first, then Receive.
1291 */
1292 NETPERFSTATS SendStats;
1293 int rc = netperfTCPThroughputSend(pParams, pBuf, &SendStats);
1294 if (RT_SUCCESS(rc))
1295 {
1296 NETPERFSTATS SrvSendStats;
1297 rc = netperfRecvStats(&SrvSendStats, pParams->hSocket);
1298 if (RT_SUCCESS(rc))
1299 {
1300 NETPERFSTATS RecvStats;
1301 rc = netperfTCPThroughputRecv(pParams, pBuf, &RecvStats);
1302 if (RT_SUCCESS(rc))
1303 {
1304 NETPERFSTATS SrvRecvStats;
1305 rc = netperfRecvStats(&SrvRecvStats, pParams->hSocket);
1306 if (RT_SUCCESS(rc))
1307 {
1308 if (pParams->fServerStats)
1309 netperfPrintThroughputStats(&SrvSendStats, &SrvRecvStats, pParams->cbPacket);
1310 else
1311 netperfPrintThroughputStats(&SendStats, &RecvStats, pParams->cbPacket);
1312 }
1313 }
1314 }
1315 }
1316
1317 RTTestISubDone();
1318 return rc;
1319}
1320
1321/**
1322 * TCP client: Do the throughput xmit test.
1323 *
1324 * @returns IPRT status code
1325 * @param pParams The parameters.
1326 */
1327static int netperfTCPClientDoThroughputXmit(NETPERFPARAMS *pParams)
1328{
1329 /*
1330 * Allocate the buffer.
1331 */
1332 NETPERFHDR *pBuf = (NETPERFHDR *)RTMemAllocZ(pParams->cbPacket);
1333 if (!pBuf)
1334 return RTTestIFailedRc(VERR_NO_MEMORY, "Out of memory");
1335
1336 /*
1337 * Do the job.
1338 */
1339 NETPERFSTATS SendStats;
1340 int rc = netperfTCPThroughputSend(pParams, pBuf, &SendStats);
1341 if (RT_SUCCESS(rc))
1342 {
1343 NETPERFSTATS SrvSendStats;
1344 rc = netperfRecvStats(&SrvSendStats, pParams->hSocket);
1345 if (RT_SUCCESS(rc))
1346 {
1347 if (pParams->fServerStats)
1348 netperfPrintThroughputStats(&SrvSendStats, NULL, pParams->cbPacket);
1349 else
1350 netperfPrintThroughputStats(&SendStats, NULL, pParams->cbPacket);
1351 }
1352 }
1353
1354 RTTestISubDone();
1355 return rc;
1356}
1357
1358/**
1359 * TCP client: Do the throughput recv test.
1360 *
1361 * @returns IPRT status code
1362 * @param pParams The parameters.
1363 */
1364static int netperfTCPClientDoThroughputRecv(NETPERFPARAMS *pParams)
1365{
1366 /*
1367 * Allocate the buffer.
1368 */
1369 NETPERFHDR *pBuf = (NETPERFHDR *)RTMemAllocZ(pParams->cbPacket);
1370 if (!pBuf)
1371 return RTTestIFailedRc(VERR_NO_MEMORY, "Out of memory");
1372
1373 /*
1374 * Do the job.
1375 */
1376 NETPERFSTATS RecvStats;
1377 int rc = netperfTCPThroughputRecv(pParams, pBuf, &RecvStats);
1378 if (RT_SUCCESS(rc))
1379 {
1380 NETPERFSTATS SrvRecvStats;
1381 rc = netperfRecvStats(&SrvRecvStats, pParams->hSocket);
1382 if (RT_SUCCESS(rc))
1383 {
1384 if (pParams->fServerStats)
1385 netperfPrintThroughputStats(NULL, &SrvRecvStats, pParams->cbPacket);
1386 else
1387 netperfPrintThroughputStats(NULL, &RecvStats, pParams->cbPacket);
1388 }
1389 }
1390
1391 RTTestISubDone();
1392 return rc;
1393}
1394
1395/**
1396 * TCP client: Do the latency test.
1397 *
1398 * @returns IPRT status code
1399 * @param pParams The parameters.
1400 */
1401static int netperfTCPClientDoLatency(NETPERFPARAMS *pParams)
1402{
1403 /*
1404 * Generate a selection of packages before we start, after all we're not
1405 * benchmarking the random number generator, are we. :-)
1406 */
1407 void *pvReadBuf = RTMemAllocZ(pParams->cbPacket);
1408 if (!pvReadBuf)
1409 return RTTestIFailedRc(VERR_NO_MEMORY, "Out of memory");
1410
1411 size_t i;
1412 NETPERFHDR *apPackets[256];
1413 for (i = 0; i < RT_ELEMENTS(apPackets); i++)
1414 {
1415 apPackets[i] = (NETPERFHDR *)RTMemAllocZ(pParams->cbPacket);
1416 if (!apPackets[i])
1417 {
1418 while (i-- > 0)
1419 RTMemFree(apPackets[i]);
1420 RTMemFree(pvReadBuf);
1421 return RTTestIFailedRc(VERR_NO_MEMORY, "Out of memory");
1422 }
1423 RTRandBytes(apPackets[i], pParams->cbPacket);
1424 apPackets[i]->u32Magic = RT_H2LE_U32_C(NETPERFHDR_MAGIC);
1425 apPackets[i]->u32State = 0;
1426 apPackets[i]->u32Seq = 0;
1427 apPackets[i]->u32Reserved = 0;
1428 }
1429
1430 /*
1431 * Create & start a timer to eventually disconnect.
1432 */
1433 bool volatile fStop = false;
1434 RTTIMERLR hTimer;
1435 int rc = RTTimerLRCreateEx(&hTimer, 0 /* nsec */, RTTIMER_FLAGS_CPU_ANY, netperfStopTimerCallback, (void *)&fStop);
1436 if (RT_SUCCESS(rc))
1437 {
1438 uint32_t u32Seq = 0;
1439 NETPERFSTATS Stats;
1440 RT_ZERO(Stats);
1441
1442 /*
1443 * Warm up.
1444 */
1445 rc = RTTimerLRStart(hTimer, pParams->cMsWarmup * UINT64_C(1000000) /* nsec */);
1446 if (RT_SUCCESS(rc))
1447 {
1448 while (!fStop)
1449 {
1450 NETPERFHDR *pPacket = apPackets[u32Seq % RT_ELEMENTS(apPackets)];
1451 u32Seq++;
1452 pPacket->u32Seq = RT_H2LE_U32(u32Seq);
1453 pPacket->u32State = RT_H2LE_U32_C(NETPERFHDR_WARMUP);
1454 rc = RTTcpWrite(pParams->hSocket, pPacket, pParams->cbPacket);
1455 if (RT_FAILURE(rc))
1456 {
1457 RTTestIFailed("RTTcpWrite/warmup: %Rrc\n", rc);
1458 break;
1459 }
1460 rc = RTTcpRead(pParams->hSocket, pvReadBuf, pParams->cbPacket, NULL);
1461 if (RT_FAILURE(rc))
1462 {
1463 RTTestIFailed("RTTcpRead/warmup: %Rrc\n", rc);
1464 break;
1465 }
1466 }
1467 }
1468 else
1469 RTTestIFailed("RTTimerLRStart/warmup: %Rrc\n", rc);
1470
1471 /*
1472 * The real thing.
1473 */
1474 if (RT_SUCCESS(rc))
1475 {
1476 fStop = false;
1477 rc = RTTimerLRStart(hTimer, pParams->cSecTimeout * UINT64_C(1000000000) /* nsec */);
1478 if (RT_SUCCESS(rc))
1479 {
1480 uint64_t u64StartTS = RTTimeNanoTS();
1481 while (!fStop)
1482 {
1483 NETPERFHDR *pPacket = apPackets[u32Seq % RT_ELEMENTS(apPackets)];
1484 u32Seq++;
1485 pPacket->u32Seq = RT_H2LE_U32(u32Seq);
1486 pPacket->u32State = RT_H2LE_U32_C(NETPERFHDR_TESTING);
1487 rc = RTTcpWrite(pParams->hSocket, pPacket, pParams->cbPacket);
1488 if (RT_FAILURE(rc))
1489 {
1490 RTTestIFailed("RTTcpWrite/testing: %Rrc\n", rc);
1491 break;
1492 }
1493 Stats.cTx++;
1494
1495 rc = RTTcpRead(pParams->hSocket, pvReadBuf, pParams->cbPacket, NULL);
1496 if (RT_FAILURE(rc))
1497 {
1498 RTTestIFailed("RTTcpRead/testing: %Rrc\n", rc);
1499 break;
1500 }
1501 Stats.cRx++;
1502
1503 if (!memcmp(pvReadBuf, pPacket, pParams->cbPacket))
1504 Stats.cEchos++;
1505 else
1506 Stats.cErrors++;
1507 }
1508 Stats.cNsElapsed = RTTimeNanoTS() - u64StartTS;
1509 }
1510 else
1511 RTTestIFailed("RTTimerLRStart/testing: %Rrc\n", rc);
1512 }
1513
1514 /*
1515 * Cool down.
1516 */
1517 if (RT_SUCCESS(rc))
1518 {
1519 fStop = false;
1520 rc = RTTimerLRStart(hTimer, pParams->cMsCoolDown * UINT64_C(1000000) /* nsec */);
1521 if (RT_SUCCESS(rc))
1522 {
1523 while (!fStop)
1524 {
1525 NETPERFHDR *pPacket = apPackets[u32Seq % RT_ELEMENTS(apPackets)];
1526 u32Seq++;
1527 pPacket->u32Seq = RT_H2LE_U32(u32Seq);
1528 pPacket->u32State = RT_H2LE_U32_C(NETPERFHDR_COOL_DOWN);
1529 rc = RTTcpWrite(pParams->hSocket, pPacket, pParams->cbPacket);
1530 if (RT_FAILURE(rc))
1531 {
1532 RTTestIFailed("RTTcpWrite/warmup: %Rrc\n", rc);
1533 break;
1534 }
1535 rc = RTTcpRead(pParams->hSocket, pvReadBuf, pParams->cbPacket, NULL);
1536 if (RT_FAILURE(rc))
1537 {
1538 RTTestIFailed("RTTcpRead/warmup: %Rrc\n", rc);
1539 break;
1540 }
1541 }
1542 }
1543 else
1544 RTTestIFailed("RTTimerLRStart/testing: %Rrc\n", rc);
1545 }
1546
1547 /*
1548 * Send DONE packet.
1549 */
1550 if (RT_SUCCESS(rc))
1551 {
1552 u32Seq++;
1553 NETPERFHDR *pPacket = apPackets[u32Seq % RT_ELEMENTS(apPackets)];
1554 pPacket->u32Seq = RT_H2LE_U32(u32Seq);
1555 pPacket->u32State = RT_H2LE_U32_C(NETPERFHDR_DONE);
1556 rc = RTTcpWrite(pParams->hSocket, pPacket, pParams->cbPacket);
1557 if (RT_FAILURE(rc))
1558 RTTestIFailed("RTTcpWrite/done: %Rrc\n", rc);
1559 }
1560
1561
1562 /*
1563 * Get and print stats.
1564 */
1565 NETPERFSTATS SrvStats;
1566 if (RT_SUCCESS(rc))
1567 {
1568 rc = netperfRecvStats(&SrvStats, pParams->hSocket);
1569 if (RT_SUCCESS(rc) && pParams->fServerStats)
1570 netperfPrintLatencyStats(&SrvStats, pParams->cbPacket);
1571 else if (!pParams->fServerStats)
1572 netperfPrintLatencyStats(&Stats, pParams->cbPacket);
1573 }
1574
1575 /* clean up*/
1576 RTTimerLRDestroy(hTimer);
1577 }
1578 else
1579 RTTestIFailed("Failed to create timer object: %Rrc\n", rc);
1580 for (i = 0; i < RT_ELEMENTS(apPackets); i++)
1581 RTMemFree(apPackets[i]);
1582
1583 RTMemFree(pvReadBuf);
1584
1585 return rc;
1586}
1587
1588/**
1589 * TCP client test driver.
1590 *
1591 * @returns IPRT status code
1592 * @param pszServer The server name.
1593 * @param pParams The parameter structure.
1594 */
1595static int netperfTCPClient(const char *pszServer, NETPERFPARAMS *pParams)
1596{
1597 AssertReturn(pParams, VERR_INVALID_POINTER);
1598 RTTestISubF("TCP - %u s, %u bytes%s", pParams->cSecTimeout,
1599 pParams->cbPacket, pParams->fNoDelay ? ", no delay" : "");
1600
1601 RTSOCKET hSocket = NIL_RTSOCKET;
1602 int rc = RTTcpClientConnect(pszServer, pParams->uPort, &hSocket);
1603 if (RT_FAILURE(rc))
1604 return RTTestIFailedRc(rc, "Failed to connect to %s on port %u: %Rrc\n", pszServer, pParams->uPort, rc);
1605 pParams->hSocket = hSocket;
1606
1607 /*
1608 * Disable send coalescing (no-delay).
1609 */
1610 if (pParams->fNoDelay)
1611 {
1612 rc = RTTcpSetSendCoalescing(hSocket, false /*fEnable*/);
1613 if (RT_FAILURE(rc))
1614 return RTTestIFailedRc(rc, "Failed to set no-delay option: %Rrc\n", rc);
1615 }
1616
1617 /*
1618 * Verify the super secret Start Connect Id to start the connection.
1619 */
1620 char szBuf[256 + NETPERF_LEN_PREFIX];
1621 RT_ZERO(szBuf);
1622 rc = RTTcpRead(hSocket, szBuf, sizeof(g_ConnectStart) - 1, NULL);
1623 if (RT_FAILURE(rc))
1624 return RTTestIFailedRc(rc, "Failed to read connection initializer: %Rrc\n", rc);
1625
1626 if (strcmp(szBuf, g_ConnectStart))
1627 return RTTestIFailedRc(VERR_INVALID_MAGIC, "Invalid connection initializer '%s'\n", szBuf);
1628
1629 /*
1630 * Send all the dynamic parameters to the server.
1631 * (If the server is newer than the client, it will select default for any
1632 * missing parameters.)
1633 */
1634 size_t cchParams = RTStrPrintf(&szBuf[NETPERF_LEN_PREFIX], sizeof(szBuf) - NETPERF_LEN_PREFIX,
1635 "%s:%s:%s:%u:%u:%u:%u:%u",
1636 g_szStartParams,
1637 "TCP",
1638 netperfModeToString(pParams->enmMode),
1639 pParams->cSecTimeout,
1640 pParams->cbPacket,
1641 pParams->cMsWarmup,
1642 pParams->cMsCoolDown,
1643 pParams->fNoDelay);
1644 RTStrPrintf(szBuf, NETPERF_LEN_PREFIX + 1, "%0*u", NETPERF_LEN_PREFIX, cchParams);
1645 szBuf[NETPERF_LEN_PREFIX] = g_szStartParams[0];
1646 Assert(strlen(szBuf) == NETPERF_LEN_PREFIX + cchParams);
1647 rc = RTTcpWrite(hSocket, szBuf, NETPERF_LEN_PREFIX + cchParams);
1648 if (RT_FAILURE(rc))
1649 return RTTestIFailedRc(rc, "Failed to send connection parameters: %Rrc\n", rc);
1650
1651 /*
1652 * Wait for acknowledgment.
1653 */
1654 rc = RTTcpRead(hSocket, szBuf, sizeof(g_szAck) - 1, NULL);
1655 if (RT_FAILURE(rc))
1656 return RTTestIFailedRc(rc, "Failed to send parameters: %Rrc\n", rc);
1657 szBuf[sizeof(g_szAck) - 1] = '\0';
1658
1659 if (!strcmp(szBuf, g_szNegative))
1660 return RTTestIFailedRc(VERR_NET_PROTOCOL_ERROR, "Server failed to accept packet size of %u bytes.\n", pParams->cbPacket);
1661 if (strcmp(szBuf, g_szAck))
1662 return RTTestIFailedRc(VERR_NET_PROTOCOL_ERROR, "Invalid response from server '%s'\n", szBuf);
1663
1664 /*
1665 * Take action according to our mode.
1666 */
1667 switch (pParams->enmMode)
1668 {
1669 case NETPERFMODE_LATENCY:
1670 RTTestIPrintf(RTTESTLVL_ALWAYS, "Connected to %s port %u, running the latency test for %u seconds.\n",
1671 pszServer, pParams->uPort, pParams->cSecTimeout);
1672 rc = netperfTCPClientDoLatency(pParams);
1673 break;
1674
1675 case NETPERFMODE_THROUGHPUT:
1676 RTTestIPrintf(RTTESTLVL_ALWAYS, "Connected to %s port %u, running the throughput test for %u seconds in each direction.\n",
1677 pszServer, pParams->uPort, pParams->cSecTimeout);
1678 rc = netperfTCPClientDoThroughput(pParams);
1679 break;
1680
1681 case NETPERFMODE_THROUGHPUT_XMIT:
1682 RTTestIPrintf(RTTESTLVL_ALWAYS, "Connected to %s port %u, running the throughput-xmit test for %u seconds.\n",
1683 pszServer, pParams->uPort, pParams->cSecTimeout);
1684 rc = netperfTCPClientDoThroughputXmit(pParams);
1685 break;
1686
1687 case NETPERFMODE_THROUGHPUT_RECV:
1688 RTTestIPrintf(RTTESTLVL_ALWAYS, "Connected to %s port %u, running the throughput-recv test for %u seconds.\n",
1689 pszServer, pParams->uPort, pParams->cSecTimeout);
1690 rc = netperfTCPClientDoThroughputRecv(pParams);
1691 break;
1692
1693 case NETPERFMODE_INVALID:
1694 rc = VERR_INTERNAL_ERROR;
1695 break;
1696
1697 /* no default! */
1698 }
1699 return rc;
1700}
1701
1702/**
1703 * The client part.
1704 *
1705 * @returns Exit code.
1706 * @param enmProto The protocol.
1707 * @param pszServer The server name.
1708 * @param pvUser The parameter block as opaque user data.
1709 */
1710static RTEXITCODE netperfClient(NETPERFPROTO enmProto, const char *pszServer, void *pvUser)
1711{
1712 switch (enmProto)
1713 {
1714 case NETPERFPROTO_TCP:
1715 {
1716 NETPERFPARAMS *pParams = (NETPERFPARAMS *)pvUser;
1717 int rc = netperfTCPClient(pszServer, pParams);
1718 if (pParams->hSocket != NIL_RTSOCKET)
1719 {
1720 RTTcpClientClose(pParams->hSocket);
1721 pParams->hSocket = NIL_RTSOCKET;
1722 }
1723 return RT_SUCCESS(rc) ? RTEXITCODE_SUCCESS : RTEXITCODE_FAILURE;
1724 }
1725
1726 default:
1727 RTTestIFailed("Protocol not supported.\n");
1728 return RTEXITCODE_FAILURE;
1729 }
1730}
1731
1732
1733int main(int argc, char *argv[])
1734{
1735 /*
1736 * Init IPRT and globals.
1737 */
1738 int rc = RTTestInitAndCreate("NetPerf", &g_hTest);
1739 if (rc)
1740 return rc;
1741
1742 /*
1743 * Special case.
1744 */
1745 if (argc < 2)
1746 {
1747 RTTestFailed(g_hTest, "No arguments given.");
1748 return RTTestSummaryAndDestroy(g_hTest);
1749 }
1750
1751 /*
1752 * Default values.
1753 */
1754 NETPERFPROTO enmProtocol = NETPERFPROTO_TCP;
1755 bool fServer = true;
1756 bool fDaemonize = false;
1757 bool fDaemonized = false;
1758 bool fPacketSizeSet = false;
1759 const char *pszServerAddress= NULL;
1760 NETPERFPARAMS Params;
1761
1762 Params.uPort = NETPERF_DEFAULT_PORT;
1763 Params.fServerStats = false;
1764 Params.fSingleClient = false;
1765
1766 Params.fNoDelay = false;
1767 Params.fCheckData = false;
1768 Params.enmMode = NETPERFMODE_LATENCY;
1769 Params.cSecTimeout = NETPERF_DEFAULT_TIMEOUT;
1770 Params.cMsWarmup = NETPERF_DEFAULT_WARMUP;
1771 Params.cMsCoolDown = NETPERF_DEFAULT_COOL_DOWN;
1772 Params.cbPacket = NETPERF_DEFAULT_PKT_SIZE_LATENCY;
1773
1774 Params.hSocket = NIL_RTSOCKET;
1775
1776 RTGETOPTUNION ValueUnion;
1777 RTGETOPTSTATE GetState;
1778 RTGetOptInit(&GetState, argc, argv, g_aCmdOptions, RT_ELEMENTS(g_aCmdOptions), 1, 0 /* fFlags */);
1779 while ((rc = RTGetOpt(&GetState, &ValueUnion)))
1780 {
1781 switch (rc)
1782 {
1783 case 's':
1784 fServer = true;
1785 break;
1786
1787 case 'c':
1788 fServer = false;
1789 pszServerAddress = ValueUnion.psz;
1790 break;
1791
1792 case 'd':
1793 fDaemonize = true;
1794 break;
1795
1796 case 'D':
1797 fDaemonized = true;
1798 break;
1799
1800 case 'i':
1801 Params.cSecTimeout = ValueUnion.u32;
1802 if ( Params.cSecTimeout < NETPERF_MIN_TIMEOUT
1803 || Params.cSecTimeout > NETPERF_MAX_TIMEOUT)
1804 {
1805 RTTestFailed(g_hTest, "Invalid interval %u s, valid range: %u-%u\n",
1806 Params.cbPacket, NETPERF_MIN_TIMEOUT, NETPERF_MAX_TIMEOUT);
1807 return RTTestSummaryAndDestroy(g_hTest);
1808 }
1809 break;
1810
1811 case 'l':
1812 Params.cbPacket = ValueUnion.u32;
1813 if ( Params.cbPacket < NETPERF_MIN_PKT_SIZE
1814 || Params.cbPacket > NETPERF_MAX_PKT_SIZE)
1815 {
1816 RTTestFailed(g_hTest, "Invalid packet size %u bytes, valid range: %u-%u\n",
1817 Params.cbPacket, NETPERF_MIN_PKT_SIZE, NETPERF_MAX_PKT_SIZE);
1818 return RTTestSummaryAndDestroy(g_hTest);
1819 }
1820 fPacketSizeSet = true;
1821 break;
1822
1823 case 'm':
1824 Params.enmMode = netperfModeFromString(ValueUnion.psz);
1825 if (Params.enmMode == NETPERFMODE_INVALID)
1826 {
1827 RTTestFailed(g_hTest, "Invalid test mode: \"%s\"\n", ValueUnion.psz);
1828 return RTTestSummaryAndDestroy(g_hTest);
1829 }
1830 if (!fPacketSizeSet)
1831 switch (Params.enmMode)
1832 {
1833 case NETPERFMODE_LATENCY:
1834 Params.cbPacket = NETPERF_DEFAULT_PKT_SIZE_LATENCY;
1835 break;
1836 case NETPERFMODE_THROUGHPUT:
1837 case NETPERFMODE_THROUGHPUT_XMIT:
1838 case NETPERFMODE_THROUGHPUT_RECV:
1839 Params.cbPacket = NETPERF_DEFAULT_PKT_SIZE_THROUGHPUT;
1840 break;
1841 case NETPERFMODE_INVALID:
1842 break;
1843 /* no default! */
1844 }
1845 break;
1846
1847 case 'p':
1848 Params.uPort = ValueUnion.u32;
1849 break;
1850
1851 case 'N':
1852 Params.fNoDelay = true;
1853 break;
1854
1855 case 'S':
1856 Params.fServerStats = true;
1857 break;
1858
1859 case '1':
1860 Params.fSingleClient = true;
1861 break;
1862
1863 case 'h':
1864 Usage(g_pStdOut);
1865 return RTEXITCODE_SUCCESS;
1866
1867 case 'V':
1868 RTPrintf("$Revision: 64366 $\n");
1869 return RTEXITCODE_SUCCESS;
1870
1871 case 'w':
1872 Params.cMsWarmup = ValueUnion.u32;
1873 if ( Params.cMsWarmup < NETPERF_MIN_WARMUP
1874 || Params.cMsWarmup > NETPERF_MAX_WARMUP)
1875 {
1876 RTTestFailed(g_hTest, "invalid warmup time %u ms, valid range: %u-%u\n",
1877 Params.cMsWarmup, NETPERF_MIN_WARMUP, NETPERF_MAX_WARMUP);
1878 return RTTestSummaryAndDestroy(g_hTest);
1879 }
1880 break;
1881
1882 case 'W':
1883 Params.cMsCoolDown = ValueUnion.u32;
1884 if ( Params.cMsCoolDown < NETPERF_MIN_COOL_DOWN
1885 || Params.cMsCoolDown > NETPERF_MAX_COOL_DOWN)
1886 {
1887 RTTestFailed(g_hTest, "invalid cool down time %u ms, valid range: %u-%u\n",
1888 Params.cMsCoolDown, NETPERF_MIN_COOL_DOWN, NETPERF_MAX_COOL_DOWN);
1889 return RTTestSummaryAndDestroy(g_hTest);
1890 }
1891 break;
1892
1893 case 'C':
1894 Params.fCheckData = true;
1895 break;
1896
1897 default:
1898 return RTGetOptPrintError(rc, &ValueUnion);
1899 }
1900 }
1901
1902 /*
1903 * Handle the server process daemoniziation.
1904 */
1905 if (fDaemonize && !fDaemonized && fServer)
1906 {
1907 rc = RTProcDaemonize(argv, "--daemonized");
1908 if (RT_FAILURE(rc))
1909 return RTMsgErrorExit(RTEXITCODE_FAILURE, "RTProcDaemonize failed: %Rrc\n", rc);
1910 return RTEXITCODE_SUCCESS;
1911 }
1912
1913 /*
1914 * Get down to business.
1915 */
1916 RTTestBanner(g_hTest);
1917 if (fServer)
1918 rc = netperfServer(enmProtocol, &Params);
1919 else if (pszServerAddress)
1920 rc = netperfClient(enmProtocol, pszServerAddress, &Params);
1921 else
1922 RTTestFailed(g_hTest, "missing server address to connect to\n");
1923
1924 RTEXITCODE rc2 = RTTestSummaryAndDestroy(g_hTest);
1925 return rc2 != RTEXITCODE_FAILURE ? (RTEXITCODE)rc2 : rc;
1926}
1927
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette