VirtualBox

source: vbox/trunk/src/VBox/NetworkServices/DHCP/VBoxNetDHCP.cpp@ 63989

Last change on this file since 63989 was 63301, checked in by vboxsync, 8 years ago

DHCP: DIFx killable window seems to have to lost when we enabled hardening (will do same for NAT in a bit).

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 24.5 KB
Line 
1/* $Id: VBoxNetDHCP.cpp 63301 2016-08-10 20:45:33Z vboxsync $ */
2/** @file
3 * VBoxNetDHCP - DHCP Service for connecting to IntNet.
4 */
5
6/*
7 * Copyright (C) 2009-2016 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18/** @page pg_net_dhcp VBoxNetDHCP
19 *
20 * Write a few words...
21 *
22 */
23
24
25/*********************************************************************************************************************************
26* Header Files *
27*********************************************************************************************************************************/
28#include <VBox/com/com.h>
29#include <VBox/com/listeners.h>
30#include <VBox/com/string.h>
31#include <VBox/com/Guid.h>
32#include <VBox/com/array.h>
33#include <VBox/com/ErrorInfo.h>
34#include <VBox/com/errorprint.h>
35#include <VBox/com/EventQueue.h>
36#include <VBox/com/VirtualBox.h>
37
38#include <iprt/alloca.h>
39#include <iprt/buildconfig.h>
40#include <iprt/err.h>
41#include <iprt/net.h> /* must come before getopt */
42#include <iprt/getopt.h>
43#include <iprt/initterm.h>
44#include <iprt/message.h>
45#include <iprt/param.h>
46#include <iprt/path.h>
47#include <iprt/stream.h>
48#include <iprt/time.h>
49#include <iprt/string.h>
50#ifdef RT_OS_WINDOWS
51# include <iprt/thread.h>
52#endif
53
54#include <VBox/sup.h>
55#include <VBox/intnet.h>
56#include <VBox/intnetinline.h>
57#include <VBox/vmm/vmm.h>
58#include <VBox/version.h>
59
60#include "../NetLib/VBoxNetLib.h"
61#include "../NetLib/shared_ptr.h"
62
63#include <vector>
64#include <list>
65#include <string>
66#include <map>
67
68#include "../NetLib/VBoxNetBaseService.h"
69#include "../NetLib/utils.h"
70
71#ifdef RT_OS_WINDOWS /* WinMain */
72# include <iprt/win/windows.h>
73# include <stdlib.h>
74# ifdef INET_ADDRSTRLEN
75/* On Windows INET_ADDRSTRLEN defined as 22 Ws2ipdef.h, because it include port number */
76# undef INET_ADDRSTRLEN
77# endif
78# define INET_ADDRSTRLEN 16
79#else
80# include <netinet/in.h>
81#endif
82
83
84#include "Config.h"
85
86
87/*********************************************************************************************************************************
88* Structures and Typedefs *
89*********************************************************************************************************************************/
90/**
91 * DHCP server instance.
92 */
93class VBoxNetDhcp : public VBoxNetBaseService, public NATNetworkEventAdapter
94{
95public:
96 VBoxNetDhcp();
97 virtual ~VBoxNetDhcp();
98
99 int init();
100 void done();
101 void usage(void) { /* XXX: document options */ };
102 int parseOpt(int rc, const RTGETOPTUNION& getOptVal);
103 int processFrame(void *, size_t) {return VERR_IGNORED; };
104 int processGSO(PCPDMNETWORKGSO, size_t) {return VERR_IGNORED; };
105 int processUDP(void *, size_t);
106
107protected:
108 bool handleDhcpMsg(uint8_t uMsgType, PCRTNETBOOTP pDhcpMsg, size_t cb);
109
110 void debugPrintV(int32_t iMinLevel, bool fMsg, const char *pszFmt, va_list va) const;
111 static const char *debugDhcpName(uint8_t uMsgType);
112
113private:
114 int initNoMain();
115 int initWithMain();
116 HRESULT HandleEvent(VBoxEventType_T aEventType, IEvent *pEvent);
117 int fetchAndUpdateDnsInfo();
118
119protected:
120 /** @name The DHCP server specific configuration data members.
121 * @{ */
122 /*
123 * XXX: what was the plan? SQL3 or plain text file?
124 * How it will coexists with managment from VBoxManagement, who should manage db
125 * in that case (VBoxManage, VBoxSVC ???)
126 */
127 std::string m_LeaseDBName;
128
129 /** @} */
130
131 /* corresponding dhcp server description in Main */
132 ComPtr<IDHCPServer> m_DhcpServer;
133
134 ComPtr<INATNetwork> m_NATNetwork;
135
136 /** Listener for Host DNS changes */
137 ComNatListenerPtr m_VBoxListener;
138 ComNatListenerPtr m_VBoxClientListener;
139
140 NetworkManager *m_NetworkManager;
141
142 /*
143 * We will ignore cmd line parameters IFF there will be some DHCP specific arguments
144 * otherwise all paramters will come from Main.
145 */
146 bool m_fIgnoreCmdLineParameters;
147
148 /*
149 * -b -n 10.0.1.2 -m 255.255.255.0 -> to the list processing in
150 */
151 typedef struct
152 {
153 char Key;
154 std::string strValue;
155 } CMDLNPRM;
156 std::list<CMDLNPRM> CmdParameterll;
157 typedef std::list<CMDLNPRM>::iterator CmdParameterIterator;
158
159 /** @name Debug stuff
160 * @{ */
161 int32_t m_cVerbosity;
162 uint8_t m_uCurMsgType;
163 size_t m_cbCurMsg;
164 PCRTNETBOOTP m_pCurMsg;
165 VBOXNETUDPHDRS m_CurHdrs;
166 /** @} */
167};
168
169
170static inline int configGetBoundryAddress(const ComDhcpServerPtr& dhcp, bool fUpperBoundry, RTNETADDRIPV4& boundryAddress)
171{
172 boundryAddress.u = INADDR_ANY;
173
174 HRESULT hrc;
175 com::Bstr strAddress;
176 if (fUpperBoundry)
177 hrc = dhcp->COMGETTER(UpperIP)(strAddress.asOutParam());
178 else
179 hrc = dhcp->COMGETTER(LowerIP)(strAddress.asOutParam());
180 AssertComRCReturn(hrc, VERR_INTERNAL_ERROR);
181
182 return RTNetStrToIPv4Addr(com::Utf8Str(strAddress).c_str(), &boundryAddress);
183}
184
185
186/*********************************************************************************************************************************
187* Global Variables *
188*********************************************************************************************************************************/
189/** Pointer to the DHCP server. */
190static VBoxNetDhcp *g_pDhcp;
191
192/* DHCP server specific options */
193static RTGETOPTDEF g_aOptionDefs[] =
194{
195 { "--lease-db", 'D', RTGETOPT_REQ_STRING },
196 { "--begin-config", 'b', RTGETOPT_REQ_NOTHING },
197 { "--gateway", 'g', RTGETOPT_REQ_IPV4ADDR },
198 { "--lower-ip", 'l', RTGETOPT_REQ_IPV4ADDR },
199 { "--upper-ip", 'u', RTGETOPT_REQ_IPV4ADDR },
200};
201
202/**
203 * Construct a DHCP server with a default configuration.
204 */
205VBoxNetDhcp::VBoxNetDhcp()
206 : VBoxNetBaseService("VBoxNetDhcp", "VBoxNetDhcp"),
207 m_NetworkManager(NULL)
208{
209 /* m_enmTrunkType = kIntNetTrunkType_WhateverNone; */
210 RTMAC mac;
211 mac.au8[0] = 0x08;
212 mac.au8[1] = 0x00;
213 mac.au8[2] = 0x27;
214 mac.au8[3] = 0x40;
215 mac.au8[4] = 0x41;
216 mac.au8[5] = 0x42;
217 setMacAddress(mac);
218
219 RTNETADDRIPV4 address;
220 address.u = RT_H2N_U32_C(RT_BSWAP_U32_C(RT_MAKE_U32_FROM_U8( 10, 0, 2, 5)));
221 setIpv4Address(address);
222
223 setSendBufSize(8 * _1K);
224 setRecvBufSize(50 * _1K);
225
226 m_uCurMsgType = UINT8_MAX;
227 m_cbCurMsg = 0;
228 m_pCurMsg = NULL;
229 RT_ZERO(m_CurHdrs);
230
231 m_fIgnoreCmdLineParameters = true;
232
233 for(unsigned int i = 0; i < RT_ELEMENTS(g_aOptionDefs); ++i)
234 addCommandLineOption(&g_aOptionDefs[i]);
235}
236
237
238/**
239 * Destruct a DHCP server.
240 */
241VBoxNetDhcp::~VBoxNetDhcp()
242{
243}
244
245
246
247
248/**
249 * Parse the DHCP specific arguments.
250 *
251 * This callback caled for each paramenter so
252 * ....
253 * we nee post analisys of the parameters, at least
254 * for -b, -g, -l, -u, -m
255 */
256int VBoxNetDhcp::parseOpt(int rc, const RTGETOPTUNION& Val)
257{
258 CMDLNPRM prm;
259
260 /* Ok, we've entered here, thus we can't ignore cmd line parameters anymore */
261 m_fIgnoreCmdLineParameters = false;
262
263 prm.Key = rc;
264
265 switch (rc)
266 {
267 case 'l':
268 case 'u':
269 case 'g':
270 {
271 char buf[17];
272 RTStrPrintf(buf, 17, "%RTnaipv4", Val.IPv4Addr.u);
273 prm.strValue = buf;
274 CmdParameterll.push_back(prm);
275 }
276 break;
277
278 case 'b': // ignore
279 case 'D': // ignore
280 break;
281
282 default:
283 rc = RTGetOptPrintError(rc, &Val);
284 RTPrintf("Use --help for more information.\n");
285 return rc;
286 }
287
288 return VINF_SUCCESS;
289}
290
291int VBoxNetDhcp::init()
292{
293 int rc = this->VBoxNetBaseService::init();
294 AssertRCReturn(rc, rc);
295
296 if (isMainNeeded())
297 rc = initWithMain();
298 else
299 rc = initNoMain();
300 AssertRCReturn(rc, rc);
301
302 m_NetworkManager = NetworkManager::getNetworkManager(m_DhcpServer);
303 AssertPtrReturn(m_NetworkManager, VERR_INTERNAL_ERROR);
304
305 m_NetworkManager->setOurAddress(getIpv4Address());
306 m_NetworkManager->setOurNetmask(getIpv4Netmask());
307 m_NetworkManager->setOurMac(getMacAddress());
308 m_NetworkManager->setService(this);
309
310 return VINF_SUCCESS;
311}
312
313void VBoxNetDhcp::done()
314{
315 destroyNatListener(m_VBoxListener, virtualbox);
316 destroyClientListener(m_VBoxClientListener, virtualboxClient);
317}
318
319int VBoxNetDhcp::processUDP(void *pv, size_t cbPv)
320{
321 PCRTNETBOOTP pDhcpMsg = (PCRTNETBOOTP)pv;
322 m_pCurMsg = pDhcpMsg;
323 m_cbCurMsg = cbPv;
324
325 uint8_t uMsgType;
326 if (RTNetIPv4IsDHCPValid(NULL /* why is this here? */, pDhcpMsg, cbPv, &uMsgType))
327 {
328 m_uCurMsgType = uMsgType;
329 {
330 /* To avoid fight with event processing thread */
331 VBoxNetALock(this);
332 handleDhcpMsg(uMsgType, pDhcpMsg, cbPv);
333 }
334 m_uCurMsgType = UINT8_MAX;
335 }
336 else
337 debugPrint(1, true, "VBoxNetDHCP: Skipping invalid DHCP packet.\n"); /** @todo handle pure bootp clients too? */
338
339 m_pCurMsg = NULL;
340 m_cbCurMsg = 0;
341
342 return VINF_SUCCESS;
343}
344
345
346/**
347 * Handles a DHCP message.
348 *
349 * @returns true if handled, false if not. (IGNORED BY CALLER)
350 * @param uMsgType The message type.
351 * @param pDhcpMsg The DHCP message.
352 * @param cb The size of the DHCP message.
353 */
354bool VBoxNetDhcp::handleDhcpMsg(uint8_t uMsgType, PCRTNETBOOTP pDhcpMsg, size_t cb)
355{
356 if (pDhcpMsg->bp_op == RTNETBOOTP_OP_REQUEST)
357 {
358 AssertPtrReturn(m_NetworkManager, false);
359
360 switch (uMsgType)
361 {
362 case RTNET_DHCP_MT_DISCOVER:
363 return m_NetworkManager->handleDhcpReqDiscover(pDhcpMsg, cb);
364
365 case RTNET_DHCP_MT_REQUEST:
366 return m_NetworkManager->handleDhcpReqRequest(pDhcpMsg, cb);
367
368 case RTNET_DHCP_MT_DECLINE:
369 return m_NetworkManager->handleDhcpReqDecline(pDhcpMsg, cb);
370
371 case RTNET_DHCP_MT_RELEASE:
372 return m_NetworkManager->handleDhcpReqRelease(pDhcpMsg, cb);
373
374 case RTNET_DHCP_MT_INFORM:
375 debugPrint(0, true, "Should we handle this?");
376 break;
377
378 default:
379 debugPrint(0, true, "Unexpected.");
380 break;
381 }
382 }
383 return false;
384}
385
386/**
387 * Print debug message depending on the m_cVerbosity level.
388 *
389 * @param iMinLevel The minimum m_cVerbosity level for this message.
390 * @param fMsg Whether to dump parts for the current DHCP message.
391 * @param pszFmt The message format string.
392 * @param va Optional arguments.
393 */
394void VBoxNetDhcp::debugPrintV(int iMinLevel, bool fMsg, const char *pszFmt, va_list va) const
395{
396 if (iMinLevel <= m_cVerbosity)
397 {
398 va_list vaCopy; /* This dude is *very* special, thus the copy. */
399 va_copy(vaCopy, va);
400 RTStrmPrintf(g_pStdErr, "VBoxNetDHCP: %s: %N\n", iMinLevel >= 2 ? "debug" : "info", pszFmt, &vaCopy);
401 va_end(vaCopy);
402
403 if ( fMsg
404 && m_cVerbosity >= 2
405 && m_pCurMsg)
406 {
407 /* XXX: export this to debugPrinfDhcpMsg or variant and other method export
408 * to base class
409 */
410 const char *pszMsg = m_uCurMsgType != UINT8_MAX ? debugDhcpName(m_uCurMsgType) : "";
411 RTStrmPrintf(g_pStdErr, "VBoxNetDHCP: debug: %8s chaddr=%.6Rhxs ciaddr=%d.%d.%d.%d yiaddr=%d.%d.%d.%d siaddr=%d.%d.%d.%d xid=%#x\n",
412 pszMsg,
413 &m_pCurMsg->bp_chaddr,
414 m_pCurMsg->bp_ciaddr.au8[0], m_pCurMsg->bp_ciaddr.au8[1], m_pCurMsg->bp_ciaddr.au8[2], m_pCurMsg->bp_ciaddr.au8[3],
415 m_pCurMsg->bp_yiaddr.au8[0], m_pCurMsg->bp_yiaddr.au8[1], m_pCurMsg->bp_yiaddr.au8[2], m_pCurMsg->bp_yiaddr.au8[3],
416 m_pCurMsg->bp_siaddr.au8[0], m_pCurMsg->bp_siaddr.au8[1], m_pCurMsg->bp_siaddr.au8[2], m_pCurMsg->bp_siaddr.au8[3],
417 m_pCurMsg->bp_xid);
418 }
419 }
420}
421
422
423/**
424 * Gets the name of given DHCP message type.
425 *
426 * @returns Readonly name.
427 * @param uMsgType The message number.
428 */
429/* static */ const char *VBoxNetDhcp::debugDhcpName(uint8_t uMsgType)
430{
431 switch (uMsgType)
432 {
433 case 0: return "MT_00";
434 case RTNET_DHCP_MT_DISCOVER: return "DISCOVER";
435 case RTNET_DHCP_MT_OFFER: return "OFFER";
436 case RTNET_DHCP_MT_REQUEST: return "REQUEST";
437 case RTNET_DHCP_MT_DECLINE: return "DECLINE";
438 case RTNET_DHCP_MT_ACK: return "ACK";
439 case RTNET_DHCP_MT_NAC: return "NAC";
440 case RTNET_DHCP_MT_RELEASE: return "RELEASE";
441 case RTNET_DHCP_MT_INFORM: return "INFORM";
442 case 9: return "MT_09";
443 case 10: return "MT_0a";
444 case 11: return "MT_0b";
445 case 12: return "MT_0c";
446 case 13: return "MT_0d";
447 case 14: return "MT_0e";
448 case 15: return "MT_0f";
449 case 16: return "MT_10";
450 case 17: return "MT_11";
451 case 18: return "MT_12";
452 case 19: return "MT_13";
453 case UINT8_MAX: return "MT_ff";
454 default: return "UNKNOWN";
455 }
456}
457
458
459int VBoxNetDhcp::initNoMain()
460{
461 CmdParameterIterator it;
462
463 RTNETADDRIPV4 address = getIpv4Address();
464 RTNETADDRIPV4 netmask = getIpv4Netmask();
465 RTNETADDRIPV4 networkId;
466 networkId.u = address.u & netmask.u;
467
468 RTNETADDRIPV4 UpperAddress;
469 RTNETADDRIPV4 LowerAddress = networkId;
470 UpperAddress.u = RT_H2N_U32(RT_N2H_U32(LowerAddress.u) | RT_N2H_U32(netmask.u));
471
472 for (it = CmdParameterll.begin(); it != CmdParameterll.end(); ++it)
473 {
474 switch(it->Key)
475 {
476 case 'l':
477 RTNetStrToIPv4Addr(it->strValue.c_str(), &LowerAddress);
478 break;
479
480 case 'u':
481 RTNetStrToIPv4Addr(it->strValue.c_str(), &UpperAddress);
482 break;
483 case 'b':
484 break;
485
486 }
487 }
488
489 ConfigurationManager *confManager = ConfigurationManager::getConfigurationManager();
490 AssertPtrReturn(confManager, VERR_INTERNAL_ERROR);
491 confManager->addNetwork(unconst(g_RootConfig),
492 networkId,
493 netmask,
494 LowerAddress,
495 UpperAddress);
496
497 return VINF_SUCCESS;
498}
499
500
501int VBoxNetDhcp::initWithMain()
502{
503 /* ok, here we should initiate instance of dhcp server
504 * and listener for Dhcp configuration events
505 */
506 AssertRCReturn(virtualbox.isNull(), VERR_INTERNAL_ERROR);
507 std::string networkName = getNetworkName();
508
509 int rc = findDhcpServer(virtualbox, networkName, m_DhcpServer);
510 AssertRCReturn(rc, rc);
511
512 rc = findNatNetwork(virtualbox, networkName, m_NATNetwork);
513 AssertRCReturn(rc, rc);
514
515 BOOL fNeedDhcpServer = isDhcpRequired(m_NATNetwork);
516 if (!fNeedDhcpServer)
517 return VERR_CANCELLED;
518
519 RTNETADDRIPV4 gateway;
520 com::Bstr strGateway;
521 HRESULT hrc = m_NATNetwork->COMGETTER(Gateway)(strGateway.asOutParam());
522 AssertComRCReturn(hrc, VERR_INTERNAL_ERROR);
523 RTNetStrToIPv4Addr(com::Utf8Str(strGateway).c_str(), &gateway);
524
525 ConfigurationManager *confManager = ConfigurationManager::getConfigurationManager();
526 AssertPtrReturn(confManager, VERR_INTERNAL_ERROR);
527 confManager->addToAddressList(RTNET_DHCP_OPT_ROUTERS, gateway);
528
529 rc = fetchAndUpdateDnsInfo();
530 AssertMsgRCReturn(rc, ("Wasn't able to fetch Dns info"), rc);
531
532 {
533 ComEventTypeArray eventTypes;
534 eventTypes.push_back(VBoxEventType_OnHostNameResolutionConfigurationChange);
535 eventTypes.push_back(VBoxEventType_OnNATNetworkStartStop);
536 rc = createNatListener(m_VBoxListener, virtualbox, this, eventTypes);
537 AssertRCReturn(rc, rc);
538 }
539
540 {
541 ComEventTypeArray eventTypes;
542 eventTypes.push_back(VBoxEventType_OnVBoxSVCAvailabilityChanged);
543 rc = createClientListener(m_VBoxClientListener, virtualboxClient, this, eventTypes);
544 AssertRCReturn(rc, rc);
545 }
546
547 RTNETADDRIPV4 LowerAddress;
548 rc = configGetBoundryAddress(m_DhcpServer, false, LowerAddress);
549 AssertMsgRCReturn(rc, ("can't get lower boundrary adderss'"),rc);
550
551 RTNETADDRIPV4 UpperAddress;
552 rc = configGetBoundryAddress(m_DhcpServer, true, UpperAddress);
553 AssertMsgRCReturn(rc, ("can't get upper boundrary adderss'"),rc);
554
555 RTNETADDRIPV4 address = getIpv4Address();
556 RTNETADDRIPV4 netmask = getIpv4Netmask();
557 RTNETADDRIPV4 networkId = networkid(address, netmask);
558 std::string name = std::string("default");
559
560 confManager->addNetwork(unconst(g_RootConfig),
561 networkId,
562 netmask,
563 LowerAddress,
564 UpperAddress);
565
566 com::Bstr bstr;
567 hrc = virtualbox->COMGETTER(HomeFolder)(bstr.asOutParam());
568 com::Utf8StrFmt strXmlLeaseFile("%ls%c%s.leases",
569 bstr.raw(), RTPATH_DELIMITER, networkName.c_str());
570 confManager->loadFromFile(strXmlLeaseFile);
571
572 return VINF_SUCCESS;
573}
574
575
576int VBoxNetDhcp::fetchAndUpdateDnsInfo()
577{
578 ComHostPtr host;
579 if (SUCCEEDED(virtualbox->COMGETTER(Host)(host.asOutParam())))
580 {
581 AddressToOffsetMapping mapIp4Addr2Off;
582 int rc = localMappings(m_NATNetwork, mapIp4Addr2Off);
583 /* XXX: here could be several cases: 1. COM error, 2. not found (empty) 3. ? */
584 AssertMsgRCReturn(rc, ("Can't fetch local mappings"), rc);
585
586 RTNETADDRIPV4 address = getIpv4Address();
587 RTNETADDRIPV4 netmask = getIpv4Netmask();
588
589 AddressList nameservers;
590 rc = hostDnsServers(host, networkid(address, netmask), mapIp4Addr2Off, nameservers);
591 AssertMsgRCReturn(rc, ("Debug me!!!"), rc);
592 /* XXX: Search strings */
593
594 std::string domain;
595 rc = hostDnsDomain(host, domain);
596 AssertMsgRCReturn(rc, ("Debug me!!"), rc);
597
598 {
599 VBoxNetALock(this);
600 ConfigurationManager *confManager = ConfigurationManager::getConfigurationManager();
601 confManager->flushAddressList(RTNET_DHCP_OPT_DNS);
602
603 for (AddressList::iterator it = nameservers.begin(); it != nameservers.end(); ++it)
604 confManager->addToAddressList(RTNET_DHCP_OPT_DNS, *it);
605
606 confManager->setString(RTNET_DHCP_OPT_DOMAIN_NAME, domain);
607 }
608 }
609
610 return VINF_SUCCESS;
611}
612
613HRESULT VBoxNetDhcp::HandleEvent(VBoxEventType_T aEventType, IEvent *pEvent)
614{
615 switch (aEventType)
616 {
617 case VBoxEventType_OnHostNameResolutionConfigurationChange:
618 fetchAndUpdateDnsInfo();
619 break;
620
621 case VBoxEventType_OnNATNetworkStartStop:
622 {
623 ComPtr <INATNetworkStartStopEvent> pStartStopEvent = pEvent;
624
625 com::Bstr networkName;
626 HRESULT hrc = pStartStopEvent->COMGETTER(NetworkName)(networkName.asOutParam());
627 AssertComRCReturn(hrc, hrc);
628 if (networkName.compare(getNetworkName().c_str()))
629 break; /* change not for our network */
630
631 BOOL fStart = TRUE;
632 hrc = pStartStopEvent->COMGETTER(StartEvent)(&fStart);
633 AssertComRCReturn(hrc, hrc);
634 if (!fStart)
635 shutdown();
636 break;
637 }
638
639 case VBoxEventType_OnVBoxSVCAvailabilityChanged:
640 {
641 shutdown();
642 break;
643 }
644
645 default: break; /* Shut up MSC. */
646 }
647
648 return S_OK;
649}
650
651#ifdef RT_OS_WINDOWS
652
653/** The class name for the DIFx-killable window. */
654static WCHAR g_wszWndClassName[] = L"VBoxNetDHCPClass";
655/** Whether to exit the process on quit. */
656static bool g_fExitProcessOnQuit = true;
657
658/**
659 * Window procedure for making us DIFx-killable.
660 */
661static LRESULT CALLBACK DIFxKillableWindowProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
662{
663 if (uMsg == WM_DESTROY)
664 {
665 PostQuitMessage(0);
666 return 0;
667 }
668 return DefWindowProc(hwnd, uMsg, wParam, lParam);
669}
670
671/** @callback_method_impl{FNRTTHREAD,
672 * Thread that creates service a window the DIFx can destroy, thereby
673 * triggering process termination. }
674 */
675static DECLCALLBACK(int) DIFxKillableProcessThreadProc(RTTHREAD hThreadSelf, void *pvUser)
676{
677 RT_NOREF(hThreadSelf, pvUser);
678 HINSTANCE hInstance = (HINSTANCE)GetModuleHandle(NULL);
679
680 /* Register the Window Class. */
681 WNDCLASSW WndCls;
682 WndCls.style = 0;
683 WndCls.lpfnWndProc = DIFxKillableWindowProc;
684 WndCls.cbClsExtra = 0;
685 WndCls.cbWndExtra = sizeof(void *);
686 WndCls.hInstance = hInstance;
687 WndCls.hIcon = NULL;
688 WndCls.hCursor = NULL;
689 WndCls.hbrBackground = (HBRUSH)(COLOR_BACKGROUND + 1);
690 WndCls.lpszMenuName = NULL;
691 WndCls.lpszClassName = g_wszWndClassName;
692
693 ATOM atomWindowClass = RegisterClassW(&WndCls);
694 if (atomWindowClass != 0)
695 {
696 /* Create the window. */
697 HWND hwnd = CreateWindowExW(WS_EX_TOOLWINDOW | WS_EX_TRANSPARENT | WS_EX_TOPMOST,
698 g_wszWndClassName, g_wszWndClassName,
699 WS_POPUPWINDOW,
700 -200, -200, 100, 100, NULL, NULL, hInstance, NULL);
701 if (hwnd)
702 {
703 SetWindowPos(hwnd, HWND_TOPMOST, -200, -200, 0, 0,
704 SWP_NOACTIVATE | SWP_HIDEWINDOW | SWP_NOCOPYBITS | SWP_NOREDRAW | SWP_NOSIZE);
705
706 MSG msg;
707 while (GetMessage(&msg, NULL, 0, 0))
708 {
709 TranslateMessage(&msg);
710 DispatchMessage(&msg);
711 }
712
713 DestroyWindow(hwnd);
714 }
715
716 UnregisterClassW(g_wszWndClassName, hInstance);
717
718 if (hwnd && g_fExitProcessOnQuit)
719 exit(0);
720 }
721 return 0;
722}
723
724#endif /* RT_OS_WINDOWS */
725
726/**
727 * Entry point.
728 */
729extern "C" DECLEXPORT(int) TrustedMain(int argc, char **argv)
730{
731 /*
732 * Instantiate the DHCP server and hand it the options.
733 */
734 VBoxNetDhcp *pDhcp = new VBoxNetDhcp();
735 if (!pDhcp)
736 {
737 RTStrmPrintf(g_pStdErr, "VBoxNetDHCP: new VBoxNetDhcp failed!\n");
738 return 1;
739 }
740
741 RTEXITCODE rcExit = (RTEXITCODE)pDhcp->parseArgs(argc - 1, argv + 1);
742 if (rcExit != RTEXITCODE_SUCCESS)
743 return rcExit;
744
745#ifdef RT_OS_WINDOWS
746 /* DIFx hack. */
747 RTTHREAD hMakeUseKillableThread = NIL_RTTHREAD;
748 int rc2 = RTThreadCreate(&hMakeUseKillableThread, DIFxKillableProcessThreadProc, NULL, 0,
749 RTTHREADTYPE_DEFAULT, RTTHREADFLAGS_WAITABLE, "DIFxKill");
750 if (RT_FAILURE(rc2))
751 hMakeUseKillableThread = NIL_RTTHREAD;
752#endif
753
754 pDhcp->init();
755
756 /*
757 * Try connect the server to the network.
758 */
759 int rc = pDhcp->tryGoOnline();
760 if (RT_SUCCESS(rc))
761 {
762 /*
763 * Process requests.
764 */
765 g_pDhcp = pDhcp;
766 rc = pDhcp->run();
767 pDhcp->done();
768
769 g_pDhcp = NULL;
770 }
771 delete pDhcp;
772
773#ifdef RT_OS_WINDOWS
774 /* Kill DIFx hack. */
775 if (hMakeUseKillableThread != NIL_RTTHREAD)
776 {
777 g_fExitProcessOnQuit = false;
778 PostThreadMessage((DWORD)RTThreadGetNative(hMakeUseKillableThread), WM_QUIT, 0, 0);
779 RTThreadWait(hMakeUseKillableThread, RT_MS_1SEC * 5U, NULL);
780 }
781#endif
782
783 return RT_SUCCESS(rc) ? RTEXITCODE_SUCCESS : RTEXITCODE_FAILURE;
784}
785
786
787#ifndef VBOX_WITH_HARDENING
788
789int main(int argc, char **argv)
790{
791 int rc = RTR3InitExe(argc, &argv, RTR3INIT_FLAGS_SUPLIB);
792 if (RT_FAILURE(rc))
793 return RTMsgInitFailure(rc);
794
795 return TrustedMain(argc, argv);
796}
797
798# ifdef RT_OS_WINDOWS
799
800
801
802/** (We don't want a console usually.) */
803int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow)
804{
805 NOREF(hInstance); NOREF(hPrevInstance); NOREF(lpCmdLine); NOREF(nCmdShow);
806 return main(__argc, __argv);
807}
808# endif /* RT_OS_WINDOWS */
809
810#endif /* !VBOX_WITH_HARDENING */
811
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