VirtualBox

source: vbox/trunk/src/VBox/Devices/Serial/DrvTCP.cpp@ 62563

Last change on this file since 62563 was 62507, checked in by vboxsync, 9 years ago

(C) 2016

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 15.0 KB
Line 
1/* $Id: DrvTCP.cpp 62507 2016-07-22 19:09:53Z vboxsync $ */
2/** @file
3 * TCP socket driver implementing the IStream interface.
4 */
5
6/*
7 * Copyright (C) 2006-2016 Oracle Corporation.
8 *
9 * This file was contributed by Alexey Eromenko (derived from DrvNamedPipe)
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
20
21/*********************************************************************************************************************************
22* Header Files *
23*********************************************************************************************************************************/
24#define LOG_GROUP LOG_GROUP_DRV_TCP
25#include <VBox/vmm/pdmdrv.h>
26#include <iprt/assert.h>
27#include <iprt/file.h>
28#include <iprt/stream.h>
29#include <iprt/alloc.h>
30#include <iprt/string.h>
31#include <iprt/semaphore.h>
32#include <iprt/uuid.h>
33#include <stdlib.h>
34
35#include "VBoxDD.h"
36
37#ifdef RT_OS_WINDOWS
38# include <ws2tcpip.h>
39#else /* !RT_OS_WINDOWS */
40# include <errno.h>
41# include <unistd.h>
42# include <sys/types.h>
43# include <sys/socket.h>
44# include <netinet/in.h>
45# include <netdb.h>
46# ifndef SHUT_RDWR /* OS/2 */
47# define SHUT_RDWR 3
48# endif
49#endif /* !RT_OS_WINDOWS */
50
51#ifndef SHUT_RDWR
52# ifdef SD_BOTH
53# define SHUT_RDWR SD_BOTH
54# else
55# define SHUT_RDWR 2
56# endif
57#endif
58
59
60/*********************************************************************************************************************************
61* Defined Constants And Macros *
62*********************************************************************************************************************************/
63/** Converts a pointer to DRVTCP::IMedia to a PDRVTCP. */
64#define PDMISTREAM_2_DRVTCP(pInterface) ( (PDRVTCP)((uintptr_t)pInterface - RT_OFFSETOF(DRVTCP, IStream)) )
65
66
67/*********************************************************************************************************************************
68* Structures and Typedefs *
69*********************************************************************************************************************************/
70/**
71 * TCP driver instance data.
72 *
73 * @implements PDMISTREAM
74 */
75typedef struct DRVTCP
76{
77 /** The stream interface. */
78 PDMISTREAM IStream;
79 /** Pointer to the driver instance. */
80 PPDMDRVINS pDrvIns;
81 /** Pointer to the TCP server address:port or port only. (Freed by MM) */
82 char *pszLocation;
83 /** Flag whether VirtualBox represents the server or client side. */
84 bool fIsServer;
85
86 /** Socket handle of the TCP socket for server. */
87 int TCPServer;
88 /** Socket handle of the TCP socket connection. */
89 int TCPConnection;
90
91 /** Thread for listening for new connections. */
92 RTTHREAD ListenThread;
93 /** Flag to signal listening thread to shut down. */
94 bool volatile fShutdown;
95} DRVTCP, *PDRVTCP;
96
97
98/*********************************************************************************************************************************
99* Internal Functions *
100*********************************************************************************************************************************/
101
102
103/** @copydoc PDMISTREAM::pfnRead */
104static DECLCALLBACK(int) drvTCPRead(PPDMISTREAM pInterface, void *pvBuf, size_t *pcbRead)
105{
106 int rc = VINF_SUCCESS;
107 PDRVTCP pThis = PDMISTREAM_2_DRVTCP(pInterface);
108 LogFlow(("%s: pvBuf=%p *pcbRead=%#x (%s)\n", __FUNCTION__, pvBuf, *pcbRead, pThis->pszLocation));
109
110 Assert(pvBuf);
111
112 if (pThis->TCPConnection != -1)
113 {
114 ssize_t cbReallyRead;
115 unsigned cbBuf = (unsigned)*pcbRead;
116 cbReallyRead = recv(pThis->TCPConnection, (char *)pvBuf, cbBuf, 0);
117 if (cbReallyRead == 0)
118 {
119 int tmp = pThis->TCPConnection;
120 pThis->TCPConnection = -1;
121#ifdef RT_OS_WINDOWS
122 closesocket(tmp);
123#else
124 close(tmp);
125#endif
126 }
127 else if (cbReallyRead == -1)
128 {
129 cbReallyRead = 0;
130 rc = RTErrConvertFromErrno(errno);
131 }
132 *pcbRead = cbReallyRead;
133 }
134 else
135 {
136 RTThreadSleep(100);
137 *pcbRead = 0;
138 }
139
140 LogFlow(("%s: *pcbRead=%zu returns %Rrc\n", __FUNCTION__, *pcbRead, rc));
141 return rc;
142}
143
144
145/** @copydoc PDMISTREAM::pfnWrite */
146static DECLCALLBACK(int) drvTCPWrite(PPDMISTREAM pInterface, const void *pvBuf, size_t *pcbWrite)
147{
148 int rc = VINF_SUCCESS;
149 PDRVTCP pThis = PDMISTREAM_2_DRVTCP(pInterface);
150 LogFlow(("%s: pvBuf=%p *pcbWrite=%#x (%s)\n", __FUNCTION__, pvBuf, *pcbWrite, pThis->pszLocation));
151
152 Assert(pvBuf);
153 if (pThis->TCPConnection != -1)
154 {
155 ssize_t cbWritten;
156 unsigned cbBuf = (unsigned)*pcbWrite;
157 cbWritten = send(pThis->TCPConnection, (const char *)pvBuf, cbBuf, 0);
158 if (cbWritten == 0)
159 {
160 int tmp = pThis->TCPConnection;
161 pThis->TCPConnection = -1;
162#ifdef RT_OS_WINDOWS
163 closesocket(tmp);
164#else
165 close(tmp);
166#endif
167 }
168 else if (cbWritten == -1)
169 {
170 cbWritten = 0;
171 rc = RTErrConvertFromErrno(errno);
172 }
173 *pcbWrite = cbWritten;
174 }
175
176 LogFlow(("%s: returns %Rrc\n", __FUNCTION__, rc));
177 return rc;
178}
179
180
181/**
182 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
183 */
184static DECLCALLBACK(void *) drvTCPQueryInterface(PPDMIBASE pInterface, const char *pszIID)
185{
186 PPDMDRVINS pDrvIns = PDMIBASE_2_PDMDRV(pInterface);
187 PDRVTCP pThis = PDMINS_2_DATA(pDrvIns, PDRVTCP);
188 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pDrvIns->IBase);
189 PDMIBASE_RETURN_INTERFACE(pszIID, PDMISTREAM, &pThis->IStream);
190 return NULL;
191}
192
193
194/* -=-=-=-=- listen thread -=-=-=-=- */
195
196/**
197 * Receive thread loop.
198 *
199 * @returns 0 on success.
200 * @param ThreadSelf Thread handle to this thread.
201 * @param pvUser User argument.
202 */
203static DECLCALLBACK(int) drvTCPListenLoop(RTTHREAD ThreadSelf, void *pvUser)
204{
205 PDRVTCP pThis = (PDRVTCP)pvUser;
206 int rc = VINF_SUCCESS;
207
208 while (RT_LIKELY(!pThis->fShutdown))
209 {
210 if (listen(pThis->TCPServer, 0) == -1)
211 {
212 rc = RTErrConvertFromErrno(errno);
213 LogRel(("DrvTCP%d: listen failed, rc=%Rrc\n", pThis->pDrvIns->iInstance, rc));
214 break;
215 }
216 int s = accept(pThis->TCPServer, NULL, NULL);
217 if (s == -1)
218 {
219 rc = RTErrConvertFromErrno(errno);
220 LogRel(("DrvTCP%d: accept failed, rc=%Rrc\n", pThis->pDrvIns->iInstance, rc));
221 break;
222 }
223 if (pThis->TCPConnection != -1)
224 {
225 LogRel(("DrvTCP%d: only single connection supported\n", pThis->pDrvIns->iInstance));
226#ifdef RT_OS_WINDOWS
227 closesocket(s);
228#else
229 close(s);
230#endif
231 }
232 else
233 pThis->TCPConnection = s;
234 }
235
236 return VINF_SUCCESS;
237}
238
239/* -=-=-=-=- PDMDRVREG -=-=-=-=- */
240
241/**
242 * Common worker for drvTCPPowerOff and drvTCPDestructor.
243 *
244 * @param pThis The instance data.
245 */
246static void drvTCPShutdownListener(PDRVTCP pThis)
247{
248 /*
249 * Signal shutdown of the listener thread.
250 */
251 pThis->fShutdown = true;
252 if ( pThis->fIsServer
253 && pThis->TCPServer != -1)
254 {
255 int rc = shutdown(pThis->TCPServer, SHUT_RDWR);
256 AssertRC(rc == 0); NOREF(rc);
257
258#ifdef RT_OS_WINDOWS
259 rc = closesocket(pThis->TCPServer);
260#else
261 rc = close(pThis->TCPServer);
262#endif
263 AssertRC(rc == 0);
264 pThis->TCPServer = -1;
265 }
266}
267
268
269/**
270 * Power off a TCP socket stream driver instance.
271 *
272 * This does most of the destruction work, to avoid ordering dependencies.
273 *
274 * @param pDrvIns The driver instance data.
275 */
276static DECLCALLBACK(void) drvTCPPowerOff(PPDMDRVINS pDrvIns)
277{
278 PDRVTCP pThis = PDMINS_2_DATA(pDrvIns, PDRVTCP);
279 LogFlow(("%s: %s\n", __FUNCTION__, pThis->pszLocation));
280
281 drvTCPShutdownListener(pThis);
282}
283
284
285/**
286 * Destruct a TCP socket stream driver instance.
287 *
288 * Most VM resources are freed by the VM. This callback is provided so that
289 * any non-VM resources can be freed correctly.
290 *
291 * @param pDrvIns The driver instance data.
292 */
293static DECLCALLBACK(void) drvTCPDestruct(PPDMDRVINS pDrvIns)
294{
295 PDRVTCP pThis = PDMINS_2_DATA(pDrvIns, PDRVTCP);
296 LogFlow(("%s: %s\n", __FUNCTION__, pThis->pszLocation));
297 PDMDRV_CHECK_VERSIONS_RETURN_VOID(pDrvIns);
298
299 drvTCPShutdownListener(pThis);
300
301 /*
302 * While the thread exits, clean up as much as we can.
303 */
304
305 Assert(pThis->TCPServer == -1);
306 if (pThis->TCPConnection != -1)
307 {
308 int rc = shutdown(pThis->TCPConnection, SHUT_RDWR);
309 AssertRC(rc == 0); NOREF(rc);
310
311#ifdef RT_OS_WINDOWS
312 rc = closesocket(pThis->TCPConnection);
313#else
314 rc = close(pThis->TCPConnection);
315#endif
316 Assert(rc == 0);
317 pThis->TCPConnection = -1;
318 }
319 if ( pThis->fIsServer
320 && pThis->pszLocation)
321 RTFileDelete(pThis->pszLocation);
322
323
324 MMR3HeapFree(pThis->pszLocation);
325 pThis->pszLocation = NULL;
326
327 /*
328 * Wait for the thread.
329 */
330 if (pThis->ListenThread != NIL_RTTHREAD)
331 {
332 int rc = RTThreadWait(pThis->ListenThread, 30000, NULL);
333 if (RT_SUCCESS(rc))
334 pThis->ListenThread = NIL_RTTHREAD;
335 else
336 LogRel(("DrvTCP%d: listen thread did not terminate (%Rrc)\n", pDrvIns->iInstance, rc));
337 }
338
339}
340
341
342/**
343 * Construct a TCP socket stream driver instance.
344 *
345 * @copydoc FNPDMDRVCONSTRUCT
346 */
347static DECLCALLBACK(int) drvTCPConstruct(PPDMDRVINS pDrvIns, PCFGMNODE pCfg, uint32_t fFlags)
348{
349 PDRVTCP pThis = PDMINS_2_DATA(pDrvIns, PDRVTCP);
350 PDMDRV_CHECK_VERSIONS_RETURN(pDrvIns);
351
352#ifdef RT_OS_WINDOWS
353 {
354 WSADATA wsaData;
355 int err;
356
357 err = WSAStartup(MAKEWORD(2,2), &wsaData);
358 if (err != 0)
359 {
360 LogRel(("DrvTCP: Failed to initialize Winsock, error %d\n", err));
361 /* XXX: let socket creation fail below */
362 }
363 }
364#endif
365
366 /*
367 * Init the static parts.
368 */
369 pThis->pDrvIns = pDrvIns;
370 pThis->pszLocation = NULL;
371 pThis->fIsServer = false;
372
373 pThis->TCPServer = -1;
374 pThis->TCPConnection = -1;
375
376 pThis->ListenThread = NIL_RTTHREAD;
377 pThis->fShutdown = false;
378 /* IBase */
379 pDrvIns->IBase.pfnQueryInterface = drvTCPQueryInterface;
380 /* IStream */
381 pThis->IStream.pfnRead = drvTCPRead;
382 pThis->IStream.pfnWrite = drvTCPWrite;
383
384 /*
385 * Validate and read the configuration.
386 */
387 PDMDRV_VALIDATE_CONFIG_RETURN(pDrvIns, "Location|IsServer", "");
388
389 int rc = CFGMR3QueryStringAlloc(pCfg, "Location", &pThis->pszLocation);
390 if (RT_FAILURE(rc))
391 return PDMDrvHlpVMSetError(pDrvIns, rc, RT_SRC_POS,
392 N_("Configuration error: querying \"Location\" resulted in %Rrc"), rc);
393 rc = CFGMR3QueryBool(pCfg, "IsServer", &pThis->fIsServer);
394 if (RT_FAILURE(rc))
395 return PDMDrvHlpVMSetError(pDrvIns, rc, RT_SRC_POS,
396 N_("Configuration error: querying \"IsServer\" resulted in %Rrc"), rc);
397
398 /*
399 * Create/Open the socket.
400 */
401 int s = socket(PF_INET, SOCK_STREAM, 0);
402 if (s == -1)
403 return PDMDrvHlpVMSetError(pDrvIns, RTErrConvertFromErrno(errno), RT_SRC_POS,
404 N_("DrvTCP#%d failed to create socket"), pDrvIns->iInstance);
405
406 struct sockaddr_in addr;
407 memset(&addr, 0, sizeof(addr));
408 addr.sin_family = AF_INET;
409
410 if (pThis->fIsServer)
411 {
412 addr.sin_addr.s_addr = INADDR_ANY;
413 addr.sin_port = htons(/*PORT*/ atoi(pThis->pszLocation));
414
415 /* Bind address to the telnet socket. */
416 pThis->TCPServer = s;
417 RTFileDelete(pThis->pszLocation);
418 if (bind(s, (struct sockaddr *)&addr, sizeof(addr)) == -1)
419 return PDMDrvHlpVMSetError(pDrvIns, RTErrConvertFromErrno(errno), RT_SRC_POS,
420 N_("DrvTCP#%d failed to bind to socket %s"),
421 pDrvIns->iInstance, pThis->pszLocation);
422 rc = RTThreadCreate(&pThis->ListenThread, drvTCPListenLoop, (void *)pThis, 0,
423 RTTHREADTYPE_IO, RTTHREADFLAGS_WAITABLE, "DrvTCPStream");
424 if (RT_FAILURE(rc))
425 return PDMDrvHlpVMSetError(pDrvIns, rc, RT_SRC_POS,
426 N_("DrvTCP#%d failed to create listening thread"), pDrvIns->iInstance);
427 }
428 else
429 {
430 char *token;
431 const char *delim = ":";
432 struct hostent *server;
433 token = strtok(pThis->pszLocation, delim);
434 if(token) {
435 server = gethostbyname(token);
436 memmove((char *)&addr.sin_addr.s_addr,
437 (char *)server->h_addr,
438 server->h_length);
439 }
440 token = strtok(NULL, delim);
441 if(token) {
442 addr.sin_port = htons(/*PORT*/ atoi(token));
443 }
444
445 /* Connect to the telnet socket. */
446 pThis->TCPConnection = s;
447 if (connect(s, (struct sockaddr *)&addr, sizeof(addr)) == -1)
448 return PDMDrvHlpVMSetError(pDrvIns, RTErrConvertFromErrno(errno), RT_SRC_POS,
449 N_("DrvTCP#%d failed to connect to socket %s"),
450 pDrvIns->iInstance, pThis->pszLocation);
451 }
452
453 LogRel(("DrvTCP: %s, %s\n", pThis->pszLocation, pThis->fIsServer ? "server" : "client"));
454 return VINF_SUCCESS;
455}
456
457
458/**
459 * TCP stream driver registration record.
460 */
461const PDMDRVREG g_DrvTCP =
462{
463 /* u32Version */
464 PDM_DRVREG_VERSION,
465 /* szName */
466 "TCP",
467 /* szRCMod */
468 "",
469 /* szR0Mod */
470 "",
471 /* pszDescription */
472 "TCP serial stream driver.",
473 /* fFlags */
474 PDM_DRVREG_FLAGS_HOST_BITS_DEFAULT,
475 /* fClass. */
476 PDM_DRVREG_CLASS_STREAM,
477 /* cMaxInstances */
478 ~0U,
479 /* cbInstance */
480 sizeof(DRVTCP),
481 /* pfnConstruct */
482 drvTCPConstruct,
483 /* pfnDestruct */
484 drvTCPDestruct,
485 /* pfnRelocate */
486 NULL,
487 /* pfnIOCtl */
488 NULL,
489 /* pfnPowerOn */
490 NULL,
491 /* pfnReset */
492 NULL,
493 /* pfnSuspend */
494 NULL,
495 /* pfnResume */
496 NULL,
497 /* pfnAttach */
498 NULL,
499 /* pfnDetach */
500 NULL,
501 /* pfnPowerOff */
502 drvTCPPowerOff,
503 /* pfnSoftReset */
504 NULL,
505 /* u32EndVersion */
506 PDM_DRVREG_VERSION
507};
508
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