VirtualBox

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

Last change on this file since 50914 was 49875, checked in by vboxsync, 11 years ago

Main: gcc warnings.

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