VirtualBox

source: vbox/trunk/src/VBox/Devices/Serial/DrvHostSerial.cpp@ 26270

Last change on this file since 26270 was 26173, checked in by vboxsync, 15 years ago

PDM: s/pCfgHandle/pCfg/g - part 2.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 50.5 KB
Line 
1/* $Id: DrvHostSerial.cpp 26173 2010-02-02 21:11:09Z vboxsync $ */
2/** @file
3 * VBox stream I/O devices: Host serial driver
4 *
5 * Contributed by: Alexander Eichner
6 */
7
8/*
9 * Copyright (C) 2006-2007 Sun Microsystems, Inc.
10 *
11 * This file is part of VirtualBox Open Source Edition (OSE), as
12 * available from http://www.virtualbox.org. This file is free software;
13 * you can redistribute it and/or modify it under the terms of the GNU
14 * General Public License (GPL) as published by the Free Software
15 * Foundation, in version 2 as it comes in the "COPYING" file of the
16 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
17 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
18 *
19 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
20 * Clara, CA 95054 USA or visit http://www.sun.com if you need
21 * additional information or have any questions.
22 */
23
24
25
26/*******************************************************************************
27* Header Files *
28*******************************************************************************/
29#define LOG_GROUP LOG_GROUP_DRV_HOST_SERIAL
30#include <VBox/pdm.h>
31#include <VBox/err.h>
32
33#include <VBox/log.h>
34#include <iprt/asm.h>
35#include <iprt/assert.h>
36#include <iprt/file.h>
37#include <iprt/mem.h>
38#include <iprt/semaphore.h>
39#include <iprt/uuid.h>
40
41#if defined(RT_OS_LINUX) || defined(RT_OS_DARWIN) || defined(RT_OS_SOLARIS) || defined(RT_OS_FREEBSD)
42# include <errno.h>
43# ifdef RT_OS_SOLARIS
44# include <sys/termios.h>
45# else
46# include <termios.h>
47# endif
48# include <sys/types.h>
49# include <fcntl.h>
50# include <string.h>
51# include <unistd.h>
52# ifdef RT_OS_DARWIN
53# include <sys/select.h>
54# else
55# include <sys/poll.h>
56# endif
57# include <sys/ioctl.h>
58# include <pthread.h>
59
60# ifdef RT_OS_LINUX
61/*
62 * TIOCM_LOOP is not defined in the above header files for some reason but in asm/termios.h.
63 * But inclusion of this file however leads to compilation errors because of redefinition of some
64 * structs. Thatswhy it is defined here until a better solution is found.
65 */
66# ifndef TIOCM_LOOP
67# define TIOCM_LOOP 0x8000
68# endif
69# endif /* linux */
70
71#elif defined(RT_OS_WINDOWS)
72# include <Windows.h>
73#endif
74
75#include "../Builtins.h"
76
77
78/** Size of the send fifo queue (in bytes) */
79#ifdef RT_OS_DARWIN
80 /** @todo This is really desperate, but it seriously looks like
81 * the data is arriving way too fast for us to push over. 9600
82 * baud and zoc reports sending at 12000+ chars/sec... */
83# define CHAR_MAX_SEND_QUEUE 0x400
84# define CHAR_MAX_SEND_QUEUE_MASK 0x3ff
85#else
86# define CHAR_MAX_SEND_QUEUE 0x80
87# define CHAR_MAX_SEND_QUEUE_MASK 0x7f
88#endif
89
90/*******************************************************************************
91* Structures and Typedefs *
92*******************************************************************************/
93
94/**
95 * Char driver instance data.
96 *
97 * @implements PDMICHARCONNECTOR
98 */
99typedef struct DRVHOSTSERIAL
100{
101 /** Pointer to the driver instance structure. */
102 PPDMDRVINS pDrvIns;
103 /** Pointer to the char port interface of the driver/device above us. */
104 PPDMICHARPORT pDrvCharPort;
105 /** Our char interface. */
106 PDMICHARCONNECTOR ICharConnector;
107 /** Receive thread. */
108 PPDMTHREAD pRecvThread;
109 /** Send thread. */
110 PPDMTHREAD pSendThread;
111 /** Status lines monitor thread. */
112 PPDMTHREAD pMonitorThread;
113 /** Send event semephore */
114 RTSEMEVENT SendSem;
115
116 /** the device path */
117 char *pszDevicePath;
118
119#if defined(RT_OS_LINUX) || defined(RT_OS_DARWIN) || defined(RT_OS_SOLARIS) || defined(RT_OS_FREEBSD)
120 /** the device handle */
121 RTFILE DeviceFile;
122# ifdef RT_OS_DARWIN
123 /** The device handle used for reading.
124 * Used to prevent the read selecto from blocking the writes. */
125 RTFILE DeviceFileR;
126# endif
127 /** The read end of the control pipe */
128 RTFILE WakeupPipeR;
129 /** The write end of the control pipe */
130 RTFILE WakeupPipeW;
131# ifndef RT_OS_LINUX
132 /** The current line status.
133 * Used by the polling version of drvHostSerialMonitorThread. */
134 int fStatusLines;
135# endif
136#elif defined(RT_OS_WINDOWS)
137 /** the device handle */
138 HANDLE hDeviceFile;
139 /** The event semaphore for waking up the receive thread */
140 HANDLE hHaltEventSem;
141 /** The event semaphore for overlapped receiving */
142 HANDLE hEventRecv;
143 /** For overlapped receiving */
144 OVERLAPPED overlappedRecv;
145 /** The event semaphore for overlapped sending */
146 HANDLE hEventSend;
147 /** For overlapped sending */
148 OVERLAPPED overlappedSend;
149#endif
150
151 /** Internal send FIFO queue */
152 uint8_t volatile aSendQueue[CHAR_MAX_SEND_QUEUE];
153 uint32_t volatile iSendQueueHead;
154 uint32_t volatile iSendQueueTail;
155
156 /** Read/write statistics */
157 STAMCOUNTER StatBytesRead;
158 STAMCOUNTER StatBytesWritten;
159#ifdef RT_OS_DARWIN
160 /** The number of bytes we've dropped because the send queue
161 * was full. */
162 STAMCOUNTER StatSendOverflows;
163#endif
164} DRVHOSTSERIAL, *PDRVHOSTSERIAL;
165
166
167/** Converts a pointer to DRVCHAR::ICharConnector to a PDRVHOSTSERIAL. */
168#define PDMICHAR_2_DRVHOSTSERIAL(pInterface) RT_FROM_MEMBER(pInterface, DRVHOSTSERIAL, ICharConnector)
169
170
171/* -=-=-=-=- IBase -=-=-=-=- */
172
173/**
174 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
175 */
176static DECLCALLBACK(void *) drvHostSerialQueryInterface(PPDMIBASE pInterface, const char *pszIID)
177{
178 PPDMDRVINS pDrvIns = PDMIBASE_2_PDMDRV(pInterface);
179 PDRVHOSTSERIAL pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTSERIAL);
180
181 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pDrvIns->IBase);
182 PDMIBASE_RETURN_INTERFACE(pszIID, PDMICHARCONNECTOR, &pThis->ICharConnector);
183 return NULL;
184}
185
186
187/* -=-=-=-=- ICharConnector -=-=-=-=- */
188
189/** @copydoc PDMICHARCONNECTOR::pfnWrite */
190static DECLCALLBACK(int) drvHostSerialWrite(PPDMICHARCONNECTOR pInterface, const void *pvBuf, size_t cbWrite)
191{
192 PDRVHOSTSERIAL pThis = PDMICHAR_2_DRVHOSTSERIAL(pInterface);
193 const uint8_t *pbBuffer = (const uint8_t *)pvBuf;
194
195 LogFlow(("%s: pvBuf=%#p cbWrite=%d\n", __FUNCTION__, pvBuf, cbWrite));
196
197 for (uint32_t i = 0; i < cbWrite; i++)
198 {
199#ifdef RT_OS_DARWIN /* don't wanna break the others here. */
200 uint32_t iHead = ASMAtomicUoReadU32(&pThis->iSendQueueHead);
201 uint32_t iTail = ASMAtomicUoReadU32(&pThis->iSendQueueTail);
202 int32_t cbAvail = iTail > iHead
203 ? iTail - iHead - 1
204 : CHAR_MAX_SEND_QUEUE - (iHead - iTail) - 1;
205 if (cbAvail <= 0)
206 {
207# ifdef DEBUG
208 uint64_t volatile u64Now = RTTimeNanoTS(); NOREF(u64Now);
209# endif
210 Log(("%s: dropping %d chars (cbAvail=%d iHead=%d iTail=%d)\n", __FUNCTION__, cbWrite - i , cbAvail, iHead, iTail));
211 STAM_COUNTER_INC(&pThis->StatSendOverflows);
212 break;
213 }
214
215 pThis->aSendQueue[iHead] = pbBuffer[i];
216 STAM_COUNTER_INC(&pThis->StatBytesWritten);
217 ASMAtomicWriteU32(&pThis->iSendQueueHead, (iHead + 1) & CHAR_MAX_SEND_QUEUE_MASK);
218 if (cbAvail < CHAR_MAX_SEND_QUEUE / 4)
219 {
220 RTSemEventSignal(pThis->SendSem);
221 RTThreadYield();
222 }
223#else /* old code */
224 uint32_t idx = ASMAtomicUoReadU32(&pThis->iSendQueueHead);
225
226 pThis->aSendQueue[idx] = pbBuffer[i];
227 idx = (idx + 1) & CHAR_MAX_SEND_QUEUE_MASK;
228
229 STAM_COUNTER_INC(&pThis->StatBytesWritten);
230 ASMAtomicWriteU32(&pThis->iSendQueueHead, idx);
231#endif /* old code */
232 }
233 RTSemEventSignal(pThis->SendSem);
234 return VINF_SUCCESS;
235}
236
237static DECLCALLBACK(int) drvHostSerialSetParameters(PPDMICHARCONNECTOR pInterface, unsigned Bps, char chParity, unsigned cDataBits, unsigned cStopBits)
238{
239 PDRVHOSTSERIAL pThis = PDMICHAR_2_DRVHOSTSERIAL(pInterface);
240#if defined(RT_OS_LINUX) || defined(RT_OS_DARWIN) || defined(RT_OS_SOLARIS) || defined(RT_OS_FREEBSD)
241 struct termios *termiosSetup;
242 int baud_rate;
243#elif defined(RT_OS_WINDOWS)
244 LPDCB comSetup;
245#endif
246
247 LogFlow(("%s: Bps=%u chParity=%c cDataBits=%u cStopBits=%u\n", __FUNCTION__, Bps, chParity, cDataBits, cStopBits));
248
249#if defined(RT_OS_LINUX) || defined(RT_OS_DARWIN) || defined(RT_OS_SOLARIS)
250 termiosSetup = (struct termios *)RTMemTmpAllocZ(sizeof(struct termios));
251
252 /* Enable receiver */
253 termiosSetup->c_cflag |= (CLOCAL | CREAD);
254
255 switch (Bps) {
256 case 50:
257 baud_rate = B50;
258 break;
259 case 75:
260 baud_rate = B75;
261 break;
262 case 110:
263 baud_rate = B110;
264 break;
265 case 134:
266 baud_rate = B134;
267 break;
268 case 150:
269 baud_rate = B150;
270 break;
271 case 200:
272 baud_rate = B200;
273 break;
274 case 300:
275 baud_rate = B300;
276 break;
277 case 600:
278 baud_rate = B600;
279 break;
280 case 1200:
281 baud_rate = B1200;
282 break;
283 case 1800:
284 baud_rate = B1800;
285 break;
286 case 2400:
287 baud_rate = B2400;
288 break;
289 case 4800:
290 baud_rate = B4800;
291 break;
292 case 9600:
293 baud_rate = B9600;
294 break;
295 case 19200:
296 baud_rate = B19200;
297 break;
298 case 38400:
299 baud_rate = B38400;
300 break;
301 case 57600:
302 baud_rate = B57600;
303 break;
304 case 115200:
305 baud_rate = B115200;
306 break;
307 default:
308 baud_rate = B9600;
309 }
310
311 cfsetispeed(termiosSetup, baud_rate);
312 cfsetospeed(termiosSetup, baud_rate);
313
314 switch (chParity) {
315 case 'E':
316 termiosSetup->c_cflag |= PARENB;
317 break;
318 case 'O':
319 termiosSetup->c_cflag |= (PARENB | PARODD);
320 break;
321 case 'N':
322 break;
323 default:
324 break;
325 }
326
327 switch (cDataBits) {
328 case 5:
329 termiosSetup->c_cflag |= CS5;
330 break;
331 case 6:
332 termiosSetup->c_cflag |= CS6;
333 break;
334 case 7:
335 termiosSetup->c_cflag |= CS7;
336 break;
337 case 8:
338 termiosSetup->c_cflag |= CS8;
339 break;
340 default:
341 break;
342 }
343
344 switch (cStopBits) {
345 case 2:
346 termiosSetup->c_cflag |= CSTOPB;
347 default:
348 break;
349 }
350
351 /* set serial port to raw input */
352 termiosSetup->c_lflag &= ~(ICANON | ECHO | ECHOE | ECHONL | ECHOK | ISIG | IEXTEN);
353
354 tcsetattr(pThis->DeviceFile, TCSANOW, termiosSetup);
355 RTMemTmpFree(termiosSetup);
356#elif defined(RT_OS_WINDOWS)
357 comSetup = (LPDCB)RTMemTmpAllocZ(sizeof(DCB));
358
359 comSetup->DCBlength = sizeof(DCB);
360
361 switch (Bps) {
362 case 110:
363 comSetup->BaudRate = CBR_110;
364 break;
365 case 300:
366 comSetup->BaudRate = CBR_300;
367 break;
368 case 600:
369 comSetup->BaudRate = CBR_600;
370 break;
371 case 1200:
372 comSetup->BaudRate = CBR_1200;
373 break;
374 case 2400:
375 comSetup->BaudRate = CBR_2400;
376 break;
377 case 4800:
378 comSetup->BaudRate = CBR_4800;
379 break;
380 case 9600:
381 comSetup->BaudRate = CBR_9600;
382 break;
383 case 14400:
384 comSetup->BaudRate = CBR_14400;
385 break;
386 case 19200:
387 comSetup->BaudRate = CBR_19200;
388 break;
389 case 38400:
390 comSetup->BaudRate = CBR_38400;
391 break;
392 case 57600:
393 comSetup->BaudRate = CBR_57600;
394 break;
395 case 115200:
396 comSetup->BaudRate = CBR_115200;
397 break;
398 default:
399 comSetup->BaudRate = CBR_9600;
400 }
401
402 comSetup->fBinary = TRUE;
403 comSetup->fOutxCtsFlow = FALSE;
404 comSetup->fOutxDsrFlow = FALSE;
405 comSetup->fDtrControl = DTR_CONTROL_DISABLE;
406 comSetup->fDsrSensitivity = FALSE;
407 comSetup->fTXContinueOnXoff = TRUE;
408 comSetup->fOutX = FALSE;
409 comSetup->fInX = FALSE;
410 comSetup->fErrorChar = FALSE;
411 comSetup->fNull = FALSE;
412 comSetup->fRtsControl = RTS_CONTROL_DISABLE;
413 comSetup->fAbortOnError = FALSE;
414 comSetup->wReserved = 0;
415 comSetup->XonLim = 5;
416 comSetup->XoffLim = 5;
417 comSetup->ByteSize = cDataBits;
418
419 switch (chParity) {
420 case 'E':
421 comSetup->Parity = EVENPARITY;
422 break;
423 case 'O':
424 comSetup->Parity = ODDPARITY;
425 break;
426 case 'N':
427 comSetup->Parity = NOPARITY;
428 break;
429 default:
430 break;
431 }
432
433 switch (cStopBits) {
434 case 1:
435 comSetup->StopBits = ONESTOPBIT;
436 break;
437 case 2:
438 comSetup->StopBits = TWOSTOPBITS;
439 break;
440 default:
441 break;
442 }
443
444 comSetup->XonChar = 0;
445 comSetup->XoffChar = 0;
446 comSetup->ErrorChar = 0;
447 comSetup->EofChar = 0;
448 comSetup->EvtChar = 0;
449
450 SetCommState(pThis->hDeviceFile, comSetup);
451 RTMemTmpFree(comSetup);
452#endif /* RT_OS_WINDOWS */
453
454 return VINF_SUCCESS;
455}
456
457/* -=-=-=-=- receive thread -=-=-=-=- */
458
459/**
460 * Send thread loop.
461 *
462 * @returns VINF_SUCCESS.
463 * @param ThreadSelf Thread handle to this thread.
464 * @param pvUser User argument.
465 */
466static DECLCALLBACK(int) drvHostSerialSendThread(PPDMDRVINS pDrvIns, PPDMTHREAD pThread)
467{
468 PDRVHOSTSERIAL pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTSERIAL);
469
470 if (pThread->enmState == PDMTHREADSTATE_INITIALIZING)
471 return VINF_SUCCESS;
472
473#ifdef RT_OS_WINDOWS
474 /* Make sure that the halt event semaphore is reset. */
475 DWORD dwRet = WaitForSingleObject(pThis->hHaltEventSem, 0);
476
477 HANDLE haWait[2];
478 haWait[0] = pThis->hEventSend;
479 haWait[1] = pThis->hHaltEventSem;
480#endif
481
482 while (pThread->enmState == PDMTHREADSTATE_RUNNING)
483 {
484 int rc = RTSemEventWait(pThis->SendSem, RT_INDEFINITE_WAIT);
485 if (RT_FAILURE(rc))
486 break;
487
488 /*
489 * Write the character to the host device.
490 */
491#ifdef RT_OS_DARWIN
492 while (pThread->enmState == PDMTHREADSTATE_RUNNING)
493 {
494 /* copy the send queue so we get a linear buffer with the maximal size. */
495 uint8_t abBuf[sizeof(pThis->aSendQueue)];
496 uint32_t cb = 0;
497 uint32_t iTail = ASMAtomicUoReadU32(&pThis->iSendQueueTail);
498 uint32_t iHead = ASMAtomicUoReadU32(&pThis->iSendQueueHead);
499 if (iTail == iHead)
500 break;
501 do
502 {
503 abBuf[cb++] = pThis->aSendQueue[iTail];
504 iTail = (iTail + 1) & CHAR_MAX_SEND_QUEUE_MASK;
505 } while (iTail != iHead);
506
507 ASMAtomicWriteU32(&pThis->iSendQueueTail, iTail);
508
509 /* write it. */
510# ifdef DEBUG
511 uint64_t volatile u64Now = RTTimeNanoTS(); NOREF(u64Now);
512# endif
513# if defined(RT_OS_LINUX) || defined(RT_OS_DARWIN) || defined(RT_OS_SOLARIS) || defined(RT_OS_FREEBSD)
514
515 size_t cbWritten;
516 rc = RTFileWrite(pThis->DeviceFile, abBuf, cb, &cbWritten);
517 if (rc == VERR_TRY_AGAIN)
518 cbWritten = 0;
519 if (cbWritten < cb && (RT_SUCCESS(rc) || rc == VERR_TRY_AGAIN))
520 {
521 /* ok, block till the device is ready for more (O_NONBLOCK) effect. */
522 rc = VINF_SUCCESS;
523 uint8_t const *pbSrc = &abBuf[0];
524 while (pThread->enmState == PDMTHREADSTATE_RUNNING)
525 {
526 /* advance */
527 cb -= cbWritten;
528 pbSrc += cbWritten;
529
530 /* wait */
531 fd_set WrSet;
532 FD_ZERO(&WrSet);
533 FD_SET(pThis->DeviceFile, &WrSet);
534 fd_set XcptSet;
535 FD_ZERO(&XcptSet);
536 FD_SET(pThis->DeviceFile, &XcptSet);
537 rc = select(pThis->DeviceFile + 1, NULL, &WrSet, &XcptSet, NULL);
538 /** @todo check rc? */
539
540 /* try write more */
541 rc = RTFileWrite(pThis->DeviceFile, pbSrc, cb, &cbWritten);
542 if (rc == VERR_TRY_AGAIN)
543 cbWritten = 0;
544 else if (RT_FAILURE(rc))
545 break;
546 else if (cbWritten >= cb)
547 break;
548 rc = VINF_SUCCESS;
549 } /* wait/write loop */
550 }
551
552# elif defined(RT_OS_WINDOWS)
553 /* perform an overlapped write operation. */
554 DWORD cbWritten;
555 memset(&pThis->overlappedSend, 0, sizeof(pThis->overlappedSend));
556 pThis->overlappedSend.hEvent = pThis->hEventSend;
557 if (!WriteFile(pThis->hDeviceFile, abBuf, cb, &cbWritten, &pThis->overlappedSend))
558 {
559 dwRet = GetLastError();
560 if (dwRet == ERROR_IO_PENDING)
561 {
562 /*
563 * write blocked, wait for completion or wakeup...
564 */
565 dwRet = WaitForMultipleObjects(2, haWait, FALSE, INFINITE);
566 if (dwRet != WAIT_OBJECT_0)
567 {
568 AssertMsg(pThread->enmState != PDMTHREADSTATE_RUNNING, ("The halt event sempahore is set but the thread is still in running state\n"));
569 break;
570 }
571 }
572 else
573 rc = RTErrConvertFromWin32(dwRet);
574 }
575
576# endif /* RT_OS_WINDOWS */
577 if (RT_FAILURE(rc))
578 {
579 LogRel(("HostSerial#%d: Serial Write failed with %Rrc; terminating send thread\n", pDrvIns->iInstance, rc));
580 return rc;
581 }
582 } /* write loop */
583
584#else /* old code */
585 uint32_t iTail;
586 while ( pThread->enmState == PDMTHREADSTATE_RUNNING
587 && (iTail = ASMAtomicUoReadU32(&pThis->iSendQueueTail)) != ASMAtomicUoReadU32(&pThis->iSendQueueHead))
588 {
589 /** @todo process more than one byte? */
590 unsigned cbProcessed = 1;
591 uint8_t abBuf[1];
592 abBuf[0] = pThis->aSendQueue[iTail];
593
594# if defined(RT_OS_LINUX) || defined(RT_OS_DARWIN) || defined(RT_OS_SOLARIS) || defined(RT_OS_FREEBSD)
595
596 rc = RTFileWrite(pThis->DeviceFile, abBuf, cbProcessed, NULL);
597
598# elif defined(RT_OS_WINDOWS)
599
600 DWORD cbBytesWritten;
601 memset(&pThis->overlappedSend, 0, sizeof(pThis->overlappedSend));
602 pThis->overlappedSend.hEvent = pThis->hEventSend;
603
604 if (!WriteFile(pThis->hDeviceFile, abBuf, cbProcessed, &cbBytesWritten, &pThis->overlappedSend))
605 {
606 dwRet = GetLastError();
607 if (dwRet == ERROR_IO_PENDING)
608 {
609 /*
610 * write blocked, wait ...
611 */
612 dwRet = WaitForMultipleObjects(2, haWait, FALSE, INFINITE);
613 if (dwRet != WAIT_OBJECT_0)
614 {
615 AssertMsg(pThread->enmState != PDMTHREADSTATE_RUNNING, ("The halt event sempahore is set but the thread is still in running state\n"));
616 break;
617 }
618 }
619 else
620 rc = RTErrConvertFromWin32(dwRet);
621 }
622
623# endif
624
625 if (RT_SUCCESS(rc))
626 {
627 Assert(cbProcessed == 1);
628 ASMAtomicWriteU32(&pThis->iSendQueueTail, (iTail + 1) & CHAR_MAX_SEND_QUEUE_MASK);
629 }
630 else if (RT_FAILURE(rc))
631 {
632 LogRel(("HostSerial#%d: Serial Write failed with %Rrc; terminating send thread\n", pDrvIns->iInstance, rc));
633 return rc;
634 }
635 }
636#endif /* old code */
637 }
638
639 return VINF_SUCCESS;
640}
641
642/**
643 * Unblock the send thread so it can respond to a state change.
644 *
645 * @returns a VBox status code.
646 * @param pDrvIns The driver instance.
647 * @param pThread The send thread.
648 */
649static DECLCALLBACK(int) drvHostSerialWakeupSendThread(PPDMDRVINS pDrvIns, PPDMTHREAD pThread)
650{
651 PDRVHOSTSERIAL pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTSERIAL);
652 int rc;
653
654 rc = RTSemEventSignal(pThis->SendSem);
655 if (RT_FAILURE(rc))
656 return rc;
657
658#ifdef RT_OS_WINDOWS
659 if (!SetEvent(pThis->hHaltEventSem))
660 return RTErrConvertFromWin32(GetLastError());
661#endif
662
663 return VINF_SUCCESS;
664}
665
666/* -=-=-=-=- receive thread -=-=-=-=- */
667
668/**
669 * Receive thread loop.
670 *
671 * This thread pushes data from the host serial device up the driver
672 * chain toward the serial device.
673 *
674 * @returns VINF_SUCCESS.
675 * @param ThreadSelf Thread handle to this thread.
676 * @param pvUser User argument.
677 */
678static DECLCALLBACK(int) drvHostSerialRecvThread(PPDMDRVINS pDrvIns, PPDMTHREAD pThread)
679{
680 PDRVHOSTSERIAL pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTSERIAL);
681 uint8_t abBuffer[256];
682 uint8_t *pbBuffer = NULL;
683 size_t cbRemaining = 0; /* start by reading host data */
684 int rc = VINF_SUCCESS;
685 int rcThread = VINF_SUCCESS;
686
687 if (pThread->enmState == PDMTHREADSTATE_INITIALIZING)
688 return VINF_SUCCESS;
689
690#ifdef RT_OS_WINDOWS
691 /* Make sure that the halt event semaphore is reset. */
692 DWORD dwRet = WaitForSingleObject(pThis->hHaltEventSem, 0);
693
694 HANDLE ahWait[2];
695 ahWait[0] = pThis->hEventRecv;
696 ahWait[1] = pThis->hHaltEventSem;
697#endif
698
699 while (pThread->enmState == PDMTHREADSTATE_RUNNING)
700 {
701 if (!cbRemaining)
702 {
703 /* Get a block of data from the host serial device. */
704
705#if defined(RT_OS_DARWIN) /* poll is broken on x86 darwin, returns POLLNVAL. */
706 fd_set RdSet;
707 FD_ZERO(&RdSet);
708 FD_SET(pThis->DeviceFileR, &RdSet);
709 FD_SET(pThis->WakeupPipeR, &RdSet);
710 fd_set XcptSet;
711 FD_ZERO(&XcptSet);
712 FD_SET(pThis->DeviceFileR, &XcptSet);
713 FD_SET(pThis->WakeupPipeR, &XcptSet);
714# if 1 /* it seems like this select is blocking the write... */
715 rc = select(RT_MAX(pThis->WakeupPipeR, pThis->DeviceFileR) + 1, &RdSet, NULL, &XcptSet, NULL);
716# else
717 struct timeval tv = { 0, 1000 };
718 rc = select(RT_MAX(pThis->WakeupPipeR, pThis->DeviceFileR) + 1, &RdSet, NULL, &XcptSet, &tv);
719# endif
720 if (rc == -1)
721 {
722 int err = errno;
723 rcThread = RTErrConvertFromErrno(err);
724 LogRel(("HostSerial#%d: select failed with errno=%d / %Rrc, terminating the worker thread.\n", pDrvIns->iInstance, err, rcThread));
725 break;
726 }
727
728 /* this might have changed in the meantime */
729 if (pThread->enmState != PDMTHREADSTATE_RUNNING)
730 break;
731 if (rc == 0)
732 continue;
733
734 /* drain the wakeup pipe */
735 size_t cbRead;
736 if ( FD_ISSET(pThis->WakeupPipeR, &RdSet)
737 || FD_ISSET(pThis->WakeupPipeR, &XcptSet))
738 {
739 rc = RTFileRead(pThis->WakeupPipeR, abBuffer, 1, &cbRead);
740 if (RT_FAILURE(rc))
741 {
742 LogRel(("HostSerial#%d: draining the wakekup pipe failed with %Rrc, terminating the worker thread.\n", pDrvIns->iInstance, rc));
743 rcThread = rc;
744 break;
745 }
746 continue;
747 }
748
749 /* read data from the serial port. */
750 rc = RTFileRead(pThis->DeviceFileR, abBuffer, sizeof(abBuffer), &cbRead);
751 if (RT_FAILURE(rc))
752 {
753 LogRel(("HostSerial#%d: (1) Read failed with %Rrc, terminating the worker thread.\n", pDrvIns->iInstance, rc));
754 rcThread = rc;
755 break;
756 }
757 cbRemaining = cbRead;
758
759#elif defined(RT_OS_LINUX) || defined(RT_OS_SOLARIS) || defined(RT_OS_FREEBSD)
760
761 size_t cbRead;
762 struct pollfd aFDs[2];
763 aFDs[0].fd = pThis->DeviceFile;
764 aFDs[0].events = POLLIN;
765 aFDs[0].revents = 0;
766 aFDs[1].fd = pThis->WakeupPipeR;
767 aFDs[1].events = POLLIN | POLLERR | POLLHUP;
768 aFDs[1].revents = 0;
769 rc = poll(aFDs, RT_ELEMENTS(aFDs), -1);
770 if (rc < 0)
771 {
772 int err = errno;
773 rcThread = RTErrConvertFromErrno(err);
774 LogRel(("HostSerial#%d: poll failed with errno=%d / %Rrc, terminating the worker thread.\n", pDrvIns->iInstance, err, rcThread));
775 break;
776 }
777 /* this might have changed in the meantime */
778 if (pThread->enmState != PDMTHREADSTATE_RUNNING)
779 break;
780 if (rc > 0 && aFDs[1].revents)
781 {
782 if (aFDs[1].revents & (POLLHUP | POLLERR | POLLNVAL))
783 break;
784 /* notification to terminate -- drain the pipe */
785 RTFileRead(pThis->WakeupPipeR, &abBuffer, 1, &cbRead);
786 continue;
787 }
788 rc = RTFileRead(pThis->DeviceFile, abBuffer, sizeof(abBuffer), &cbRead);
789 if (RT_FAILURE(rc))
790 {
791 /* don't terminate worker thread when data unavailable */
792 if (rc == VERR_TRY_AGAIN)
793 continue;
794
795 LogRel(("HostSerial#%d: (2) Read failed with %Rrc, terminating the worker thread.\n", pDrvIns->iInstance, rc));
796 rcThread = rc;
797 break;
798 }
799 cbRemaining = cbRead;
800
801#elif defined(RT_OS_WINDOWS)
802
803 DWORD dwEventMask = 0;
804 DWORD dwNumberOfBytesTransferred;
805
806 memset(&pThis->overlappedRecv, 0, sizeof(pThis->overlappedRecv));
807 pThis->overlappedRecv.hEvent = pThis->hEventRecv;
808
809 if (!WaitCommEvent(pThis->hDeviceFile, &dwEventMask, &pThis->overlappedRecv))
810 {
811 dwRet = GetLastError();
812 if (dwRet == ERROR_IO_PENDING)
813 {
814 dwRet = WaitForMultipleObjects(2, ahWait, FALSE, INFINITE);
815 if (dwRet != WAIT_OBJECT_0)
816 {
817 /* notification to terminate */
818 AssertMsg(pThread->enmState != PDMTHREADSTATE_RUNNING, ("The halt event sempahore is set but the thread is still in running state\n"));
819 break;
820 }
821 }
822 else
823 {
824 rcThread = RTErrConvertFromWin32(dwRet);
825 LogRel(("HostSerial#%d: Wait failed with error %Rrc; terminating the worker thread.\n", pDrvIns->iInstance, rcThread));
826 break;
827 }
828 }
829 /* this might have changed in the meantime */
830 if (pThread->enmState != PDMTHREADSTATE_RUNNING)
831 break;
832
833 /* Check the event */
834 if (dwEventMask & EV_RXCHAR)
835 {
836 if (!ReadFile(pThis->hDeviceFile, abBuffer, sizeof(abBuffer), &dwNumberOfBytesTransferred, &pThis->overlappedRecv))
837 {
838 rcThread = RTErrConvertFromWin32(GetLastError());
839 LogRel(("HostSerial#%d: Read failed with error %Rrc; terminating the worker thread.\n", pDrvIns->iInstance, rcThread));
840 break;
841 }
842 cbRemaining = dwNumberOfBytesTransferred;
843 }
844 else if (dwEventMask & EV_BREAK)
845 {
846 Log(("HostSerial#%d: Detected break\n"));
847 rc = pThis->pDrvCharPort->pfnNotifyBreak(pThis->pDrvCharPort);
848 }
849 else
850 {
851 /* The status lines have changed. Notify the device. */
852 DWORD dwNewStatusLinesState = 0;
853 uint32_t uNewStatusLinesState = 0;
854
855 /* Get the new state */
856 if (GetCommModemStatus(pThis->hDeviceFile, &dwNewStatusLinesState))
857 {
858 if (dwNewStatusLinesState & MS_RLSD_ON)
859 uNewStatusLinesState |= PDMICHARPORT_STATUS_LINES_DCD;
860 if (dwNewStatusLinesState & MS_RING_ON)
861 uNewStatusLinesState |= PDMICHARPORT_STATUS_LINES_RI;
862 if (dwNewStatusLinesState & MS_DSR_ON)
863 uNewStatusLinesState |= PDMICHARPORT_STATUS_LINES_DSR;
864 if (dwNewStatusLinesState & MS_CTS_ON)
865 uNewStatusLinesState |= PDMICHARPORT_STATUS_LINES_CTS;
866 rc = pThis->pDrvCharPort->pfnNotifyStatusLinesChanged(pThis->pDrvCharPort, uNewStatusLinesState);
867 if (RT_FAILURE(rc))
868 {
869 /* Notifying device failed, continue but log it */
870 LogRel(("HostSerial#%d: Notifying device failed with error %Rrc; continuing.\n", pDrvIns->iInstance, rc));
871 }
872 }
873 else
874 {
875 /* Getting new state failed, continue but log it */
876 LogRel(("HostSerial#%d: Getting status lines state failed with error %Rrc; continuing.\n", pDrvIns->iInstance, RTErrConvertFromWin32(GetLastError())));
877 }
878 }
879#endif
880
881 Log(("Read %d bytes.\n", cbRemaining));
882 pbBuffer = abBuffer;
883 }
884 else
885 {
886 /* Send data to the guest. */
887 size_t cbProcessed = cbRemaining;
888 rc = pThis->pDrvCharPort->pfnNotifyRead(pThis->pDrvCharPort, pbBuffer, &cbProcessed);
889 if (RT_SUCCESS(rc))
890 {
891 Assert(cbProcessed); Assert(cbProcessed <= cbRemaining);
892 pbBuffer += cbProcessed;
893 cbRemaining -= cbProcessed;
894 STAM_COUNTER_ADD(&pThis->StatBytesRead, cbProcessed);
895 }
896 else if (rc == VERR_TIMEOUT)
897 {
898 /* Normal case, just means that the guest didn't accept a new
899 * character before the timeout elapsed. Just retry. */
900 rc = VINF_SUCCESS;
901 }
902 else
903 {
904 LogRel(("HostSerial#%d: NotifyRead failed with %Rrc, terminating the worker thread.\n", pDrvIns->iInstance, rc));
905 rcThread = rc;
906 break;
907 }
908 }
909 }
910
911 return rcThread;
912}
913
914/**
915 * Unblock the send thread so it can respond to a state change.
916 *
917 * @returns a VBox status code.
918 * @param pDrvIns The driver instance.
919 * @param pThread The send thread.
920 */
921static DECLCALLBACK(int) drvHostSerialWakeupRecvThread(PPDMDRVINS pDrvIns, PPDMTHREAD pThread)
922{
923 PDRVHOSTSERIAL pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTSERIAL);
924#if defined(RT_OS_LINUX) || defined(RT_OS_DARWIN) || defined(RT_OS_SOLARIS) || defined(RT_OS_FREEBSD)
925 return RTFileWrite(pThis->WakeupPipeW, "", 1, NULL);
926#elif defined(RT_OS_WINDOWS)
927 if (!SetEvent(pThis->hHaltEventSem))
928 return RTErrConvertFromWin32(GetLastError());
929 return VINF_SUCCESS;
930#else
931# error adapt me!
932#endif
933}
934
935#if defined(RT_OS_LINUX) || defined(RT_OS_DARWIN) || defined(RT_OS_SOLARIS) || defined(RT_OS_FREEBSD)
936/* -=-=-=-=- Monitor thread -=-=-=-=- */
937
938/**
939 * Monitor thread loop.
940 *
941 * This thread monitors the status lines and notifies the device
942 * if they change.
943 *
944 * @returns VINF_SUCCESS.
945 * @param ThreadSelf Thread handle to this thread.
946 * @param pvUser User argument.
947 */
948static DECLCALLBACK(int) drvHostSerialMonitorThread(PPDMDRVINS pDrvIns, PPDMTHREAD pThread)
949{
950 PDRVHOSTSERIAL pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTSERIAL);
951 int rc = VINF_SUCCESS;
952 unsigned uStatusLinesToCheck = 0;
953
954 uStatusLinesToCheck = TIOCM_CAR | TIOCM_RNG | TIOCM_LE | TIOCM_CTS;
955
956 if (pThread->enmState == PDMTHREADSTATE_INITIALIZING)
957 return VINF_SUCCESS;
958
959 while (pThread->enmState == PDMTHREADSTATE_RUNNING)
960 {
961 uint32_t newStatusLine = 0;
962 unsigned int statusLines;
963
964# ifdef RT_OS_LINUX
965 /*
966 * Wait for status line change.
967 */
968 rc = ioctl(pThis->DeviceFile, TIOCMIWAIT, uStatusLinesToCheck);
969 if (pThread->enmState != PDMTHREADSTATE_RUNNING)
970 break;
971 if (rc < 0)
972 {
973ioctl_error:
974 PDMDrvHlpVMSetRuntimeError(pDrvIns, 0 /*fFlags*/, "DrvHostSerialFail",
975 N_("Ioctl failed for serial host device '%s' (%Rrc). The device will not work properly"),
976 pThis->pszDevicePath, RTErrConvertFromErrno(errno));
977 break;
978 }
979
980 rc = ioctl(pThis->DeviceFile, TIOCMGET, &statusLines);
981 if (rc < 0)
982 goto ioctl_error;
983# else /* !RT_OS_LINUX */
984 /*
985 * Poll for the status line change.
986 */
987 rc = ioctl(pThis->DeviceFile, TIOCMGET, &statusLines);
988 if (rc < 0)
989 {
990 PDMDrvHlpVMSetRuntimeError(pDrvIns, 0 /*fFlags*/, "DrvHostSerialFail",
991 N_("Ioctl failed for serial host device '%s' (%Rrc). The device will not work properly"),
992 pThis->pszDevicePath, RTErrConvertFromErrno(errno));
993 break;
994 }
995 if (!((statusLines ^ pThis->fStatusLines) & uStatusLinesToCheck))
996 {
997 PDMR3ThreadSleep(pThread, 500); /* 0.5 sec */
998 continue;
999 }
1000 pThis->fStatusLines = statusLines;
1001# endif /* !RT_OS_LINUX */
1002
1003 if (statusLines & TIOCM_CAR)
1004 newStatusLine |= PDMICHARPORT_STATUS_LINES_DCD;
1005 if (statusLines & TIOCM_RNG)
1006 newStatusLine |= PDMICHARPORT_STATUS_LINES_RI;
1007 if (statusLines & TIOCM_LE)
1008 newStatusLine |= PDMICHARPORT_STATUS_LINES_DSR;
1009 if (statusLines & TIOCM_CTS)
1010 newStatusLine |= PDMICHARPORT_STATUS_LINES_CTS;
1011 rc = pThis->pDrvCharPort->pfnNotifyStatusLinesChanged(pThis->pDrvCharPort, newStatusLine);
1012 }
1013
1014 return VINF_SUCCESS;
1015}
1016
1017/**
1018 * Unblock the monitor thread so it can respond to a state change.
1019 * We need to execute this code exactly once during initialization.
1020 * But we don't want to block --- therefore this dedicated thread.
1021 *
1022 * @returns a VBox status code.
1023 * @param pDrvIns The driver instance.
1024 * @param pThread The send thread.
1025 */
1026static DECLCALLBACK(int) drvHostSerialWakeupMonitorThread(PPDMDRVINS pDrvIns, PPDMTHREAD pThread)
1027{
1028# ifdef RT_OS_LINUX
1029 PDRVHOSTSERIAL pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTSERIAL);
1030 int rc = VINF_SUCCESS;
1031# if 0
1032 unsigned int uSerialLineFlags;
1033 unsigned int uSerialLineStatus;
1034 unsigned int uIoctl;
1035# endif
1036
1037 /*
1038 * Linux is a bit difficult as the thread is sleeping in an ioctl call.
1039 * So there is no way to have a wakeup pipe.
1040 *
1041 * 1. Thatswhy we set the serial device into loopback mode and change one of the
1042 * modem control bits.
1043 * This should make the ioctl call return.
1044 *
1045 * 2. We still got reports about long shutdown times. It may bepossible
1046 * that the loopback mode is not implemented on all devices.
1047 * The next possible solution is to close the device file to make the ioctl
1048 * return with EBADF and be able to suspend the thread.
1049 *
1050 * 3. The second approach doesn't work too, the ioctl doesn't return.
1051 * But it seems that the ioctl is interruptible (return code in errno is EINTR).
1052 */
1053
1054# if 0 /* Disabled because it does not work for all. */
1055 /* Get current status of control lines. */
1056 rc = ioctl(pThis->DeviceFile, TIOCMGET, &uSerialLineStatus);
1057 if (rc < 0)
1058 goto ioctl_error;
1059
1060 uSerialLineFlags = TIOCM_LOOP;
1061 rc = ioctl(pThis->DeviceFile, TIOCMBIS, &uSerialLineFlags);
1062 if (rc < 0)
1063 goto ioctl_error;
1064
1065 /*
1066 * Change current level on the RTS pin to make the ioctl call return in the
1067 * monitor thread.
1068 */
1069 uIoctl = (uSerialLineStatus & TIOCM_CTS) ? TIOCMBIC : TIOCMBIS;
1070 uSerialLineFlags = TIOCM_RTS;
1071
1072 rc = ioctl(pThis->DeviceFile, uIoctl, &uSerialLineFlags);
1073 if (rc < 0)
1074 goto ioctl_error;
1075
1076 /* Change RTS back to the previous level. */
1077 uIoctl = (uIoctl == TIOCMBIC) ? TIOCMBIS : TIOCMBIC;
1078
1079 rc = ioctl(pThis->DeviceFile, uIoctl, &uSerialLineFlags);
1080 if (rc < 0)
1081 goto ioctl_error;
1082
1083 /*
1084 * Set serial device into normal state.
1085 */
1086 uSerialLineFlags = TIOCM_LOOP;
1087 rc = ioctl(pThis->DeviceFile, TIOCMBIC, &uSerialLineFlags);
1088 if (rc >= 0)
1089 return VINF_SUCCESS;
1090
1091ioctl_error:
1092 PDMDrvHlpVMSetRuntimeError(pDrvIns, 0 /*fFlags*/, "DrvHostSerialFail",
1093 N_("Ioctl failed for serial host device '%s' (%Rrc). The device will not work properly"),
1094 pThis->pszDevicePath, RTErrConvertFromErrno(errno));
1095# endif
1096
1097# if 0
1098 /* Close file to make ioctl return. */
1099 RTFileClose(pData->DeviceFile);
1100 /* Open again to make use after suspend possible again. */
1101 rc = RTFileOpen(&pData->DeviceFile, pData->pszDevicePath, RTFILE_O_OPEN | RTFILE_O_READWRITE);
1102 AssertMsgRC(rc, ("Opening device file again failed rc=%Rrc\n", rc));
1103
1104 if (RT_FAILURE(rc))
1105 PDMDrvHlpVMSetRuntimeError(pDrvIns, false, "DrvHostSerialFail",
1106 N_("Opening failed for serial host device '%s' (%Rrc). The device will not work"),
1107 pData->pszDevicePath, rc);
1108# endif
1109
1110 rc = RTThreadPoke(pThread->Thread);
1111 if (RT_FAILURE(rc))
1112 PDMDrvHlpVMSetRuntimeError(pDrvIns, 0 /*fFlags*/, "DrvHostSerialFail",
1113 N_("Suspending serial monitor thread failed for serial device '%s' (%Rrc). The shutdown may take longer than expected"),
1114 pThis->pszDevicePath, RTErrConvertFromErrno(rc));
1115
1116# else /* !RT_OS_LINUX*/
1117
1118 /* In polling mode there is nobody to wake up (PDMThread will cancel the sleep). */
1119 NOREF(pDrvIns);
1120 NOREF(pThread);
1121
1122# endif /* RT_OS_LINUX */
1123
1124 return VINF_SUCCESS;
1125}
1126#endif /* RT_OS_LINUX || RT_OS_DARWIN || RT_OS_SOLARIS */
1127
1128/**
1129 * Set the modem lines.
1130 *
1131 * @returns VBox status code
1132 * @param pInterface Pointer to the interface structure.
1133 * @param RequestToSend Set to true if this control line should be made active.
1134 * @param DataTerminalReady Set to true if this control line should be made active.
1135 */
1136static DECLCALLBACK(int) drvHostSerialSetModemLines(PPDMICHARCONNECTOR pInterface, bool RequestToSend, bool DataTerminalReady)
1137{
1138 PDRVHOSTSERIAL pThis = PDMICHAR_2_DRVHOSTSERIAL(pInterface);
1139
1140#if defined(RT_OS_LINUX) || defined(RT_OS_DARWIN) || defined(RT_OS_SOLARIS) || defined(RT_OS_FREEBSD)
1141 int modemStateSet = 0;
1142 int modemStateClear = 0;
1143
1144 if (RequestToSend)
1145 modemStateSet |= TIOCM_RTS;
1146 else
1147 modemStateClear |= TIOCM_RTS;
1148
1149 if (DataTerminalReady)
1150 modemStateSet |= TIOCM_DTR;
1151 else
1152 modemStateClear |= TIOCM_DTR;
1153
1154 if (modemStateSet)
1155 ioctl(pThis->DeviceFile, TIOCMBIS, &modemStateSet);
1156
1157 if (modemStateClear)
1158 ioctl(pThis->DeviceFile, TIOCMBIC, &modemStateClear);
1159#elif defined(RT_OS_WINDOWS)
1160 if (RequestToSend)
1161 EscapeCommFunction(pThis->hDeviceFile, SETRTS);
1162 else
1163 EscapeCommFunction(pThis->hDeviceFile, CLRRTS);
1164
1165 if (DataTerminalReady)
1166 EscapeCommFunction(pThis->hDeviceFile, SETDTR);
1167 else
1168 EscapeCommFunction(pThis->hDeviceFile, CLRDTR);
1169#endif
1170
1171 return VINF_SUCCESS;
1172}
1173
1174/**
1175 * Sets the TD line into break condition.
1176 *
1177 * @returns VBox status code.
1178 * @param pInterface Pointer to the interface structure containing the called function pointer.
1179 * @param fBreak Set to true to let the device send a break false to put into normal operation.
1180 * @thread Any thread.
1181 */
1182static DECLCALLBACK(int) drvHostSerialSetBreak(PPDMICHARCONNECTOR pInterface, bool fBreak)
1183{
1184 PDRVHOSTSERIAL pThis = PDMICHAR_2_DRVHOSTSERIAL(pInterface);
1185
1186#if defined(RT_OS_LINUX) || defined(RT_OS_DARWIN) || defined(RT_OS_SOLARIS) || defined(RT_OS_FREEBSD)
1187 if (fBreak)
1188 ioctl(pThis->DeviceFile, TIOCSBRK);
1189 else
1190 ioctl(pThis->DeviceFile, TIOCCBRK);
1191
1192#elif defined(RT_OS_WINDOWS)
1193 if (fBreak)
1194 SetCommBreak(pThis->hDeviceFile);
1195 else
1196 ClearCommBreak(pThis->hDeviceFile);
1197#endif
1198
1199 return VINF_SUCCESS;
1200}
1201
1202/* -=-=-=-=- driver interface -=-=-=-=- */
1203
1204/**
1205 * Destruct a char driver instance.
1206 *
1207 * Most VM resources are freed by the VM. This callback is provided so that
1208 * any non-VM resources can be freed correctly.
1209 *
1210 * @param pDrvIns The driver instance data.
1211 */
1212static DECLCALLBACK(void) drvHostSerialDestruct(PPDMDRVINS pDrvIns)
1213{
1214 PDRVHOSTSERIAL pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTSERIAL);
1215 LogFlow(("%s: iInstance=%d\n", __FUNCTION__, pDrvIns->iInstance));
1216 PDMDRV_CHECK_VERSIONS_RETURN_VOID(pDrvIns);
1217
1218 /* Empty the send queue */
1219 pThis->iSendQueueTail = pThis->iSendQueueHead = 0;
1220
1221 RTSemEventDestroy(pThis->SendSem);
1222 pThis->SendSem = NIL_RTSEMEVENT;
1223
1224#if defined(RT_OS_LINUX) || defined(RT_OS_DARWIN) || defined(RT_OS_SOLARIS) || defined(RT_OS_FREEBSD)
1225
1226 if (pThis->WakeupPipeW != NIL_RTFILE)
1227 {
1228 int rc = RTFileClose(pThis->WakeupPipeW);
1229 AssertRC(rc);
1230 pThis->WakeupPipeW = NIL_RTFILE;
1231 }
1232 if (pThis->WakeupPipeR != NIL_RTFILE)
1233 {
1234 int rc = RTFileClose(pThis->WakeupPipeR);
1235 AssertRC(rc);
1236 pThis->WakeupPipeR = NIL_RTFILE;
1237 }
1238# if defined(RT_OS_DARWIN)
1239 if (pThis->DeviceFileR != NIL_RTFILE)
1240 {
1241 if (pThis->DeviceFileR != pThis->DeviceFile)
1242 {
1243 int rc = RTFileClose(pThis->DeviceFileR);
1244 AssertRC(rc);
1245 }
1246 pThis->DeviceFileR = NIL_RTFILE;
1247 }
1248# endif
1249 if (pThis->DeviceFile != NIL_RTFILE)
1250 {
1251 int rc = RTFileClose(pThis->DeviceFile);
1252 AssertRC(rc);
1253 pThis->DeviceFile = NIL_RTFILE;
1254 }
1255
1256#elif defined(RT_OS_WINDOWS)
1257
1258 CloseHandle(pThis->hEventRecv);
1259 CloseHandle(pThis->hEventSend);
1260 CancelIo(pThis->hDeviceFile);
1261 CloseHandle(pThis->hDeviceFile);
1262
1263#endif
1264}
1265
1266/**
1267 * Construct a char driver instance.
1268 *
1269 * @copydoc FNPDMDRVCONSTRUCT
1270 */
1271static DECLCALLBACK(int) drvHostSerialConstruct(PPDMDRVINS pDrvIns, PCFGMNODE pCfg, uint32_t fFlags)
1272{
1273 PDRVHOSTSERIAL pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTSERIAL);
1274 LogFlow(("%s: iInstance=%d\n", __FUNCTION__, pDrvIns->iInstance));
1275 PDMDRV_CHECK_VERSIONS_RETURN(pDrvIns);
1276
1277 /*
1278 * Init basic data members and interfaces.
1279 */
1280#if defined(RT_OS_LINUX) || defined(RT_OS_DARWIN) || defined(RT_OS_SOLARIS) || defined(RT_OS_FREEBSD)
1281 pThis->DeviceFile = NIL_RTFILE;
1282# ifdef RT_OS_DARWIN
1283 pThis->DeviceFileR = NIL_RTFILE;
1284# endif
1285 pThis->WakeupPipeR = NIL_RTFILE;
1286 pThis->WakeupPipeW = NIL_RTFILE;
1287#endif
1288 /* IBase. */
1289 pDrvIns->IBase.pfnQueryInterface = drvHostSerialQueryInterface;
1290 /* ICharConnector. */
1291 pThis->ICharConnector.pfnWrite = drvHostSerialWrite;
1292 pThis->ICharConnector.pfnSetParameters = drvHostSerialSetParameters;
1293 pThis->ICharConnector.pfnSetModemLines = drvHostSerialSetModemLines;
1294 pThis->ICharConnector.pfnSetBreak = drvHostSerialSetBreak;
1295
1296/** @todo Initialize all members with NIL values!! The destructor is ALWAYS called. */
1297
1298 /*
1299 * Query configuration.
1300 */
1301 /* Device */
1302 int rc = CFGMR3QueryStringAlloc(pCfg, "DevicePath", &pThis->pszDevicePath);
1303 if (RT_FAILURE(rc))
1304 {
1305 AssertMsgFailed(("Configuration error: query for \"DevicePath\" string returned %Rra.\n", rc));
1306 return rc;
1307 }
1308
1309 /*
1310 * Open the device
1311 */
1312#ifdef RT_OS_WINDOWS
1313
1314 pThis->hHaltEventSem = CreateEvent(NULL, FALSE, FALSE, NULL);
1315 AssertReturn(pThis->hHaltEventSem != NULL, VERR_NO_MEMORY);
1316
1317 pThis->hEventRecv = CreateEvent(NULL, FALSE, FALSE, NULL);
1318 AssertReturn(pThis->hEventRecv != NULL, VERR_NO_MEMORY);
1319
1320 pThis->hEventSend = CreateEvent(NULL, FALSE, FALSE, NULL);
1321 AssertReturn(pThis->hEventSend != NULL, VERR_NO_MEMORY);
1322
1323 HANDLE hFile = CreateFile(pThis->pszDevicePath,
1324 GENERIC_READ | GENERIC_WRITE,
1325 0, // must be opened with exclusive access
1326 NULL, // no SECURITY_ATTRIBUTES structure
1327 OPEN_EXISTING, // must use OPEN_EXISTING
1328 FILE_FLAG_OVERLAPPED, // overlapped I/O
1329 NULL); // no template file
1330 if (hFile == INVALID_HANDLE_VALUE)
1331 rc = RTErrConvertFromWin32(GetLastError());
1332 else
1333 {
1334 pThis->hDeviceFile = hFile;
1335 /* for overlapped read */
1336 if (!SetCommMask(hFile, EV_RXCHAR | EV_CTS | EV_DSR | EV_RING | EV_RLSD))
1337 {
1338 LogRel(("HostSerial#%d: SetCommMask failed with error %d.\n", pDrvIns->iInstance, GetLastError()));
1339 return VERR_FILE_IO_ERROR;
1340 }
1341 rc = VINF_SUCCESS;
1342 }
1343
1344#else
1345
1346 rc = RTFileOpen(&pThis->DeviceFile, pThis->pszDevicePath, RTFILE_O_READWRITE | RTFILE_O_OPEN | RTFILE_O_DENY_NONE);
1347# ifdef RT_OS_DARWIN
1348 if (RT_SUCCESS(rc))
1349 rc = RTFileOpen(&pThis->DeviceFileR, pThis->pszDevicePath, RTFILE_O_READ | RTFILE_O_OPEN | RTFILE_O_DENY_NONE);
1350# endif
1351
1352
1353#endif
1354
1355 if (RT_FAILURE(rc))
1356 {
1357 AssertMsgFailed(("Could not open host device %s, rc=%Rrc\n", pThis->pszDevicePath, rc));
1358 switch (rc)
1359 {
1360 case VERR_ACCESS_DENIED:
1361 return PDMDrvHlpVMSetError(pDrvIns, rc, RT_SRC_POS,
1362#if defined(RT_OS_LINUX) || defined(RT_OS_DARWIN) || defined(RT_OS_SOLARIS) || defined(RT_OS_FREEBSD)
1363 N_("Cannot open host device '%s' for read/write access. Check the permissions "
1364 "of that device ('/bin/ls -l %s'): Most probably you need to be member "
1365 "of the device group. Make sure that you logout/login after changing "
1366 "the group settings of the current user"),
1367#else
1368 N_("Cannot open host device '%s' for read/write access. Check the permissions "
1369 "of that device"),
1370#endif
1371 pThis->pszDevicePath, pThis->pszDevicePath);
1372 default:
1373 return PDMDrvHlpVMSetError(pDrvIns, rc, RT_SRC_POS,
1374 N_("Failed to open host device '%s'"),
1375 pThis->pszDevicePath);
1376 }
1377 }
1378
1379 /* Set to non blocking I/O */
1380#if defined(RT_OS_LINUX) || defined(RT_OS_DARWIN) || defined(RT_OS_SOLARIS) || defined(RT_OS_FREEBSD)
1381
1382 fcntl(pThis->DeviceFile, F_SETFL, O_NONBLOCK);
1383# ifdef RT_OS_DARWIN
1384 fcntl(pThis->DeviceFileR, F_SETFL, O_NONBLOCK);
1385# endif
1386 int aFDs[2];
1387 if (pipe(aFDs) != 0)
1388 {
1389 rc = RTErrConvertFromErrno(errno);
1390 AssertRC(rc);
1391 return rc;
1392 }
1393 pThis->WakeupPipeR = aFDs[0];
1394 pThis->WakeupPipeW = aFDs[1];
1395
1396#elif defined(RT_OS_WINDOWS)
1397
1398 /* Set the COMMTIMEOUTS to get non blocking I/O */
1399 COMMTIMEOUTS comTimeout;
1400
1401 comTimeout.ReadIntervalTimeout = MAXDWORD;
1402 comTimeout.ReadTotalTimeoutMultiplier = 0;
1403 comTimeout.ReadTotalTimeoutConstant = 0;
1404 comTimeout.WriteTotalTimeoutMultiplier = 0;
1405 comTimeout.WriteTotalTimeoutConstant = 0;
1406
1407 SetCommTimeouts(pThis->hDeviceFile, &comTimeout);
1408
1409#endif
1410
1411 /*
1412 * Get the ICharPort interface of the above driver/device.
1413 */
1414 pThis->pDrvCharPort = PDMIBASE_QUERY_INTERFACE(pDrvIns->pUpBase, PDMICHARPORT);
1415 if (!pThis->pDrvCharPort)
1416 return PDMDrvHlpVMSetError(pDrvIns, VERR_PDM_MISSING_INTERFACE_ABOVE, RT_SRC_POS, N_("HostSerial#%d has no char port interface above"), pDrvIns->iInstance);
1417
1418 /*
1419 * Create the receive, send and monitor threads pluss the related send semaphore.
1420 */
1421 rc = PDMDrvHlpPDMThreadCreate(pDrvIns, &pThis->pRecvThread, pThis, drvHostSerialRecvThread, drvHostSerialWakeupRecvThread, 0, RTTHREADTYPE_IO, "SerRecv");
1422 if (RT_FAILURE(rc))
1423 return PDMDrvHlpVMSetError(pDrvIns, rc, RT_SRC_POS, N_("HostSerial#%d cannot create receive thread"), pDrvIns->iInstance);
1424
1425 rc = RTSemEventCreate(&pThis->SendSem);
1426 AssertRC(rc);
1427
1428 rc = PDMDrvHlpPDMThreadCreate(pDrvIns, &pThis->pSendThread, pThis, drvHostSerialSendThread, drvHostSerialWakeupSendThread, 0, RTTHREADTYPE_IO, "SerSend");
1429 if (RT_FAILURE(rc))
1430 return PDMDrvHlpVMSetError(pDrvIns, rc, RT_SRC_POS, N_("HostSerial#%d cannot create send thread"), pDrvIns->iInstance);
1431
1432#if defined(RT_OS_LINUX) || defined(RT_OS_DARWIN) || defined(RT_OS_SOLARIS) || defined(RT_OS_FREEBSD)
1433 /* Linux & darwin needs a separate thread which monitors the status lines. */
1434# ifndef RT_OS_LINUX
1435 ioctl(pThis->DeviceFile, TIOCMGET, &pThis->fStatusLines);
1436# endif
1437 rc = PDMDrvHlpPDMThreadCreate(pDrvIns, &pThis->pMonitorThread, pThis, drvHostSerialMonitorThread, drvHostSerialWakeupMonitorThread, 0, RTTHREADTYPE_IO, "SerMon");
1438 if (RT_FAILURE(rc))
1439 return PDMDrvHlpVMSetError(pDrvIns, rc, RT_SRC_POS, N_("HostSerial#%d cannot create monitor thread"), pDrvIns->iInstance);
1440#endif
1441
1442 /*
1443 * Register release statistics.
1444 */
1445 PDMDrvHlpSTAMRegisterF(pDrvIns, &pThis->StatBytesWritten, STAMTYPE_COUNTER, STAMVISIBILITY_USED, STAMUNIT_BYTES, "Nr of bytes written", "/Devices/HostSerial%d/Written", pDrvIns->iInstance);
1446 PDMDrvHlpSTAMRegisterF(pDrvIns, &pThis->StatBytesRead, STAMTYPE_COUNTER, STAMVISIBILITY_USED, STAMUNIT_BYTES, "Nr of bytes read", "/Devices/HostSerial%d/Read", pDrvIns->iInstance);
1447#ifdef RT_OS_DARWIN /* new Write code, not darwin specific. */
1448 PDMDrvHlpSTAMRegisterF(pDrvIns, &pThis->StatSendOverflows, STAMTYPE_COUNTER, STAMVISIBILITY_USED, STAMUNIT_BYTES, "Nr of bytes overflowed", "/Devices/HostSerial%d/SendOverflow", pDrvIns->iInstance);
1449#endif
1450
1451 return VINF_SUCCESS;
1452}
1453
1454/**
1455 * Char driver registration record.
1456 */
1457const PDMDRVREG g_DrvHostSerial =
1458{
1459 /* u32Version */
1460 PDM_DRVREG_VERSION,
1461 /* szName */
1462 "Host Serial",
1463 /* szRCMod */
1464 "",
1465 /* szR0Mod */
1466 "",
1467/* pszDescription */
1468 "Host serial driver.",
1469 /* fFlags */
1470 PDM_DRVREG_FLAGS_HOST_BITS_DEFAULT,
1471 /* fClass. */
1472 PDM_DRVREG_CLASS_CHAR,
1473 /* cMaxInstances */
1474 ~0,
1475 /* cbInstance */
1476 sizeof(DRVHOSTSERIAL),
1477 /* pfnConstruct */
1478 drvHostSerialConstruct,
1479 /* pfnDestruct */
1480 drvHostSerialDestruct,
1481 /* pfnRelocate */
1482 NULL,
1483 /* pfnIOCtl */
1484 NULL,
1485 /* pfnPowerOn */
1486 NULL,
1487 /* pfnReset */
1488 NULL,
1489 /* pfnSuspend */
1490 NULL,
1491 /* pfnResume */
1492 NULL,
1493 /* pfnAttach */
1494 NULL,
1495 /* pfnDetach */
1496 NULL,
1497 /* pfnPowerOff */
1498 NULL,
1499 /* pfnSoftReset */
1500 NULL,
1501 /* u32EndVersion */
1502 PDM_DRVREG_VERSION
1503};
1504
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