VirtualBox

source: vbox/trunk/src/VBox/ValidationKit/utils/serial/SerialTest.cpp@ 71214

Last change on this file since 71214 was 71029, checked in by vboxsync, 7 years ago

ValidationKit/SerialTest: Copy'n'paste in test error message

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 34.5 KB
Line 
1/* $Id: SerialTest.cpp 71029 2018-02-15 15:29:00Z vboxsync $ */
2/** @file
3 * SerialTest - Serial port testing utility.
4 */
5
6/*
7 * Copyright (C) 2017-2018 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/err.h>
32#include <iprt/getopt.h>
33#include <iprt/mem.h>
34#include <iprt/path.h>
35#include <iprt/param.h>
36#include <iprt/process.h>
37#include <iprt/rand.h>
38#include <iprt/serialport.h>
39#include <iprt/stream.h>
40#include <iprt/string.h>
41#include <iprt/test.h>
42
43
44/*********************************************************************************************************************************
45* Defined Constants And Macros *
46*********************************************************************************************************************************/
47
48/** Number of times to toggle the status lines during the test. */
49#define SERIALTEST_STS_LINE_TOGGLE_COUNT 100
50
51
52/*********************************************************************************************************************************
53* Structures and Typedefs *
54*********************************************************************************************************************************/
55
56
57/**
58 * Serial test mode.
59 */
60typedef enum SERIALTESTMODE
61{
62 /** Invalid mode. */
63 SERIALTESTMODE_INVALID = 0,
64 /** Serial port is looped back to itself */
65 SERIALTESTMODE_LOOPBACK,
66 /** A secondary serial port is used with a null modem cable in between. */
67 SERIALTESTMODE_SECONDARY,
68 /** The serial port is connected externally over which we have no control. */
69 SERIALTESTMODE_EXTERNAL,
70 /** Usual 32bit hack. */
71 SERIALTESTMODE_32BIT_HACK = 0x7fffffff
72} SERIALTESTMODE;
73/** Pointer to a serial test mode. */
74typedef SERIALTESTMODE *PSERIALTESTMDOE;
75
76/** Pointer to the serial test data instance. */
77typedef struct SERIALTEST *PSERIALTEST;
78
79/**
80 * Test callback function.
81 *
82 * @returns IPRT status code.
83 * @param pSerialTest The serial test instance data.
84 */
85typedef DECLCALLBACK(int) FNSERIALTESTRUN(PSERIALTEST pSerialTest);
86/** Pointer to the serial test callback. */
87typedef FNSERIALTESTRUN *PFNSERIALTESTRUN;
88
89
90/**
91 * The serial test instance data.
92 */
93typedef struct SERIALTEST
94{
95 /** The assigned test handle. */
96 RTTEST hTest;
97 /** The assigned serial port. */
98 RTSERIALPORT hSerialPort;
99 /** The currently active config. */
100 PCRTSERIALPORTCFG pSerialCfg;
101} SERIALTEST;
102
103
104/**
105 * Test descriptor.
106 */
107typedef struct SERIALTESTDESC
108{
109 /** Test ID. */
110 const char *pszId;
111 /** Test description. */
112 const char *pszDesc;
113 /** Test run callback. */
114 PFNSERIALTESTRUN pfnRun;
115} SERIALTESTDESC;
116/** Pointer to a test descriptor. */
117typedef SERIALTESTDESC *PSERIALTESTDESC;
118/** Pointer to a constant test descriptor. */
119typedef const SERIALTESTDESC *PCSERIALTESTDESC;
120
121
122/**
123 * TX/RX buffer containing a simple counter.
124 */
125typedef struct SERIALTESTTXRXBUFCNT
126{
127 /** The current counter value. */
128 uint32_t iCnt;
129 /** Number of bytes left to receive/transmit. */
130 size_t cbTxRxLeft;
131 /** The offset into the buffer to receive to/send from. */
132 size_t offBuf;
133 /** Maximum size to send/receive before processing is needed again. */
134 size_t cbTxRxMax;
135 /** The data buffer. */
136 uint8_t abBuf[_1K];
137} SERIALTESTTXRXBUFCNT;
138/** Pointer to a TX/RX buffer. */
139typedef SERIALTESTTXRXBUFCNT *PSERIALTESTTXRXBUFCNT;
140
141
142/*********************************************************************************************************************************
143* Global Variables *
144*********************************************************************************************************************************/
145
146
147/** Command line parameters */
148static const RTGETOPTDEF g_aCmdOptions[] =
149{
150 {"--device", 'd', RTGETOPT_REQ_STRING },
151 {"--baudrate", 'b', RTGETOPT_REQ_UINT32 },
152 {"--parity", 'p', RTGETOPT_REQ_STRING },
153 {"--databits", 'c', RTGETOPT_REQ_UINT32 },
154 {"--stopbits", 's', RTGETOPT_REQ_STRING },
155 {"--mode", 'm', RTGETOPT_REQ_STRING },
156 {"--secondarydevice", 'l', RTGETOPT_REQ_STRING },
157 {"--tests", 't', RTGETOPT_REQ_STRING },
158 {"--txbytes", 'x', RTGETOPT_REQ_UINT32 },
159 {"--help", 'h', RTGETOPT_REQ_NOTHING}
160};
161
162
163static DECLCALLBACK(int) serialTestRunReadWrite(PSERIALTEST pSerialTest);
164static DECLCALLBACK(int) serialTestRunWrite(PSERIALTEST pSerialTest);
165static DECLCALLBACK(int) serialTestRunStsLines(PSERIALTEST pSerialTest);
166
167/** Implemented tests. */
168static const SERIALTESTDESC g_aSerialTests[] =
169{
170 {"readwrite", "Simple Read/Write test on the same serial port", serialTestRunReadWrite },
171 {"write", "Simple write test (verification done somewhere else)", serialTestRunWrite },
172 {"stslines", "Testing the status line setting and receiving", serialTestRunStsLines }
173};
174
175/** The test handle. */
176static RTTEST g_hTest = NIL_RTTEST;
177/** The serial test mode. */
178static SERIALTESTMODE g_enmMode = SERIALTESTMODE_LOOPBACK;
179/** Random number generator. */
180static RTRAND g_hRand = NIL_RTRAND;
181/** The serial port handle. */
182static RTSERIALPORT g_hSerialPort = NIL_RTSERIALPORT;
183/** The loopback serial port handle if configured. */
184static RTSERIALPORT g_hSerialPortSecondary = NIL_RTSERIALPORT;
185/** Number of bytes to transmit for read/write tests. */
186static size_t g_cbTx = _1M;
187/** The config used. */
188static RTSERIALPORTCFG g_SerialPortCfg =
189{
190 /* uBaudRate */
191 115200,
192 /* enmParity */
193 RTSERIALPORTPARITY_NONE,
194 /* enmDataBitCount */
195 RTSERIALPORTDATABITS_8BITS,
196 /* enmStopBitCount */
197 RTSERIALPORTSTOPBITS_ONE
198};
199
200
201/**
202 * Initializes a TX buffer.
203 *
204 * @returns nothing.
205 * @param pSerBuf The serial buffer to initialize.
206 * @param cbTx Maximum number of bytes to transmit.
207 */
208static void serialTestTxBufInit(PSERIALTESTTXRXBUFCNT pSerBuf, size_t cbTx)
209{
210 pSerBuf->iCnt = 0;
211 pSerBuf->offBuf = 0;
212 pSerBuf->cbTxRxMax = 0;
213 pSerBuf->cbTxRxLeft = cbTx;
214 RT_ZERO(pSerBuf->abBuf);
215}
216
217
218/**
219 * Initializes a RX buffer.
220 *
221 * @returns nothing.
222 * @param pSerBuf The serial buffer to initialize.
223 * @param cbRx Maximum number of bytes to receive.
224 */
225static void serialTestRxBufInit(PSERIALTESTTXRXBUFCNT pSerBuf, size_t cbRx)
226{
227 pSerBuf->iCnt = 0;
228 pSerBuf->offBuf = 0;
229 pSerBuf->cbTxRxMax = sizeof(pSerBuf->abBuf);
230 pSerBuf->cbTxRxLeft = cbRx;
231 RT_ZERO(pSerBuf->abBuf);
232}
233
234
235/**
236 * Prepares the given TX buffer with data for sending it out.
237 *
238 * @returns nothing.
239 * @param pSerBuf The TX buffer pointer.
240 */
241static void serialTestTxBufPrepare(PSERIALTESTTXRXBUFCNT pSerBuf)
242{
243 /* Move the data to the front to make room at the end to fill. */
244 if (pSerBuf->offBuf)
245 {
246 memmove(&pSerBuf->abBuf[0], &pSerBuf->abBuf[pSerBuf->offBuf], sizeof(pSerBuf->abBuf) - pSerBuf->offBuf);
247 pSerBuf->offBuf = 0;
248 }
249
250 /* Fill up with data. */
251 uint32_t offData = 0;
252 while (pSerBuf->cbTxRxMax + sizeof(uint32_t) <= sizeof(pSerBuf->abBuf))
253 {
254 pSerBuf->iCnt++;
255 *(uint32_t *)&pSerBuf->abBuf[pSerBuf->offBuf + offData] = pSerBuf->iCnt;
256 pSerBuf->cbTxRxMax += sizeof(uint32_t);
257 offData += sizeof(uint32_t);
258 }
259}
260
261
262/**
263 * Sends a new batch of data from the TX buffer preapring new data if required.
264 *
265 * @returns IPRT status code.
266 * @param hSerialPort The serial port handle to send the data to.
267 * @param pSerBuf The TX buffer pointer.
268 */
269static int serialTestTxBufSend(RTSERIALPORT hSerialPort, PSERIALTESTTXRXBUFCNT pSerBuf)
270{
271 int rc = VINF_SUCCESS;
272
273 if (pSerBuf->cbTxRxLeft)
274 {
275 if (!pSerBuf->cbTxRxMax)
276 serialTestTxBufPrepare(pSerBuf);
277
278 size_t cbToWrite = RT_MIN(pSerBuf->cbTxRxMax, pSerBuf->cbTxRxLeft);
279 size_t cbWritten = 0;
280 rc = RTSerialPortWriteNB(hSerialPort, &pSerBuf->abBuf[pSerBuf->offBuf], cbToWrite, &cbWritten);
281 if (RT_SUCCESS(rc))
282 {
283 pSerBuf->cbTxRxMax -= cbWritten;
284 pSerBuf->offBuf += cbWritten;
285 pSerBuf->cbTxRxLeft -= cbWritten;
286 }
287 }
288
289 return rc;
290}
291
292
293/**
294 * Receives dat from the given serial port into the supplied RX buffer and does some validity checking.
295 *
296 * @returns IPRT status code.
297 * @param hSerialPort The serial port handle to receive data from.
298 * @param pSerBuf The RX buffer pointer.
299 */
300static int serialTestRxBufRecv(RTSERIALPORT hSerialPort, PSERIALTESTTXRXBUFCNT pSerBuf)
301{
302 int rc = VINF_SUCCESS;
303
304 if (pSerBuf->cbTxRxLeft)
305 {
306 size_t cbToRead = RT_MIN(pSerBuf->cbTxRxMax, pSerBuf->cbTxRxLeft);
307 size_t cbRead = 0;
308 rc = RTSerialPortReadNB(hSerialPort, &pSerBuf->abBuf[pSerBuf->offBuf], cbToRead, &cbRead);
309 if (RT_SUCCESS(rc))
310 {
311 pSerBuf->offBuf += cbRead;
312 pSerBuf->cbTxRxMax -= cbRead;
313 pSerBuf->cbTxRxLeft -= cbRead;
314 }
315 }
316
317 return rc;
318}
319
320
321/**
322 * Verifies the data in the given RX buffer for correct transmission.
323 *
324 * @returns nothing.
325 * @param hTest The test handle to report errors to.
326 * @param pSerBuf The RX buffer pointer.
327 * @param iCntTx The current TX counter value the RX buffer should never get ahead of.
328 */
329static void serialTestRxBufVerify(RTTEST hTest, PSERIALTESTTXRXBUFCNT pSerBuf, uint32_t iCntTx)
330{
331 uint32_t offRx = 0;
332
333 while (offRx + sizeof(uint32_t) < pSerBuf->offBuf)
334 {
335 uint32_t u32Val = *(uint32_t *)&pSerBuf->abBuf[offRx];
336 offRx += sizeof(uint32_t);
337
338 if (RT_UNLIKELY(u32Val != ++pSerBuf->iCnt))
339 RTTestFailed(hTest, "Data corruption/loss detected, expected counter value %u got %u\n",
340 pSerBuf->iCnt, u32Val);
341 }
342
343 if (RT_UNLIKELY(pSerBuf->iCnt > iCntTx))
344 RTTestFailed(hTest, "Overtook the send buffer, expected maximum counter value %u got %u\n",
345 iCntTx, pSerBuf->iCnt);
346
347 /* Remove processed data from the buffer and move the rest to the front. */
348 if (offRx)
349 {
350 memmove(&pSerBuf->abBuf[0], &pSerBuf->abBuf[offRx], sizeof(pSerBuf->abBuf) - offRx);
351 pSerBuf->offBuf -= offRx;
352 pSerBuf->cbTxRxMax += offRx;
353 }
354}
355
356
357DECLINLINE(bool) serialTestRndTrue(void)
358{
359 return RTRandAdvU32Ex(g_hRand, 0, 1) == 1;
360}
361
362/**
363 * Runs a simple read/write test.
364 *
365 * @returns IPRT status code.
366 * @param pSerialTest The serial test configuration.
367 */
368static DECLCALLBACK(int) serialTestRunReadWrite(PSERIALTEST pSerialTest)
369{
370 uint64_t tsStart = RTTimeMilliTS();
371 SERIALTESTTXRXBUFCNT SerBufTx;
372 SERIALTESTTXRXBUFCNT SerBufRx;
373
374 serialTestTxBufInit(&SerBufTx, g_cbTx);
375 serialTestRxBufInit(&SerBufRx, g_cbTx);
376
377 int rc = serialTestTxBufSend(pSerialTest->hSerialPort, &SerBufTx);
378 while ( RT_SUCCESS(rc)
379 && ( SerBufTx.cbTxRxLeft
380 || SerBufRx.cbTxRxLeft))
381 {
382 uint32_t fEvts = 0;
383 uint32_t fEvtsQuery = 0;
384 if (SerBufTx.cbTxRxLeft)
385 fEvtsQuery |= RTSERIALPORT_EVT_F_DATA_TX;
386 if (SerBufRx.cbTxRxLeft)
387 fEvtsQuery |= RTSERIALPORT_EVT_F_DATA_RX;
388
389 rc = RTSerialPortEvtPoll(pSerialTest->hSerialPort, fEvtsQuery, &fEvts, RT_INDEFINITE_WAIT);
390 if (RT_FAILURE(rc))
391 break;
392
393 if (fEvts & RTSERIALPORT_EVT_F_DATA_RX)
394 {
395 rc = serialTestRxBufRecv(pSerialTest->hSerialPort, &SerBufRx);
396 if (RT_FAILURE(rc))
397 break;
398
399 serialTestRxBufVerify(pSerialTest->hTest, &SerBufRx, SerBufTx.iCnt);
400 }
401 if ( RT_SUCCESS(rc)
402 && (fEvts & RTSERIALPORT_EVT_F_DATA_TX))
403 rc = serialTestTxBufSend(pSerialTest->hSerialPort, &SerBufTx);
404 }
405
406 uint64_t tsRuntime = RTTimeMilliTS() - tsStart;
407 tsRuntime /= 1000; /* Seconds */
408 RTTestValue(pSerialTest->hTest, "Throughput", g_cbTx / tsRuntime, RTTESTUNIT_BYTES_PER_SEC);
409
410 return rc;
411}
412
413
414/**
415 * Runs a simple write test without doing any verification.
416 *
417 * @returns IPRT status code.
418 * @param pSerialTest The serial test configuration.
419 */
420static DECLCALLBACK(int) serialTestRunWrite(PSERIALTEST pSerialTest)
421{
422 uint64_t tsStart = RTTimeMilliTS();
423 SERIALTESTTXRXBUFCNT SerBufTx;
424
425 serialTestTxBufInit(&SerBufTx, g_cbTx);
426
427 int rc = serialTestTxBufSend(pSerialTest->hSerialPort, &SerBufTx);
428 while ( RT_SUCCESS(rc)
429 && SerBufTx.cbTxRxLeft)
430 {
431 uint32_t fEvts = 0;
432
433 rc = RTSerialPortEvtPoll(pSerialTest->hSerialPort, RTSERIALPORT_EVT_F_DATA_TX, &fEvts, RT_INDEFINITE_WAIT);
434 if (RT_FAILURE(rc))
435 break;
436
437 if (fEvts & RTSERIALPORT_EVT_F_DATA_TX)
438 rc = serialTestTxBufSend(pSerialTest->hSerialPort, &SerBufTx);
439 }
440
441 uint64_t tsRuntime = RTTimeMilliTS() - tsStart;
442 tsRuntime /= 1000; /* Seconds */
443 RTTestValue(pSerialTest->hTest, "Throughput", g_cbTx / tsRuntime, RTTESTUNIT_BYTES_PER_SEC);
444
445 return rc;
446}
447
448
449/**
450 * Tests setting status lines and getting notified about status line changes.
451 *
452 * @returns IPRT status code.
453 * @param pSerialTest The serial test configuration.
454 */
455static DECLCALLBACK(int) serialTestRunStsLines(PSERIALTEST pSerialTest)
456{
457 int rc = VINF_SUCCESS;
458
459 if (g_enmMode == SERIALTESTMODE_LOOPBACK)
460 {
461 uint32_t fStsLinesQueriedOld = 0;
462
463 rc = RTSerialPortChgStatusLines(pSerialTest->hSerialPort,
464 RTSERIALPORT_CHG_STS_LINES_F_RTS | RTSERIALPORT_CHG_STS_LINES_F_DTR,
465 0);
466 if (RT_SUCCESS(rc))
467 {
468 rc = RTSerialPortQueryStatusLines(pSerialTest->hSerialPort, &fStsLinesQueriedOld);
469 if (RT_SUCCESS(rc))
470 {
471 /* Everything should be clear at this stage. */
472 if (!fStsLinesQueriedOld)
473 {
474 uint32_t fStsLinesSetOld = 0;
475
476 for (uint32_t i = 0; i < SERIALTEST_STS_LINE_TOGGLE_COUNT; i++)
477 {
478 uint32_t fStsLinesSet = 0;
479 uint32_t fStsLinesClear = 0;
480
481 /* Change RTS? */
482 if (serialTestRndTrue())
483 {
484 /* Clear, if set previously otherwise set it. */
485 if (fStsLinesSetOld & RTSERIALPORT_CHG_STS_LINES_F_RTS)
486 fStsLinesClear |= RTSERIALPORT_CHG_STS_LINES_F_RTS;
487 else
488 fStsLinesSet |= RTSERIALPORT_CHG_STS_LINES_F_RTS;
489 }
490
491 /* Change DTR? */
492 if (serialTestRndTrue())
493 {
494 /* Clear, if set previously otherwise set it. */
495 if (fStsLinesSetOld & RTSERIALPORT_CHG_STS_LINES_F_DTR)
496 fStsLinesClear |= RTSERIALPORT_CHG_STS_LINES_F_DTR;
497 else
498 fStsLinesSet |= RTSERIALPORT_CHG_STS_LINES_F_DTR;
499 }
500
501 rc = RTSerialPortChgStatusLines(pSerialTest->hSerialPort, fStsLinesClear, fStsLinesSet);
502 if (RT_FAILURE(rc))
503 {
504 RTTestFailed(g_hTest, "Changing status lines failed with %Rrc on iteration %u (fSet=%#x fClear=%#x)\n",
505 rc, i, fStsLinesSet, fStsLinesClear);
506 break;
507 }
508
509 /* Wait for status line monitor event. */
510 uint32_t fEvtsRecv = 0;
511 rc = RTSerialPortEvtPoll(pSerialTest->hSerialPort, RTSERIALPORT_EVT_F_STATUS_LINE_CHANGED,
512 &fEvtsRecv, RT_MS_1SEC);
513 if ( RT_FAILURE(rc)
514 && (rc != VERR_TIMEOUT && !fStsLinesSet && !fStsLinesClear))
515 {
516 RTTestFailed(g_hTest, "Waiting for status line change failed with %Rrc on iteration %u\n",
517 rc, i);
518 break;
519 }
520
521 uint32_t fStsLinesQueried = 0;
522 rc = RTSerialPortQueryStatusLines(pSerialTest->hSerialPort, &fStsLinesQueried);
523 if (RT_FAILURE(rc))
524 {
525 RTTestFailed(g_hTest, "Querying status lines failed with %Rrc on iteration %u\n",
526 rc, i);
527 break;
528 }
529
530 /* Compare expected and real result. */
531 if ( (fStsLinesQueried & RTSERIALPORT_STS_LINE_DSR)
532 != (fStsLinesQueriedOld & RTSERIALPORT_STS_LINE_DSR))
533 {
534 if ( (fStsLinesQueried & RTSERIALPORT_STS_LINE_DSR)
535 && !(fStsLinesSet & RTSERIALPORT_CHG_STS_LINES_F_DTR))
536 RTTestFailed(g_hTest, "DSR line got set when it shouldn't be on iteration %u\n", i);
537 else if ( !(fStsLinesQueried & RTSERIALPORT_STS_LINE_DSR)
538 && !(fStsLinesClear & RTSERIALPORT_CHG_STS_LINES_F_DTR))
539 RTTestFailed(g_hTest, "DSR line got cleared when it shouldn't be on iteration %u\n", i);
540 }
541 else if ( (fStsLinesSet & RTSERIALPORT_CHG_STS_LINES_F_DTR)
542 || (fStsLinesClear & RTSERIALPORT_CHG_STS_LINES_F_DTR))
543 RTTestFailed(g_hTest, "DSR line didn't change when it should have on iteration %u\n", i);
544
545 if ( (fStsLinesQueried & RTSERIALPORT_STS_LINE_DCD)
546 != (fStsLinesQueriedOld & RTSERIALPORT_STS_LINE_DCD))
547 {
548 if ( (fStsLinesQueried & RTSERIALPORT_STS_LINE_DCD)
549 && !(fStsLinesSet & RTSERIALPORT_CHG_STS_LINES_F_DTR))
550 RTTestFailed(g_hTest, "DCD line got set when it shouldn't be on iteration %u\n", i);
551 else if ( !(fStsLinesQueried & RTSERIALPORT_STS_LINE_DCD)
552 && !(fStsLinesClear & RTSERIALPORT_CHG_STS_LINES_F_DTR))
553 RTTestFailed(g_hTest, "DCD line got cleared when it shouldn't be on iteration %u\n", i);
554 }
555 else if ( (fStsLinesSet & RTSERIALPORT_CHG_STS_LINES_F_DTR)
556 || (fStsLinesClear & RTSERIALPORT_CHG_STS_LINES_F_DTR))
557 RTTestFailed(g_hTest, "DCD line didn't change when it should have on iteration %u\n", i);
558
559 if ( (fStsLinesQueried & RTSERIALPORT_STS_LINE_CTS)
560 != (fStsLinesQueriedOld & RTSERIALPORT_STS_LINE_CTS))
561 {
562 if ( (fStsLinesQueried & RTSERIALPORT_STS_LINE_CTS)
563 && !(fStsLinesSet & RTSERIALPORT_CHG_STS_LINES_F_RTS))
564 RTTestFailed(g_hTest, "CTS line got set when it shouldn't be on iteration %u\n", i);
565 else if ( !(fStsLinesQueried & RTSERIALPORT_STS_LINE_CTS)
566 && !(fStsLinesClear & RTSERIALPORT_CHG_STS_LINES_F_RTS))
567 RTTestFailed(g_hTest, "CTS line got cleared when it shouldn't be on iteration %u\n", i);
568 }
569 else if ( (fStsLinesSet & RTSERIALPORT_CHG_STS_LINES_F_RTS)
570 || (fStsLinesClear & RTSERIALPORT_CHG_STS_LINES_F_RTS))
571 RTTestFailed(g_hTest, "CTS line didn't change when it should have on iteration %u\n", i);
572
573 if (RTTestErrorCount(g_hTest) > 0)
574 break;
575
576 fStsLinesSetOld |= fStsLinesSet;
577 fStsLinesSetOld &= ~fStsLinesClear;
578 fStsLinesQueriedOld = fStsLinesQueried;
579 }
580 }
581 else
582 RTTestFailed(g_hTest, "Status lines active which should be clear (%#x, but expected %#x)\n",
583 fStsLinesQueriedOld, 0);
584 }
585 else
586 RTTestFailed(g_hTest, "Querying status lines failed with %Rrc\n", rc);
587 }
588 else
589 RTTestFailed(g_hTest, "Clearing status lines failed with %Rrc\n", rc);
590 }
591 else
592 rc = VERR_NOT_IMPLEMENTED;
593
594 return rc;
595}
596
597
598/**
599 * Returns an array of test descriptors get from the given string.
600 *
601 * @returns Pointer to the array of test descriptors.
602 * @param pszTests The string containing the tests separated with ':'.
603 */
604static PSERIALTESTDESC serialTestSelectFromCmdLine(const char *pszTests)
605{
606 size_t cTests = 1;
607
608 const char *pszNext = strchr(pszTests, ':');
609 while (pszNext)
610 {
611 pszNext++;
612 cTests++;
613 pszNext = strchr(pszNext, ':');
614 }
615
616 PSERIALTESTDESC paTests = (PSERIALTESTDESC)RTMemAllocZ((cTests + 1) * sizeof(SERIALTESTDESC));
617 if (RT_LIKELY(paTests))
618 {
619 uint32_t iTest = 0;
620
621 pszNext = strchr(pszTests, ':');
622 while (pszNext)
623 {
624 bool fFound = false;
625
626 pszNext++; /* Skip : character. */
627
628 for (unsigned i = 0; i < RT_ELEMENTS(g_aSerialTests); i++)
629 {
630 if (!RTStrNICmp(pszTests, g_aSerialTests[i].pszId, pszNext - pszTests - 1))
631 {
632 memcpy(&paTests[iTest], &g_aSerialTests[i], sizeof(SERIALTESTDESC));
633 fFound = true;
634 break;
635 }
636 }
637
638 if (RT_UNLIKELY(!fFound))
639 {
640 RTPrintf("Testcase \"%.*s\" not known\n", pszNext - pszTests - 1, pszTests);
641 RTMemFree(paTests);
642 return NULL;
643 }
644
645 pszTests = pszNext;
646 pszNext = strchr(pszTests, ':');
647 }
648
649 /* Fill last descriptor. */
650 bool fFound = false;
651 for (unsigned i = 0; i < RT_ELEMENTS(g_aSerialTests); i++)
652 {
653 if (!RTStrICmp(pszTests, g_aSerialTests[i].pszId))
654 {
655 memcpy(&paTests[iTest], &g_aSerialTests[i], sizeof(SERIALTESTDESC));
656 fFound = true;
657 break;
658 }
659 }
660
661 if (RT_UNLIKELY(!fFound))
662 {
663 RTPrintf("Testcase \"%s\" not known\n", pszTests);
664 RTMemFree(paTests);
665 paTests = NULL;
666 }
667 }
668 else
669 RTPrintf("Failed to allocate test descriptors for %u selected tests\n", cTests);
670
671 return paTests;
672}
673
674
675/**
676 * Shows tool usage text.
677 */
678static void serialTestUsage(PRTSTREAM pStrm)
679{
680 char szExec[RTPATH_MAX];
681 RTStrmPrintf(pStrm, "usage: %s [options]\n",
682 RTPathFilename(RTProcGetExecutablePath(szExec, sizeof(szExec))));
683 RTStrmPrintf(pStrm, "\n");
684 RTStrmPrintf(pStrm, "options: \n");
685
686
687 for (unsigned i = 0; i < RT_ELEMENTS(g_aCmdOptions); i++)
688 {
689 const char *pszHelp;
690 switch (g_aCmdOptions[i].iShort)
691 {
692 case 'h':
693 pszHelp = "Displays this help and exit";
694 break;
695 case 'd':
696 pszHelp = "Use the specified serial port device";
697 break;
698 case 'b':
699 pszHelp = "Use the given baudrate";
700 break;
701 case 'p':
702 pszHelp = "Use the given parity, valid modes are: none, even, odd, mark, space";
703 break;
704 case 'c':
705 pszHelp = "Use the given data bitcount, valid are: 5, 6, 7, 8";
706 break;
707 case 's':
708 pszHelp = "Use the given stop bitcount, valid are: 1, 1.5, 2";
709 break;
710 case 'm':
711 pszHelp = "Mode of the serial port, valid are: loopback, secondary, external";
712 break;
713 case 'l':
714 pszHelp = "Use the given serial port device as the secondary device";
715 break;
716 case 't':
717 pszHelp = "The tests to run separated by ':'";
718 break;
719 case 'x':
720 pszHelp = "Number of bytes to transmit during read/write tests";
721 break;
722 default:
723 pszHelp = "Option undocumented";
724 break;
725 }
726 char szOpt[256];
727 RTStrPrintf(szOpt, sizeof(szOpt), "%s, -%c", g_aCmdOptions[i].pszLong, g_aCmdOptions[i].iShort);
728 RTStrmPrintf(pStrm, " %-30s%s\n", szOpt, pszHelp);
729 }
730}
731
732
733int main(int argc, char *argv[])
734{
735 /*
736 * Init IPRT and globals.
737 */
738 int rc = RTTestInitAndCreate("SerialTest", &g_hTest);
739 if (rc)
740 return rc;
741
742 /*
743 * Default values.
744 */
745 const char *pszDevice = NULL;
746 const char *pszDeviceSecondary = NULL;
747 PSERIALTESTDESC paTests = NULL;
748
749 RTGETOPTUNION ValueUnion;
750 RTGETOPTSTATE GetState;
751 RTGetOptInit(&GetState, argc, argv, g_aCmdOptions, RT_ELEMENTS(g_aCmdOptions), 1, 0 /* fFlags */);
752 while ((rc = RTGetOpt(&GetState, &ValueUnion)))
753 {
754 switch (rc)
755 {
756 case 'h':
757 serialTestUsage(g_pStdOut);
758 return RTEXITCODE_SUCCESS;
759 case 'd':
760 pszDevice = ValueUnion.psz;
761 break;
762 case 'l':
763 pszDeviceSecondary = ValueUnion.psz;
764 break;
765 case 'b':
766 g_SerialPortCfg.uBaudRate = ValueUnion.u32;
767 break;
768 case 'p':
769 if (!RTStrICmp(ValueUnion.psz, "none"))
770 g_SerialPortCfg.enmParity = RTSERIALPORTPARITY_NONE;
771 else if (!RTStrICmp(ValueUnion.psz, "even"))
772 g_SerialPortCfg.enmParity = RTSERIALPORTPARITY_EVEN;
773 else if (!RTStrICmp(ValueUnion.psz, "odd"))
774 g_SerialPortCfg.enmParity = RTSERIALPORTPARITY_ODD;
775 else if (!RTStrICmp(ValueUnion.psz, "mark"))
776 g_SerialPortCfg.enmParity = RTSERIALPORTPARITY_MARK;
777 else if (!RTStrICmp(ValueUnion.psz, "space"))
778 g_SerialPortCfg.enmParity = RTSERIALPORTPARITY_SPACE;
779 else
780 {
781 RTPrintf("Unknown parity \"%s\" given\n", ValueUnion.psz);
782 return RTEXITCODE_FAILURE;
783 }
784 break;
785 case 'c':
786 if (!RTStrICmp(ValueUnion.psz, "5"))
787 g_SerialPortCfg.enmDataBitCount = RTSERIALPORTDATABITS_5BITS;
788 else if (!RTStrICmp(ValueUnion.psz, "6"))
789 g_SerialPortCfg.enmDataBitCount = RTSERIALPORTDATABITS_6BITS;
790 else if (!RTStrICmp(ValueUnion.psz, "7"))
791 g_SerialPortCfg.enmDataBitCount = RTSERIALPORTDATABITS_7BITS;
792 else if (!RTStrICmp(ValueUnion.psz, "8"))
793 g_SerialPortCfg.enmDataBitCount = RTSERIALPORTDATABITS_8BITS;
794 else
795 {
796 RTPrintf("Unknown data bitcount \"%s\" given\n", ValueUnion.psz);
797 return RTEXITCODE_FAILURE;
798 }
799 break;
800 case 's':
801 if (!RTStrICmp(ValueUnion.psz, "1"))
802 g_SerialPortCfg.enmStopBitCount = RTSERIALPORTSTOPBITS_ONE;
803 else if (!RTStrICmp(ValueUnion.psz, "1.5"))
804 g_SerialPortCfg.enmStopBitCount = RTSERIALPORTSTOPBITS_ONEPOINTFIVE;
805 else if (!RTStrICmp(ValueUnion.psz, "2"))
806 g_SerialPortCfg.enmStopBitCount = RTSERIALPORTSTOPBITS_TWO;
807 else
808 {
809 RTPrintf("Unknown stop bitcount \"%s\" given\n", ValueUnion.psz);
810 return RTEXITCODE_FAILURE;
811 }
812 break;
813 case 'm':
814 if (!RTStrICmp(ValueUnion.psz, "loopback"))
815 g_enmMode = SERIALTESTMODE_LOOPBACK;
816 else if (!RTStrICmp(ValueUnion.psz, "secondary"))
817 g_enmMode = SERIALTESTMODE_SECONDARY;
818 else if (!RTStrICmp(ValueUnion.psz, "external"))
819 g_enmMode = SERIALTESTMODE_EXTERNAL;
820 else
821 {
822 RTPrintf("Unknown serial test mode \"%s\" given\n", ValueUnion.psz);
823 return RTEXITCODE_FAILURE;
824 }
825 break;
826 case 't':
827 paTests = serialTestSelectFromCmdLine(ValueUnion.psz);
828 if (!paTests)
829 return RTEXITCODE_FAILURE;
830 break;
831 case 'x':
832 g_cbTx = ValueUnion.u32;
833 break;
834 default:
835 return RTGetOptPrintError(rc, &ValueUnion);
836 }
837 }
838
839 if (g_enmMode == SERIALTESTMODE_SECONDARY && !pszDeviceSecondary)
840 {
841 RTPrintf("Mode set to secondary device but no secondary device given\n");
842 return RTEXITCODE_FAILURE;
843 }
844
845 if (!paTests)
846 {
847 /* Select all. */
848 paTests = (PSERIALTESTDESC)RTMemAllocZ((RT_ELEMENTS(g_aSerialTests) + 1) * sizeof(SERIALTESTDESC));
849 if (RT_UNLIKELY(!paTests))
850 {
851 RTPrintf("Failed to allocate memory for test descriptors\n");
852 return RTEXITCODE_FAILURE;
853 }
854 memcpy(paTests, &g_aSerialTests[0], RT_ELEMENTS(g_aSerialTests) * sizeof(SERIALTESTDESC));
855 }
856
857 rc = RTRandAdvCreateParkMiller(&g_hRand);
858 if (RT_FAILURE(rc))
859 {
860 RTPrintf("Failed to create random number generator: %Rrc\n", rc);
861 return RTEXITCODE_FAILURE;
862 }
863
864 rc = RTRandAdvSeed(g_hRand, UINT64_C(0x123456789abcdef));
865 AssertRC(rc);
866
867 /*
868 * Start testing.
869 */
870 RTTestBanner(g_hTest);
871
872 if (pszDevice)
873 {
874 uint32_t fFlags = RTSERIALPORT_OPEN_F_READ
875 | RTSERIALPORT_OPEN_F_WRITE
876 | RTSERIALPORT_OPEN_F_SUPPORT_STATUS_LINE_MONITORING;
877
878 RTTestSub(g_hTest, "Opening device");
879 rc = RTSerialPortOpen(&g_hSerialPort, pszDevice, fFlags);
880 if (RT_SUCCESS(rc))
881 {
882 if (g_enmMode == SERIALTESTMODE_SECONDARY)
883 {
884 RTTestSub(g_hTest, "Opening secondary device");
885 rc = RTSerialPortOpen(&g_hSerialPortSecondary, pszDeviceSecondary, fFlags);
886 if (RT_FAILURE(rc))
887 RTTestFailed(g_hTest, "Opening secondary device \"%s\" failed with %Rrc\n", pszDevice, rc);
888 }
889
890 if (RT_SUCCESS(rc))
891 {
892 RTTestSub(g_hTest, "Setting serial port configuration");
893
894 rc = RTSerialPortCfgSet(g_hSerialPort, &g_SerialPortCfg ,NULL);
895 if (RT_SUCCESS(rc))
896 {
897 if (g_enmMode == SERIALTESTMODE_SECONDARY)
898 {
899 RTTestSub(g_hTest, "Setting serial port configuration for secondary device");
900 rc = RTSerialPortCfgSet(g_hSerialPortSecondary, &g_SerialPortCfg, NULL);
901 if (RT_FAILURE(rc))
902 RTTestFailed(g_hTest, "Setting configuration of secondary device \"%s\" failed with %Rrc\n", pszDevice, rc);
903 }
904
905 if (RT_SUCCESS(rc))
906 {
907 SERIALTEST Test;
908 PSERIALTESTDESC pTest = &paTests[0];
909
910 Test.hTest = g_hTest;
911 Test.hSerialPort = g_hSerialPort;
912 Test.pSerialCfg = &g_SerialPortCfg;
913
914 while (pTest->pszId)
915 {
916 RTTestSub(g_hTest, pTest->pszDesc);
917 rc = pTest->pfnRun(&Test);
918 if ( RT_FAILURE(rc)
919 || RTTestErrorCount(g_hTest) > 0)
920 RTTestFailed(g_hTest, "Running test \"%s\" failed (%Rrc, cErrors=%u)\n",
921 pTest->pszId, rc, RTTestErrorCount(g_hTest));
922
923 RTTestSubDone(g_hTest);
924 pTest++;
925 }
926 }
927 }
928 else
929 RTTestFailed(g_hTest, "Setting configuration of device \"%s\" failed with %Rrc\n", pszDevice, rc);
930
931 RTSerialPortClose(g_hSerialPort);
932 }
933 }
934 else
935 RTTestFailed(g_hTest, "Opening device \"%s\" failed with %Rrc\n", pszDevice, rc);
936 }
937 else
938 RTTestFailed(g_hTest, "No device given on command line\n");
939
940 RTRandAdvDestroy(g_hRand);
941 RTMemFree(paTests);
942 RTEXITCODE rcExit = RTTestSummaryAndDestroy(g_hTest);
943 return rcExit;
944}
945
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