VirtualBox

source: vbox/trunk/src/VBox/Main/src-server/NATEngineImpl.cpp@ 59154

Last change on this file since 59154 was 57751, checked in by vboxsync, 9 years ago

Main/NATEngine: Fix fallout from r102600. Whn removeRedirect()
backups NATEngineData it ends up deleting iterator for the old
mNATRules from the new copy, corrupting it. It causes a crash later.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 17.6 KB
Line 
1/* $Id: NATEngineImpl.cpp 57751 2015-09-15 00:41:12Z vboxsync $ */
2/** @file
3 * Implementation of INATEngine in VBoxSVC.
4 */
5
6/*
7 * Copyright (C) 2010-2015 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#include "NATEngineImpl.h"
19#include "AutoCaller.h"
20#include "Logging.h"
21#include "MachineImpl.h"
22#include "GuestOSTypeImpl.h"
23
24#include <iprt/string.h>
25#include <iprt/cpp/utils.h>
26
27#include <VBox/err.h>
28#include <VBox/settings.h>
29#include <VBox/com/array.h>
30
31typedef std::map<Utf8Str, settings::NATRule> NATRuleMap;
32
33struct NATEngineData
34{
35 NATEngineData() : mMtu(0),
36 mSockRcv(0),
37 mSockSnd(0),
38 mTcpRcv(0),
39 mTcpSnd(0),
40 mDNSPassDomain(TRUE),
41 mDNSProxy(FALSE),
42 mDNSUseHostResolver(FALSE),
43 mAliasMode(0)
44 {}
45
46 com::Utf8Str mNetwork;
47 com::Utf8Str mBindIP;
48 uint32_t mMtu;
49 uint32_t mSockRcv;
50 uint32_t mSockSnd;
51 uint32_t mTcpRcv;
52 uint32_t mTcpSnd;
53 /* TFTP service */
54 Utf8Str mTFTPPrefix;
55 Utf8Str mTFTPBootFile;
56 Utf8Str mTFTPNextServer;
57 /* DNS service */
58 BOOL mDNSPassDomain;
59 BOOL mDNSProxy;
60 BOOL mDNSUseHostResolver;
61 /* Alias service */
62 ULONG mAliasMode;
63 /* Port forwarding rules */
64 NATRuleMap mNATRules;
65};
66
67struct NATEngine::Data
68{
69 Backupable<NATEngineData> m;
70};
71
72
73// constructor / destructor
74////////////////////////////////////////////////////////////////////////////////
75
76NATEngine::NATEngine():mData(NULL), mParent(NULL), mAdapter(NULL) {}
77NATEngine::~NATEngine(){}
78
79HRESULT NATEngine::FinalConstruct()
80{
81 return BaseFinalConstruct();
82}
83
84void NATEngine::FinalRelease()
85{
86 uninit();
87 BaseFinalRelease();
88}
89
90
91HRESULT NATEngine::init(Machine *aParent, INetworkAdapter *aAdapter)
92{
93 AutoInitSpan autoInitSpan(this);
94 AssertReturn(autoInitSpan.isOk(), E_FAIL);
95 autoInitSpan.setSucceeded();
96 mData = new Data();
97 mData->m.allocate();
98 mData->m->mNetwork.setNull();
99 mData->m->mBindIP.setNull();
100 unconst(mParent) = aParent;
101 unconst(mAdapter) = aAdapter;
102 return S_OK;
103}
104
105HRESULT NATEngine::init(Machine *aParent, INetworkAdapter *aAdapter, NATEngine *aThat)
106{
107 AutoInitSpan autoInitSpan(this);
108 AssertReturn(autoInitSpan.isOk(), E_FAIL);
109 Log(("init that:%p this:%p\n", aThat, this));
110
111 AutoCaller thatCaller(aThat);
112 AssertComRCReturnRC(thatCaller.rc());
113
114 AutoReadLock thatLock(aThat COMMA_LOCKVAL_SRC_POS);
115
116 mData = new Data();
117 mData->m.share(aThat->mData->m);
118 unconst(mParent) = aParent;
119 unconst(mAdapter) = aAdapter;
120 unconst(mPeer) = aThat;
121 autoInitSpan.setSucceeded();
122 return S_OK;
123}
124
125HRESULT NATEngine::initCopy(Machine *aParent, INetworkAdapter *aAdapter, NATEngine *aThat)
126{
127 AutoInitSpan autoInitSpan(this);
128 AssertReturn(autoInitSpan.isOk(), E_FAIL);
129
130 Log(("initCopy that:%p this:%p\n", aThat, this));
131
132 AutoCaller thatCaller(aThat);
133 AssertComRCReturnRC(thatCaller.rc());
134
135 AutoReadLock thatLock(aThat COMMA_LOCKVAL_SRC_POS);
136
137 mData = new Data();
138 mData->m.attachCopy(aThat->mData->m);
139 unconst(mAdapter) = aAdapter;
140 unconst(mParent) = aParent;
141 autoInitSpan.setSucceeded();
142 return BaseFinalConstruct();
143}
144
145
146void NATEngine::uninit()
147{
148 AutoUninitSpan autoUninitSpan(this);
149 if (autoUninitSpan.uninitDone())
150 return;
151
152 mData->m.free();
153 delete mData;
154 mData = NULL;
155 unconst(mPeer) = NULL;
156 unconst(mParent) = NULL;
157}
158
159bool NATEngine::i_isModified()
160{
161 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
162 bool fModified = mData->m.isBackedUp();
163 return fModified;
164}
165
166void NATEngine::i_rollback()
167{
168 AutoCaller autoCaller(this);
169 AssertComRCReturnVoid(autoCaller.rc());
170
171 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
172
173 mData->m.rollback();
174}
175
176void NATEngine::i_commit()
177{
178 AutoCaller autoCaller(this);
179 AssertComRCReturnVoid(autoCaller.rc());
180
181 /* sanity too */
182 AutoCaller peerCaller(mPeer);
183 AssertComRCReturnVoid(peerCaller.rc());
184
185 /* lock both for writing since we modify both (mPeer is "master" so locked
186 * first) */
187 AutoMultiWriteLock2 alock(mPeer, this COMMA_LOCKVAL_SRC_POS);
188 if (mData->m.isBackedUp())
189 {
190 mData->m.commit();
191 if (mPeer)
192 mPeer->mData->m.attach(mData->m);
193 }
194}
195
196void NATEngine::i_copyFrom(NATEngine *aThat)
197{
198 AssertReturnVoid(aThat != NULL);
199
200 /* sanity */
201 AutoCaller autoCaller(this);
202 AssertComRCReturnVoid(autoCaller.rc());
203
204 /* sanity too */
205 AutoCaller thatCaller(aThat);
206 AssertComRCReturnVoid(thatCaller.rc());
207
208 /* peer is not modified, lock it for reading (aThat is "master" so locked
209 * first) */
210 AutoReadLock rl(aThat COMMA_LOCKVAL_SRC_POS);
211 AutoWriteLock wl(this COMMA_LOCKVAL_SRC_POS);
212
213 /* this will back up current data */
214 mData->m.assignCopy(aThat->mData->m);
215}
216
217HRESULT NATEngine::getNetworkSettings(ULONG *aMtu, ULONG *aSockSnd, ULONG *aSockRcv, ULONG *aTcpWndSnd, ULONG *aTcpWndRcv)
218{
219 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
220 if (aMtu)
221 *aMtu = mData->m->mMtu;
222 if (aSockSnd)
223 *aSockSnd = mData->m->mSockSnd;
224 if (aSockRcv)
225 *aSockRcv = mData->m->mSockRcv;
226 if (aTcpWndSnd)
227 *aTcpWndSnd = mData->m->mTcpSnd;
228 if (aTcpWndRcv)
229 *aTcpWndRcv = mData->m->mTcpRcv;
230
231 return S_OK;
232}
233
234HRESULT NATEngine::setNetworkSettings(ULONG aMtu, ULONG aSockSnd, ULONG aSockRcv, ULONG aTcpWndSnd, ULONG aTcpWndRcv)
235{
236 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
237 if ( aMtu || aSockSnd || aSockRcv
238 || aTcpWndSnd || aTcpWndRcv)
239 {
240 mData->m.backup();
241 mParent->i_setModified(Machine::IsModified_NetworkAdapters);
242 }
243 if (aMtu)
244 mData->m->mMtu = aMtu;
245 if (aSockSnd)
246 mData->m->mSockSnd = aSockSnd;
247 if (aSockRcv)
248 mData->m->mSockRcv = aSockSnd;
249 if (aTcpWndSnd)
250 mData->m->mTcpSnd = aTcpWndSnd;
251 if (aTcpWndRcv)
252 mData->m->mTcpRcv = aTcpWndRcv;
253
254 return S_OK;
255}
256
257
258HRESULT NATEngine::getRedirects(std::vector<com::Utf8Str> &aRedirects)
259{
260 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
261
262 aRedirects.resize(mData->m->mNATRules.size());
263 size_t i = 0;
264 NATRuleMap::const_iterator it;
265 for (it = mData->m->mNATRules.begin(); it != mData->m->mNATRules.end(); ++it, ++i)
266 {
267 settings::NATRule r = it->second;
268 aRedirects[i] = Utf8StrFmt("%s,%d,%s,%d,%s,%d",
269 r.strName.c_str(),
270 r.proto,
271 r.strHostIP.c_str(),
272 r.u16HostPort,
273 r.strGuestIP.c_str(),
274 r.u16GuestPort);
275 }
276 return S_OK;
277}
278
279HRESULT NATEngine::addRedirect(const com::Utf8Str &aName, NATProtocol_T aProto, const com::Utf8Str &aHostIP,
280 USHORT aHostPort, const com::Utf8Str &aGuestIP, USHORT aGuestPort)
281{
282 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
283 Utf8Str name = aName;
284 settings::NATRule r;
285 const char *proto;
286 switch (aProto)
287 {
288 case NATProtocol_TCP:
289 proto = "tcp";
290 break;
291 case NATProtocol_UDP:
292 proto = "udp";
293 break;
294 default:
295 return E_INVALIDARG;
296 }
297 if (name.isEmpty())
298 name = Utf8StrFmt("%s_%d_%d", proto, aHostPort, aGuestPort);
299
300 NATRuleMap::iterator it;
301 for (it = mData->m->mNATRules.begin(); it != mData->m->mNATRules.end(); ++it)
302 {
303 r = it->second;
304 if (it->first == name)
305 return setError(E_INVALIDARG,
306 tr("A NAT rule of this name already exists"));
307 if ( r.strHostIP == aHostIP
308 && r.u16HostPort == aHostPort
309 && r.proto == aProto)
310 return setError(E_INVALIDARG,
311 tr("A NAT rule for this host port and this host IP already exists"));
312 }
313
314 mData->m.backup();
315 r.strName = name.c_str();
316 r.proto = aProto;
317 r.strHostIP = aHostIP;
318 r.u16HostPort = aHostPort;
319 r.strGuestIP = aGuestIP;
320 r.u16GuestPort = aGuestPort;
321 mData->m->mNATRules.insert(std::make_pair(name, r));
322 mParent->i_setModified(Machine::IsModified_NetworkAdapters);
323
324 ULONG ulSlot;
325 mAdapter->COMGETTER(Slot)(&ulSlot);
326
327 alock.release();
328 mParent->i_onNATRedirectRuleChange(ulSlot, FALSE, Bstr(name).raw(), aProto, Bstr(r.strHostIP).raw(),
329 r.u16HostPort, Bstr(r.strGuestIP).raw(), r.u16GuestPort);
330 return S_OK;
331}
332
333HRESULT NATEngine::removeRedirect(const com::Utf8Str &aName)
334{
335 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
336 NATRuleMap::iterator it = mData->m->mNATRules.find(aName);
337 if (it == mData->m->mNATRules.end())
338 return E_INVALIDARG;
339 mData->m.backup();
340 /*
341 * NB: "it" may now point to the backup! In that case it's ok to
342 * get data from the backup copy of mNATRules via it, but we can't
343 * erase(it) from potentially new mNATRules.
344 */
345 settings::NATRule r = it->second;
346 Utf8Str strHostIP = r.strHostIP;
347 Utf8Str strGuestIP = r.strGuestIP;
348 NATProtocol_T proto = r.proto;
349 uint16_t u16HostPort = r.u16HostPort;
350 uint16_t u16GuestPort = r.u16GuestPort;
351 ULONG ulSlot;
352 mAdapter->COMGETTER(Slot)(&ulSlot);
353
354 mData->m->mNATRules.erase(aName); /* NB: erase by key, "it" may not be valid */
355 mParent->i_setModified(Machine::IsModified_NetworkAdapters);
356 alock.release();
357 mParent->i_onNATRedirectRuleChange(ulSlot, TRUE, Bstr(aName).raw(), proto, Bstr(strHostIP).raw(),
358 u16HostPort, Bstr(strGuestIP).raw(), u16GuestPort);
359 return S_OK;
360}
361
362HRESULT NATEngine::i_loadSettings(const settings::NAT &data)
363{
364 AutoCaller autoCaller(this);
365 AssertComRCReturnRC(autoCaller.rc());
366
367 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
368 HRESULT rc = S_OK;
369 mData->m->mNetwork = data.strNetwork;
370 mData->m->mBindIP = data.strBindIP;
371 mData->m->mMtu = data.u32Mtu;
372 mData->m->mSockSnd = data.u32SockSnd;
373 mData->m->mTcpRcv = data.u32TcpRcv;
374 mData->m->mTcpSnd = data.u32TcpSnd;
375 /* TFTP */
376 mData->m->mTFTPPrefix = data.strTFTPPrefix;
377 mData->m->mTFTPBootFile = data.strTFTPBootFile;
378 mData->m->mTFTPNextServer = data.strTFTPNextServer;
379 /* DNS */
380 mData->m->mDNSPassDomain = data.fDNSPassDomain;
381 mData->m->mDNSProxy = data.fDNSProxy;
382 mData->m->mDNSUseHostResolver = data.fDNSUseHostResolver;
383 /* Alias */
384 mData->m->mAliasMode = (data.fAliasUseSamePorts ? NATAliasMode_AliasUseSamePorts : 0);
385 mData->m->mAliasMode |= (data.fAliasLog ? NATAliasMode_AliasLog : 0);
386 mData->m->mAliasMode |= (data.fAliasProxyOnly ? NATAliasMode_AliasProxyOnly : 0);
387 /* port forwarding */
388 mData->m->mNATRules.clear();
389 for (settings::NATRuleList::const_iterator it = data.llRules.begin();
390 it != data.llRules.end(); ++it)
391 {
392 mData->m->mNATRules.insert(std::make_pair(it->strName, *it));
393 }
394 return rc;
395}
396
397
398HRESULT NATEngine::i_saveSettings(settings::NAT &data)
399{
400 AutoCaller autoCaller(this);
401 AssertComRCReturnRC(autoCaller.rc());
402
403 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
404 HRESULT rc = S_OK;
405 data.strNetwork = mData->m->mNetwork;
406 data.strBindIP = mData->m->mBindIP;
407 data.u32Mtu = mData->m->mMtu;
408 data.u32SockRcv = mData->m->mSockRcv;
409 data.u32SockSnd = mData->m->mSockSnd;
410 data.u32TcpRcv = mData->m->mTcpRcv;
411 data.u32TcpSnd = mData->m->mTcpSnd;
412 /* TFTP */
413 data.strTFTPPrefix = mData->m->mTFTPPrefix;
414 data.strTFTPBootFile = mData->m->mTFTPBootFile;
415 data.strTFTPNextServer = mData->m->mTFTPNextServer;
416 /* DNS */
417 data.fDNSPassDomain = !!mData->m->mDNSPassDomain;
418 data.fDNSProxy = !!mData->m->mDNSProxy;
419 data.fDNSUseHostResolver = !!mData->m->mDNSUseHostResolver;
420 /* Alias */
421 data.fAliasLog = !!(mData->m->mAliasMode & NATAliasMode_AliasLog);
422 data.fAliasProxyOnly = !!(mData->m->mAliasMode & NATAliasMode_AliasProxyOnly);
423 data.fAliasUseSamePorts = !!(mData->m->mAliasMode & NATAliasMode_AliasUseSamePorts);
424
425 for (NATRuleMap::iterator it = mData->m->mNATRules.begin();
426 it != mData->m->mNATRules.end(); ++it)
427 data.llRules.push_back(it->second);
428 return rc;
429}
430
431HRESULT NATEngine::setNetwork(const com::Utf8Str &aNetwork)
432{
433 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
434 if (Bstr(mData->m->mNetwork) != aNetwork)
435 {
436 mData->m.backup();
437 mData->m->mNetwork = aNetwork;
438 mParent->i_setModified(Machine::IsModified_NetworkAdapters);
439 }
440 return S_OK;
441}
442
443
444HRESULT NATEngine::getNetwork(com::Utf8Str &aNetwork)
445{
446 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
447 if (!mData->m->mNetwork.isEmpty())
448 {
449 aNetwork = mData->m->mNetwork;
450 Log(("Getter (this:%p) Network: %s\n", this, mData->m->mNetwork.c_str()));
451 }
452 return S_OK;
453}
454
455HRESULT NATEngine::setHostIP(const com::Utf8Str &aHostIP)
456{
457 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
458 if (Bstr(mData->m->mBindIP) != aHostIP)
459 {
460 mData->m.backup();
461 mData->m->mBindIP = aHostIP;
462 mParent->i_setModified(Machine::IsModified_NetworkAdapters);
463 }
464 return S_OK;
465}
466
467HRESULT NATEngine::getHostIP(com::Utf8Str &aBindIP)
468{
469 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
470
471 if (!mData->m->mBindIP.isEmpty())
472 aBindIP = mData->m->mBindIP;
473 return S_OK;
474}
475
476HRESULT NATEngine::setTFTPPrefix(const com::Utf8Str &aTFTPPrefix)
477{
478 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
479 if (Bstr(mData->m->mTFTPPrefix) != aTFTPPrefix)
480 {
481 mData->m.backup();
482 mData->m->mTFTPPrefix = aTFTPPrefix;
483 mParent->i_setModified(Machine::IsModified_NetworkAdapters);
484 }
485 return S_OK;
486}
487
488
489HRESULT NATEngine::getTFTPPrefix(com::Utf8Str &aTFTPPrefix)
490{
491 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
492
493 if (!mData->m->mTFTPPrefix.isEmpty())
494 {
495 aTFTPPrefix = mData->m->mTFTPPrefix;
496 Log(("Getter (this:%p) TFTPPrefix: %s\n", this, mData->m->mTFTPPrefix.c_str()));
497 }
498 return S_OK;
499}
500
501HRESULT NATEngine::setTFTPBootFile(const com::Utf8Str &aTFTPBootFile)
502{
503 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
504 if (Bstr(mData->m->mTFTPBootFile) != aTFTPBootFile)
505 {
506 mData->m.backup();
507 mData->m->mTFTPBootFile = aTFTPBootFile;
508 mParent->i_setModified(Machine::IsModified_NetworkAdapters);
509 }
510 return S_OK;
511}
512
513
514HRESULT NATEngine::getTFTPBootFile(com::Utf8Str &aTFTPBootFile)
515{
516 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
517 if (!mData->m->mTFTPBootFile.isEmpty())
518 {
519 aTFTPBootFile = mData->m->mTFTPBootFile;
520 Log(("Getter (this:%p) BootFile: %s\n", this, mData->m->mTFTPBootFile.c_str()));
521 }
522 return S_OK;
523}
524
525
526HRESULT NATEngine::setTFTPNextServer(const com::Utf8Str &aTFTPNextServer)
527{
528 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
529 if (Bstr(mData->m->mTFTPNextServer) != aTFTPNextServer)
530 {
531 mData->m.backup();
532 mData->m->mTFTPNextServer = aTFTPNextServer;
533 mParent->i_setModified(Machine::IsModified_NetworkAdapters);
534 }
535 return S_OK;
536}
537
538HRESULT NATEngine::getTFTPNextServer(com::Utf8Str &aTFTPNextServer)
539{
540 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
541 if (!mData->m->mTFTPNextServer.isEmpty())
542 {
543 aTFTPNextServer = mData->m->mTFTPNextServer;
544 Log(("Getter (this:%p) NextServer: %s\n", this, mData->m->mTFTPNextServer.c_str()));
545 }
546 return S_OK;
547}
548
549/* DNS */
550HRESULT NATEngine::setDNSPassDomain(BOOL aDNSPassDomain)
551{
552 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
553
554 if (mData->m->mDNSPassDomain != aDNSPassDomain)
555 {
556 mData->m.backup();
557 mData->m->mDNSPassDomain = aDNSPassDomain;
558 mParent->i_setModified(Machine::IsModified_NetworkAdapters);
559 }
560 return S_OK;
561}
562
563HRESULT NATEngine::getDNSPassDomain(BOOL *aDNSPassDomain)
564{
565 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
566 *aDNSPassDomain = mData->m->mDNSPassDomain;
567 return S_OK;
568}
569
570
571HRESULT NATEngine::setDNSProxy(BOOL aDNSProxy)
572{
573 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
574
575 if (mData->m->mDNSProxy != aDNSProxy)
576 {
577 mData->m.backup();
578 mData->m->mDNSProxy = aDNSProxy;
579 mParent->i_setModified(Machine::IsModified_NetworkAdapters);
580 }
581 return S_OK;
582}
583
584HRESULT NATEngine::getDNSProxy(BOOL *aDNSProxy)
585{
586 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
587 *aDNSProxy = mData->m->mDNSProxy;
588 return S_OK;
589}
590
591
592HRESULT NATEngine::getDNSUseHostResolver(BOOL *aDNSUseHostResolver)
593{
594 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
595 *aDNSUseHostResolver = mData->m->mDNSUseHostResolver;
596 return S_OK;
597}
598
599
600HRESULT NATEngine::setDNSUseHostResolver(BOOL aDNSUseHostResolver)
601{
602 if (mData->m->mDNSUseHostResolver != aDNSUseHostResolver)
603 {
604 mData->m.backup();
605 mData->m->mDNSUseHostResolver = aDNSUseHostResolver;
606 mParent->i_setModified(Machine::IsModified_NetworkAdapters);
607 }
608 return S_OK;
609}
610
611HRESULT NATEngine::setAliasMode(ULONG aAliasMode)
612{
613 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
614
615 if (mData->m->mAliasMode != aAliasMode)
616 {
617 mData->m.backup();
618 mData->m->mAliasMode = aAliasMode;
619 mParent->i_setModified(Machine::IsModified_NetworkAdapters);
620 }
621 return S_OK;
622}
623
624HRESULT NATEngine::getAliasMode(ULONG *aAliasMode)
625{
626 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
627 *aAliasMode = mData->m->mAliasMode;
628 return S_OK;
629}
630
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