VirtualBox

source: vbox/trunk/src/VBox/Main/src-server/HostNetworkInterfaceImpl.cpp@ 94959

Last change on this file since 94959 was 94722, checked in by vboxsync, 3 years ago

Main: Compile fixes when VBOX_WITH_HOSTNETIF_API is not defined.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 25.6 KB
Line 
1/* $Id: HostNetworkInterfaceImpl.cpp 94722 2022-04-27 13:42:50Z vboxsync $ */
2/** @file
3 * VirtualBox COM class implementation
4 */
5
6/*
7 * Copyright (C) 2006-2022 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#define LOG_GROUP LOG_GROUP_MAIN_HOSTNETWORKINTERFACE
19#include "HostNetworkInterfaceImpl.h"
20#include "AutoCaller.h"
21#include "netif.h"
22#ifdef VBOX_WITH_RESOURCE_USAGE_API
23# include "Performance.h"
24# include "PerformanceImpl.h"
25#endif
26#include "LoggingNew.h"
27
28#include <iprt/cpp/utils.h>
29
30#ifdef RT_OS_FREEBSD
31# include <netinet/in.h> /* INADDR_NONE */
32#endif /* RT_OS_FREEBSD */
33
34#include "VirtualBoxImpl.h"
35
36// constructor / destructor
37/////////////////////////////////////////////////////////////////////////////
38
39HostNetworkInterface::HostNetworkInterface()
40 : mVirtualBox(NULL)
41{
42}
43
44HostNetworkInterface::~HostNetworkInterface()
45{
46}
47
48HRESULT HostNetworkInterface::FinalConstruct()
49{
50 return BaseFinalConstruct();
51}
52
53void HostNetworkInterface::FinalRelease()
54{
55 uninit();
56 BaseFinalRelease();
57}
58
59// public initializer/uninitializer for internal purposes only
60/////////////////////////////////////////////////////////////////////////////
61
62/**
63 * Initializes the host object.
64 *
65 * @returns COM result indicator
66 * @param aInterfaceName name of the network interface
67 * @param aShortName short name of the network interface
68 * @param aGuid GUID of the host network interface
69 * @param ifType interface type
70 */
71HRESULT HostNetworkInterface::init(Utf8Str aInterfaceName, Utf8Str aShortName, Guid aGuid, HostNetworkInterfaceType_T ifType)
72{
73 LogFlowThisFunc(("aInterfaceName={%s}, aGuid={%s}\n",
74 aInterfaceName.c_str(), aGuid.toString().c_str()));
75
76 ComAssertRet(!aInterfaceName.isEmpty(), E_INVALIDARG);
77 ComAssertRet(aGuid.isValid(), E_INVALIDARG);
78
79 /* Enclose the state transition NotReady->InInit->Ready */
80 AutoInitSpan autoInitSpan(this);
81 AssertReturn(autoInitSpan.isOk(), E_FAIL);
82
83 unconst(mInterfaceName) = aInterfaceName;
84#ifdef VBOX_WITH_HOSTNETIF_API
85 unconst(mNetworkName) = i_composeNetworkName(aShortName);
86#endif
87 unconst(mShortName) = aShortName;
88 unconst(mGuid) = aGuid;
89 mIfType = ifType;
90
91 /* Confirm a successful initialization */
92 autoInitSpan.setSucceeded();
93
94 return S_OK;
95}
96
97#ifdef VBOX_WITH_RESOURCE_USAGE_API
98
99void HostNetworkInterface::i_registerMetrics(PerformanceCollector *aCollector, ComPtr<IUnknown> objptr)
100{
101 LogFlowThisFunc(("mShortName={%s}, mInterfaceName={%s}, mGuid={%s}, mSpeedMbits=%u\n",
102 mShortName.c_str(), mInterfaceName.c_str(), mGuid.toString().c_str(), m.speedMbits));
103 pm::CollectorHAL *hal = aCollector->getHAL();
104 /* Create sub metrics */
105 Utf8StrFmt strName("Net/%s", mShortName.c_str());
106 pm::SubMetric *networkLoadRx = new pm::SubMetric(strName + "/Load/Rx",
107 "Percentage of network interface receive bandwidth used.");
108 pm::SubMetric *networkLoadTx = new pm::SubMetric(strName + "/Load/Tx",
109 "Percentage of network interface transmit bandwidth used.");
110 pm::SubMetric *networkLinkSpeed = new pm::SubMetric(strName + "/LinkSpeed",
111 "Physical link speed.");
112
113 /* Create and register base metrics */
114 pm::BaseMetric *networkSpeed = new pm::HostNetworkSpeed(hal, objptr, strName + "/LinkSpeed",
115 Utf8Str(mShortName), Utf8Str(mInterfaceName),
116 m.speedMbits, networkLinkSpeed);
117 aCollector->registerBaseMetric(networkSpeed);
118 pm::BaseMetric *networkLoad = new pm::HostNetworkLoadRaw(hal, objptr, strName + "/Load",
119 Utf8Str(mShortName), Utf8Str(mInterfaceName),
120 m.speedMbits, networkLoadRx, networkLoadTx);
121 aCollector->registerBaseMetric(networkLoad);
122
123 aCollector->registerMetric(new pm::Metric(networkSpeed, networkLinkSpeed, 0));
124 aCollector->registerMetric(new pm::Metric(networkSpeed, networkLinkSpeed,
125 new pm::AggregateAvg()));
126 aCollector->registerMetric(new pm::Metric(networkSpeed, networkLinkSpeed,
127 new pm::AggregateMin()));
128 aCollector->registerMetric(new pm::Metric(networkSpeed, networkLinkSpeed,
129 new pm::AggregateMax()));
130
131 aCollector->registerMetric(new pm::Metric(networkLoad, networkLoadRx, 0));
132 aCollector->registerMetric(new pm::Metric(networkLoad, networkLoadRx,
133 new pm::AggregateAvg()));
134 aCollector->registerMetric(new pm::Metric(networkLoad, networkLoadRx,
135 new pm::AggregateMin()));
136 aCollector->registerMetric(new pm::Metric(networkLoad, networkLoadRx,
137 new pm::AggregateMax()));
138
139 aCollector->registerMetric(new pm::Metric(networkLoad, networkLoadTx, 0));
140 aCollector->registerMetric(new pm::Metric(networkLoad, networkLoadTx,
141 new pm::AggregateAvg()));
142 aCollector->registerMetric(new pm::Metric(networkLoad, networkLoadTx,
143 new pm::AggregateMin()));
144 aCollector->registerMetric(new pm::Metric(networkLoad, networkLoadTx,
145 new pm::AggregateMax()));
146}
147
148void HostNetworkInterface::i_unregisterMetrics(PerformanceCollector *aCollector, ComPtr<IUnknown> objptr)
149{
150 LogFlowThisFunc(("mShortName={%s}, mInterfaceName={%s}, mGuid={%s}\n",
151 mShortName.c_str(), mInterfaceName.c_str(), mGuid.toString().c_str()));
152 Utf8StrFmt name("Net/%s", mShortName.c_str());
153 aCollector->unregisterMetricsFor(objptr, name + "/*");
154 aCollector->unregisterBaseMetricsFor(objptr, name);
155}
156
157#endif /* VBOX_WITH_RESOURCE_USAGE_API */
158
159#ifdef VBOX_WITH_HOSTNETIF_API
160#if defined(RT_OS_WINDOWS)
161
162HRESULT HostNetworkInterface::saveAdapterConfigParameter(const char *szParamName, const Utf8Str &strValue)
163{
164 AssertReturn(mVirtualBox != NULL, E_POINTER);
165 return mVirtualBox->SetExtraData(BstrFmt("HostOnly/{%RTuuid}/%s", mGuid.raw(), szParamName).raw(), Bstr(strValue).raw());
166}
167
168HRESULT HostNetworkInterface::eraseAdapterConfigParameter(const char *szParamName)
169{
170 AssertReturn(mVirtualBox != NULL, E_POINTER);
171 return mVirtualBox->SetExtraData(BstrFmt("HostOnly/{%RTuuid}/%s", mGuid.raw(), szParamName).raw(), NULL);
172}
173
174HRESULT HostNetworkInterface::saveAdapterConfigIPv4Dhcp()
175{
176 HRESULT hrc = saveAdapterConfigParameter("IPAddress", "DHCP");
177 if (hrc == S_OK)
178 hrc = eraseAdapterConfigParameter("IPNetMask");
179 return hrc;
180}
181
182HRESULT HostNetworkInterface::saveAdapterConfigIPv4(ULONG addr, ULONG mask)
183{
184 HRESULT hrc = saveAdapterConfigParameter("IPAddress", Utf8StrFmt("%RTnaipv4", addr));
185 if (hrc == S_OK)
186 hrc = saveAdapterConfigParameter("IPNetMask", Utf8StrFmt("%RTnaipv4", mask));
187 return hrc;
188}
189
190HRESULT HostNetworkInterface::saveAdapterConfigIPv6(const Utf8Str& addr, ULONG prefix)
191{
192 HRESULT hrc = saveAdapterConfigParameter("IPV6Address", addr);
193 if (hrc == S_OK)
194 hrc = saveAdapterConfigParameter("IPV6PrefixLen", Utf8StrFmt("%u", prefix));
195 return hrc;
196}
197
198bool HostNetworkInterface::isInConfigFile(void)
199{
200 /* We care about host-only adapters only */
201 if (mIfType != HostNetworkInterfaceType_HostOnly)
202 return true;
203
204 Assert(mVirtualBox != NULL);
205 if (mVirtualBox == NULL)
206 return false; /* Trigger config update, which will fail with proper return code */
207 Bstr tmpName;
208 mVirtualBox->GetExtraData(BstrFmt("HostOnly/{%RTuuid}/Name", mGuid.raw()).raw(), tmpName.asOutParam());
209 return (tmpName.isNotEmpty() && tmpName == mInterfaceName);
210
211}
212
213HRESULT HostNetworkInterface::saveAdapterConfig(void)
214{
215 /* We care about host-only adapters only */
216 if (mIfType != HostNetworkInterfaceType_HostOnly)
217 return true;
218
219 HRESULT hrc = saveAdapterConfigParameter("Name", mInterfaceName.c_str());
220 if (FAILED(hrc))
221 return hrc;
222 if (m.dhcpEnabled)
223 hrc = saveAdapterConfigIPv4Dhcp();
224 else
225 hrc = saveAdapterConfigIPv4(m.IPAddress, m.networkMask);
226 if (SUCCEEDED(hrc))
227 hrc = saveAdapterConfigIPv6(m.IPV6Address.c_str(), m.IPV6NetworkMaskPrefixLength);
228 return hrc;
229}
230
231HRESULT HostNetworkInterface::i_updatePersistentConfig(void)
232{
233 if (mVirtualBox == NULL)
234 return E_POINTER;
235
236 HRESULT hrc = S_OK;
237 if (!isInConfigFile())
238 {
239 hrc = saveAdapterConfig();
240 }
241 return hrc;
242}
243
244#endif /* defined(RT_OS_WINDOWS) */
245
246HRESULT HostNetworkInterface::updateConfig()
247{
248 NETIFINFO info;
249 int rc = NetIfGetConfig(this, &info);
250 if (RT_SUCCESS(rc))
251 {
252 int iPrefixIPv6;
253
254 m.realIPAddress = m.IPAddress = info.IPAddress.u;
255 m.realNetworkMask = m.networkMask = info.IPNetMask.u;
256 m.dhcpEnabled = info.fDhcpEnabled;
257 if (info.IPv6Address.s.Lo || info.IPv6Address.s.Hi)
258 m.realIPV6Address = m.IPV6Address = Utf8StrFmt("%RTnaipv6", &info.IPv6Address);
259 else
260 m.realIPV6Address = m.IPV6Address = Utf8Str::Empty;
261 RTNetMaskToPrefixIPv6(&info.IPv6NetMask, &iPrefixIPv6);
262 m.realIPV6PrefixLength = m.IPV6NetworkMaskPrefixLength = (ULONG)iPrefixIPv6;
263 m.hardwareAddress = Utf8StrFmt("%RTmac", &info.MACAddress);
264 AssertCompile((unsigned)NETIF_T_UNKNOWN == (unsigned)HostNetworkInterfaceMediumType_Unknown);
265 m.mediumType = (HostNetworkInterfaceMediumType_T)info.enmMediumType;
266 AssertCompile((unsigned)NETIF_S_UNKNOWN == (unsigned)HostNetworkInterfaceStatus_Unknown);
267 m.status = (HostNetworkInterfaceStatus_T)info.enmStatus;
268 m.speedMbits = info.uSpeedMbits;
269 m.wireless = info.fWireless;
270 return S_OK;
271 }
272 return rc == VERR_NOT_IMPLEMENTED ? E_NOTIMPL : E_FAIL;
273}
274
275Utf8Str HostNetworkInterface::i_composeNetworkName(const Utf8Str aShortName)
276{
277 return Utf8Str("HostInterfaceNetworking-").append(aShortName);
278}
279/**
280 * Initializes the host object.
281 *
282 * @returns COM result indicator
283 * @param aInterfaceName name of the network interface
284 * @param aGuid GUID of the host network interface
285 */
286HRESULT HostNetworkInterface::init(Utf8Str aInterfaceName, HostNetworkInterfaceType_T ifType, PNETIFINFO pIf)
287{
288// LogFlowThisFunc(("aInterfaceName={%s}, aGuid={%s}\n",
289// aInterfaceName.c_str(), aGuid.toString().c_str()));
290
291// ComAssertRet(aInterfaceName, E_INVALIDARG);
292// ComAssertRet(aGuid.isValid(), E_INVALIDARG);
293 ComAssertRet(pIf, E_INVALIDARG);
294
295 /* Enclose the state transition NotReady->InInit->Ready */
296 AutoInitSpan autoInitSpan(this);
297 AssertReturn(autoInitSpan.isOk(), E_FAIL);
298
299 unconst(mInterfaceName) = aInterfaceName;
300 unconst(mGuid) = pIf->Uuid;
301 if (pIf->szShortName[0])
302 {
303 unconst(mNetworkName) = i_composeNetworkName(pIf->szShortName);
304 unconst(mShortName) = pIf->szShortName;
305 }
306 else
307 {
308 unconst(mNetworkName) = i_composeNetworkName(aInterfaceName);
309 unconst(mShortName) = aInterfaceName;
310 }
311 mIfType = ifType;
312
313 int iPrefixIPv6;
314
315 m.realIPAddress = m.IPAddress = pIf->IPAddress.u;
316 m.realNetworkMask = m.networkMask = pIf->IPNetMask.u;
317 if (pIf->IPv6Address.s.Lo || pIf->IPv6Address.s.Hi)
318 m.realIPV6Address = m.IPV6Address = Utf8StrFmt("%RTnaipv6", &pIf->IPv6Address);
319 else
320 m.realIPV6Address = m.IPV6Address = Utf8Str::Empty;
321 RTNetMaskToPrefixIPv6(&pIf->IPv6NetMask, &iPrefixIPv6);
322 m.realIPV6PrefixLength = m.IPV6NetworkMaskPrefixLength = (ULONG)iPrefixIPv6;
323 m.dhcpEnabled = pIf->fDhcpEnabled;
324 m.hardwareAddress = Utf8StrFmt("%RTmac", &pIf->MACAddress);
325 AssertCompile((unsigned)NETIF_T_UNKNOWN == (unsigned)HostNetworkInterfaceMediumType_Unknown);
326 m.mediumType = (HostNetworkInterfaceMediumType_T)pIf->enmMediumType;
327 AssertCompile((unsigned)NETIF_S_UNKNOWN == (unsigned)HostNetworkInterfaceStatus_Unknown);
328 m.status = (HostNetworkInterfaceStatus_T)pIf->enmStatus;
329 m.speedMbits = pIf->uSpeedMbits;
330 m.wireless = pIf->fWireless;
331
332 /* Confirm a successful initialization */
333 autoInitSpan.setSucceeded();
334
335 return S_OK;
336}
337
338#endif /* VBOX_WITH_HOSTNETIF_API */
339
340// wrapped IHostNetworkInterface properties
341/////////////////////////////////////////////////////////////////////////////
342/**
343 * Returns the name of the host network interface.
344 *
345 * @returns COM status code
346 * @param aInterfaceName - Interface Name
347 */
348
349HRESULT HostNetworkInterface::getName(com::Utf8Str &aInterfaceName)
350{
351 aInterfaceName = mInterfaceName;
352 return S_OK;
353}
354
355/**
356 * Returns the short name of the host network interface.
357 *
358 * @returns COM status code
359 * @param aShortName Short Name
360 */
361
362HRESULT HostNetworkInterface::getShortName(com::Utf8Str &aShortName)
363{
364 aShortName = mShortName;
365
366 return S_OK;
367}
368
369/**
370 * Returns the GUID of the host network interface.
371 *
372 * @returns COM status code
373 * @param aGuid GUID
374 */
375HRESULT HostNetworkInterface::getId(com::Guid &aGuid)
376{
377 aGuid = mGuid;
378
379 return S_OK;
380}
381
382HRESULT HostNetworkInterface::getDHCPEnabled(BOOL *aDHCPEnabled)
383{
384 *aDHCPEnabled = m.dhcpEnabled;
385
386 return S_OK;
387}
388
389
390/**
391 * Returns the IP address of the host network interface.
392 *
393 * @returns COM status code
394 * @param aIPAddress Address name
395 */
396HRESULT HostNetworkInterface::getIPAddress(com::Utf8Str &aIPAddress)
397{
398 in_addr tmp;
399#if defined(RT_OS_WINDOWS)
400 tmp.S_un.S_addr = m.IPAddress;
401#else
402 tmp.s_addr = m.IPAddress;
403#endif
404 char *addr = inet_ntoa(tmp);
405 if (addr)
406 {
407 aIPAddress = addr;
408 return S_OK;
409 }
410
411 return E_FAIL;
412}
413
414/**
415 * Returns the netwok mask of the host network interface.
416 *
417 * @returns COM status code
418 * @param aNetworkMask name.
419 */
420HRESULT HostNetworkInterface::getNetworkMask(com::Utf8Str &aNetworkMask)
421{
422
423 in_addr tmp;
424#if defined(RT_OS_WINDOWS)
425 tmp.S_un.S_addr = m.networkMask;
426#else
427 tmp.s_addr = m.networkMask;
428#endif
429 char *addr = inet_ntoa(tmp);
430 if (addr)
431 {
432 aNetworkMask = Utf8Str(addr);
433 return S_OK;
434 }
435
436 return E_FAIL;
437}
438
439HRESULT HostNetworkInterface::getIPV6Supported(BOOL *aIPV6Supported)
440{
441#if defined(RT_OS_WINDOWS)
442 *aIPV6Supported = FALSE;
443#else
444 *aIPV6Supported = TRUE;
445#endif
446
447 return S_OK;
448}
449
450/**
451 * Returns the IP V6 address of the host network interface.
452 *
453 * @returns COM status code
454 * @param aIPV6Address
455 */
456HRESULT HostNetworkInterface::getIPV6Address(com::Utf8Str &aIPV6Address)
457{
458 aIPV6Address = m.IPV6Address;
459 return S_OK;
460}
461
462/**
463 * Returns the IP V6 network mask prefix length of the host network interface.
464 *
465 * @returns COM status code
466 * @param aIPV6NetworkMaskPrefixLength address of result pointer
467 */
468HRESULT HostNetworkInterface::getIPV6NetworkMaskPrefixLength(ULONG *aIPV6NetworkMaskPrefixLength)
469{
470 *aIPV6NetworkMaskPrefixLength = m.IPV6NetworkMaskPrefixLength;
471
472 return S_OK;
473}
474
475/**
476 * Returns the hardware address of the host network interface.
477 *
478 * @returns COM status code
479 * @param aHardwareAddress hardware address
480 */
481HRESULT HostNetworkInterface::getHardwareAddress(com::Utf8Str &aHardwareAddress)
482{
483 aHardwareAddress = m.hardwareAddress;
484 return S_OK;
485}
486
487/**
488 * Returns the encapsulation protocol type of the host network interface.
489 *
490 * @returns COM status code
491 * @param aType address of result pointer
492 */
493HRESULT HostNetworkInterface::getMediumType(HostNetworkInterfaceMediumType_T *aType)
494{
495 *aType = m.mediumType;
496
497 return S_OK;
498}
499
500/**
501 * Returns the current state of the host network interface.
502 *
503 * @returns COM status code
504 * @param aStatus address of result pointer
505 */
506HRESULT HostNetworkInterface::getStatus(HostNetworkInterfaceStatus_T *aStatus)
507{
508 *aStatus = m.status;
509
510 return S_OK;
511}
512
513/**
514 * Returns network interface type
515 *
516 * @returns COM status code
517 * @param aType address of result pointer
518 */
519HRESULT HostNetworkInterface::getInterfaceType(HostNetworkInterfaceType_T *aType)
520{
521 *aType = mIfType;
522
523 return S_OK;
524
525}
526
527HRESULT HostNetworkInterface::getNetworkName(com::Utf8Str &aNetworkName)
528{
529 aNetworkName = mNetworkName;
530
531 return S_OK;
532}
533
534HRESULT HostNetworkInterface::getWireless(BOOL *aWireless)
535{
536 *aWireless = m.wireless;
537
538 return S_OK;
539}
540
541HRESULT HostNetworkInterface::enableStaticIPConfig(const com::Utf8Str &aIPAddress,
542 const com::Utf8Str &aNetworkMask)
543{
544#ifndef VBOX_WITH_HOSTNETIF_API
545 RT_NOREF(aIPAddress, aNetworkMask);
546 return E_NOTIMPL;
547#else
548 HRESULT hrc;
549
550 if (aIPAddress.isEmpty())
551 {
552 if (m.IPAddress)
553 {
554 int rc = NetIfEnableStaticIpConfig(mVirtualBox, this, m.IPAddress, 0, 0);
555 if (RT_SUCCESS(rc))
556 {
557 m.realIPAddress = 0;
558#if defined(RT_OS_WINDOWS)
559 eraseAdapterConfigParameter("IPAddress");
560 eraseAdapterConfigParameter("IPNetMask");
561#else /* !defined(RT_OS_WINDOWS) */
562 if (FAILED(mVirtualBox->SetExtraData(BstrFmt("HostOnly/%s/IPAddress",
563 mInterfaceName.c_str()).raw(), NULL)))
564 return E_FAIL;
565 if (FAILED(mVirtualBox->SetExtraData(BstrFmt("HostOnly/%s/IPNetMask",
566 mInterfaceName.c_str()).raw(), NULL)))
567 return E_FAIL;
568#endif /* !defined(RT_OS_WINDOWS) */
569 return S_OK;
570 }
571 }
572 else
573 return S_OK;
574 }
575
576 ULONG ip, mask;
577 ip = inet_addr(aIPAddress.c_str());
578 if (ip != INADDR_NONE)
579 {
580 if (aNetworkMask.isEmpty())
581 mask = 0xFFFFFF;
582 else
583 mask = inet_addr(aNetworkMask.c_str());
584 if (mask != INADDR_NONE)
585 {
586 if (m.realIPAddress == ip && m.realNetworkMask == mask)
587 return S_OK;
588 int rc = NetIfEnableStaticIpConfig(mVirtualBox, this, m.IPAddress, ip, mask);
589 if (RT_SUCCESS(rc))
590 {
591 m.realIPAddress = ip;
592 m.realNetworkMask = mask;
593#if defined(RT_OS_WINDOWS)
594 saveAdapterConfigIPv4(ip, mask);
595#else /* !defined(RT_OS_WINDOWS) */
596 if (FAILED(mVirtualBox->SetExtraData(BstrFmt("HostOnly/%s/IPAddress",
597 mInterfaceName.c_str()).raw(),
598 Bstr(aIPAddress).raw())))
599 return E_FAIL;
600 if (FAILED(mVirtualBox->SetExtraData(BstrFmt("HostOnly/%s/IPNetMask",
601 mInterfaceName.c_str()).raw(),
602 Bstr(aNetworkMask).raw())))
603 return E_FAIL;
604#endif /* !defined(RT_OS_WINDOWS) */
605 return S_OK;
606 }
607 else
608 {
609 LogRel(("Failed to EnableStaticIpConfig with rc=%Rrc\n", rc));
610 /* Global::vboxStatusCodeToCOM assert things we can guarantee */
611 switch (rc)
612 {
613 case VERR_NOT_IMPLEMENTED:
614 hrc = E_NOTIMPL;
615 break;
616 case VERR_ACCESS_DENIED:
617 hrc = E_ACCESSDENIED;
618 break;
619 default:
620 hrc = E_FAIL;
621 break;
622 }
623 return hrc;
624 }
625
626 }
627 }
628 return E_FAIL;
629#endif
630}
631
632HRESULT HostNetworkInterface::enableStaticIPConfigV6(const com::Utf8Str &aIPV6Address,
633 ULONG aIPV6NetworkMaskPrefixLength)
634{
635#ifndef VBOX_WITH_HOSTNETIF_API
636 RT_NOREF(aIPV6Address, aIPV6NetworkMaskPrefixLength);
637 return E_NOTIMPL;
638#else
639 if (aIPV6NetworkMaskPrefixLength > 128)
640 return mVirtualBox->setErrorBoth(E_INVALIDARG, VERR_INVALID_PARAMETER,
641 tr("Invalid IPv6 prefix length"));
642
643 HRESULT hrc;
644 int rc;
645
646 RTNETADDRIPV6 AddrOld, AddrNew;
647 char *pszZoneIgnored;
648 bool fAddrChanged;
649
650 rc = RTNetStrToIPv6Addr(aIPV6Address.c_str(), &AddrNew, &pszZoneIgnored);
651 if (RT_FAILURE(rc))
652 {
653 return mVirtualBox->setErrorBoth(E_INVALIDARG, rc, tr("Invalid IPv6 address"));
654 }
655
656 rc = RTNetStrToIPv6Addr(com::Utf8Str(m.realIPV6Address).c_str(), &AddrOld, &pszZoneIgnored);
657 if (RT_SUCCESS(rc))
658 {
659 fAddrChanged = (AddrNew.s.Lo != AddrOld.s.Lo || AddrNew.s.Hi != AddrOld.s.Hi);
660 }
661 else
662 {
663 fAddrChanged = true;
664 }
665
666 if ( fAddrChanged
667 || m.realIPV6PrefixLength != aIPV6NetworkMaskPrefixLength)
668 {
669 if (aIPV6NetworkMaskPrefixLength == 0)
670 aIPV6NetworkMaskPrefixLength = 64;
671 rc = NetIfEnableStaticIpConfigV6(mVirtualBox, this, m.IPV6Address.c_str(),
672 aIPV6Address.c_str(),
673 aIPV6NetworkMaskPrefixLength);
674 if (RT_FAILURE(rc))
675 {
676 LogRel(("Failed to EnableStaticIpConfigV6 with rc=%Rrc\n", rc));
677 /* Global::vboxStatusCodeToCOM assert things we can guarantee */
678 switch (rc)
679 {
680 case VERR_NOT_IMPLEMENTED:
681 hrc = E_NOTIMPL;
682 break;
683 case VERR_ACCESS_DENIED:
684 hrc = E_ACCESSDENIED;
685 break;
686 default:
687 hrc = E_FAIL;
688 break;
689 }
690 return hrc;
691 }
692 else
693 {
694 m.realIPV6Address = aIPV6Address;
695 m.realIPV6PrefixLength = aIPV6NetworkMaskPrefixLength;
696#if defined(RT_OS_WINDOWS)
697 saveAdapterConfigIPv6(Bstr(aIPV6Address).raw(), aIPV6NetworkMaskPrefixLength);
698#else /* !defined(RT_OS_WINDOWS) */
699 if (FAILED(mVirtualBox->SetExtraData(BstrFmt("HostOnly/%s/IPV6Address",
700 mInterfaceName.c_str()).raw(),
701 Bstr(aIPV6Address).raw())))
702 return E_FAIL;
703 if (FAILED(mVirtualBox->SetExtraData(BstrFmt("HostOnly/%s/IPV6NetMask",
704 mInterfaceName.c_str()).raw(),
705 BstrFmt("%u", aIPV6NetworkMaskPrefixLength).raw())))
706#endif /* !defined(RT_OS_WINDOWS) */
707 return E_FAIL;
708 }
709
710 }
711 return S_OK;
712#endif
713}
714
715HRESULT HostNetworkInterface::enableDynamicIPConfig()
716{
717#ifndef VBOX_WITH_HOSTNETIF_API
718 return E_NOTIMPL;
719#else
720 int rc = NetIfEnableDynamicIpConfig(mVirtualBox, this);
721 if (RT_FAILURE(rc))
722 {
723 LogRel(("Failed to EnableDynamicIpConfig with rc=%Rrc\n", rc));
724 return rc == VERR_NOT_IMPLEMENTED ? E_NOTIMPL : E_FAIL;
725 }
726 return S_OK;
727#endif
728}
729
730HRESULT HostNetworkInterface::dHCPRediscover()
731{
732#ifndef VBOX_WITH_HOSTNETIF_API
733 return E_NOTIMPL;
734#else
735 int rc = NetIfDhcpRediscover(mVirtualBox, this);
736 if (RT_FAILURE(rc))
737 {
738 LogRel(("Failed to DhcpRediscover with rc=%Rrc\n", rc));
739 return rc == VERR_NOT_IMPLEMENTED ? E_NOTIMPL : E_FAIL;
740 }
741 return S_OK;
742#endif
743}
744
745HRESULT HostNetworkInterface::i_setVirtualBox(VirtualBox *pVirtualBox)
746{
747 AutoCaller autoCaller(this);
748 if (FAILED(autoCaller.rc())) return autoCaller.rc();
749
750 AssertReturn(mVirtualBox != pVirtualBox, S_OK);
751
752 unconst(mVirtualBox) = pVirtualBox;
753
754#if !defined(RT_OS_WINDOWS)
755 /* If IPv4 address hasn't been initialized */
756 if (m.IPAddress == 0 && mIfType == HostNetworkInterfaceType_HostOnly)
757 {
758 Bstr tmpAddr, tmpMask;
759 HRESULT hrc = mVirtualBox->GetExtraData(BstrFmt("HostOnly/%s/IPAddress",
760 mInterfaceName.c_str()).raw(),
761 tmpAddr.asOutParam());
762 if (FAILED(hrc) || tmpAddr.isEmpty())
763 tmpAddr = getDefaultIPv4Address(mInterfaceName);
764
765 hrc = mVirtualBox->GetExtraData(BstrFmt("HostOnly/%s/IPNetMask",
766 mInterfaceName.c_str()).raw(),
767 tmpMask.asOutParam());
768 if (FAILED(hrc) || tmpMask.isEmpty())
769 tmpMask = Bstr(VBOXNET_IPV4MASK_DEFAULT);
770
771 m.IPAddress = inet_addr(Utf8Str(tmpAddr).c_str());
772 m.networkMask = inet_addr(Utf8Str(tmpMask).c_str());
773 }
774
775 if (m.IPV6Address.isEmpty())
776 {
777 Bstr bstrIPV4Addr;
778 Bstr tmpPrefixLen;
779 HRESULT hrc = mVirtualBox->GetExtraData(BstrFmt("HostOnly/%s/IPV6Address",
780 mInterfaceName.c_str()).raw(),
781 bstrIPV4Addr.asOutParam());
782 if (SUCCEEDED(hrc))
783 {
784 m.IPV6Address = bstrIPV4Addr;
785 if (!m.IPV6Address.isEmpty())
786 {
787 hrc = mVirtualBox->GetExtraData(BstrFmt("HostOnly/%s/IPV6PrefixLen",
788 mInterfaceName.c_str()).raw(),
789 tmpPrefixLen.asOutParam());
790 if (SUCCEEDED(hrc) && !tmpPrefixLen.isEmpty())
791 m.IPV6NetworkMaskPrefixLength = Utf8Str(tmpPrefixLen).toUInt32();
792 else
793 m.IPV6NetworkMaskPrefixLength = 64;
794 }
795 }
796 }
797#endif
798
799 return S_OK;
800}
801
802/* vi: set tabstop=4 shiftwidth=4 expandtab: */
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