VirtualBox

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

Last change on this file since 59113 was 57358, checked in by vboxsync, 9 years ago

*: scm cleanup run.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 14.7 KB
Line 
1/* $Id: DrvTCP.cpp 57358 2015-08-14 15:16:38Z vboxsync $ */
2/** @file
3 * TCP socket driver implementing the IStream interface.
4 */
5
6/*
7 * Copyright (C) 2006-2015 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 /*
353 * Init the static parts.
354 */
355 pThis->pDrvIns = pDrvIns;
356 pThis->pszLocation = NULL;
357 pThis->fIsServer = false;
358
359 pThis->TCPServer = -1;
360 pThis->TCPConnection = -1;
361
362 pThis->ListenThread = NIL_RTTHREAD;
363 pThis->fShutdown = false;
364 /* IBase */
365 pDrvIns->IBase.pfnQueryInterface = drvTCPQueryInterface;
366 /* IStream */
367 pThis->IStream.pfnRead = drvTCPRead;
368 pThis->IStream.pfnWrite = drvTCPWrite;
369
370 /*
371 * Validate and read the configuration.
372 */
373 PDMDRV_VALIDATE_CONFIG_RETURN(pDrvIns, "Location|IsServer", "");
374
375 int rc = CFGMR3QueryStringAlloc(pCfg, "Location", &pThis->pszLocation);
376 if (RT_FAILURE(rc))
377 return PDMDrvHlpVMSetError(pDrvIns, rc, RT_SRC_POS,
378 N_("Configuration error: querying \"Location\" resulted in %Rrc"), rc);
379 rc = CFGMR3QueryBool(pCfg, "IsServer", &pThis->fIsServer);
380 if (RT_FAILURE(rc))
381 return PDMDrvHlpVMSetError(pDrvIns, rc, RT_SRC_POS,
382 N_("Configuration error: querying \"IsServer\" resulted in %Rrc"), rc);
383
384 /*
385 * Create/Open the socket.
386 */
387 int s = socket(PF_INET, SOCK_STREAM, 0);
388 if (s == -1)
389 return PDMDrvHlpVMSetError(pDrvIns, RTErrConvertFromErrno(errno), RT_SRC_POS,
390 N_("DrvTCP#%d failed to create socket"), pDrvIns->iInstance);
391
392 struct sockaddr_in addr;
393 memset(&addr, 0, sizeof(addr));
394 addr.sin_family = AF_INET;
395
396 if (pThis->fIsServer)
397 {
398 addr.sin_addr.s_addr = INADDR_ANY;
399 addr.sin_port = htons(/*PORT*/ atoi(pThis->pszLocation));
400
401 /* Bind address to the telnet socket. */
402 pThis->TCPServer = s;
403 RTFileDelete(pThis->pszLocation);
404 if (bind(s, (struct sockaddr *)&addr, sizeof(addr)) == -1)
405 return PDMDrvHlpVMSetError(pDrvIns, RTErrConvertFromErrno(errno), RT_SRC_POS,
406 N_("DrvTCP#%d failed to bind to socket %s"),
407 pDrvIns->iInstance, pThis->pszLocation);
408 rc = RTThreadCreate(&pThis->ListenThread, drvTCPListenLoop, (void *)pThis, 0,
409 RTTHREADTYPE_IO, RTTHREADFLAGS_WAITABLE, "DrvTCPStream");
410 if (RT_FAILURE(rc))
411 return PDMDrvHlpVMSetError(pDrvIns, rc, RT_SRC_POS,
412 N_("DrvTCP#%d failed to create listening thread"), pDrvIns->iInstance);
413 }
414 else
415 {
416 char *token;
417 const char *delim = ":";
418 struct hostent *server;
419 token = strtok(pThis->pszLocation, delim);
420 if(token) {
421 server = gethostbyname(token);
422 memmove((char *)&addr.sin_addr.s_addr,
423 (char *)server->h_addr,
424 server->h_length);
425 }
426 token = strtok(NULL, delim);
427 if(token) {
428 addr.sin_port = htons(/*PORT*/ atoi(token));
429 }
430
431 /* Connect to the telnet socket. */
432 pThis->TCPConnection = s;
433 if (connect(s, (struct sockaddr *)&addr, sizeof(addr)) == -1)
434 return PDMDrvHlpVMSetError(pDrvIns, RTErrConvertFromErrno(errno), RT_SRC_POS,
435 N_("DrvTCP#%d failed to connect to socket %s"),
436 pDrvIns->iInstance, pThis->pszLocation);
437 }
438
439 LogRel(("DrvTCP: %s, %s\n", pThis->pszLocation, pThis->fIsServer ? "server" : "client"));
440 return VINF_SUCCESS;
441}
442
443
444/**
445 * TCP stream driver registration record.
446 */
447const PDMDRVREG g_DrvTCP =
448{
449 /* u32Version */
450 PDM_DRVREG_VERSION,
451 /* szName */
452 "TCP",
453 /* szRCMod */
454 "",
455 /* szR0Mod */
456 "",
457 /* pszDescription */
458 "TCP serial stream driver.",
459 /* fFlags */
460 PDM_DRVREG_FLAGS_HOST_BITS_DEFAULT,
461 /* fClass. */
462 PDM_DRVREG_CLASS_STREAM,
463 /* cMaxInstances */
464 ~0U,
465 /* cbInstance */
466 sizeof(DRVTCP),
467 /* pfnConstruct */
468 drvTCPConstruct,
469 /* pfnDestruct */
470 drvTCPDestruct,
471 /* pfnRelocate */
472 NULL,
473 /* pfnIOCtl */
474 NULL,
475 /* pfnPowerOn */
476 NULL,
477 /* pfnReset */
478 NULL,
479 /* pfnSuspend */
480 NULL,
481 /* pfnResume */
482 NULL,
483 /* pfnAttach */
484 NULL,
485 /* pfnDetach */
486 NULL,
487 /* pfnPowerOff */
488 drvTCPPowerOff,
489 /* pfnSoftReset */
490 NULL,
491 /* u32EndVersion */
492 PDM_DRVREG_VERSION
493};
494
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