VirtualBox

source: vbox/trunk/src/VBox/Main/src-server/HostDnsService.cpp@ 54803

Last change on this file since 54803 was 54662, checked in by vboxsync, 10 years ago

Main/HostDnsService: instead of keeping fragile state g/c notifyAll()
and just notify proxy directly from setInfo() when info changed.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 8.4 KB
Line 
1/* $Id: HostDnsService.cpp 54662 2015-03-06 05:29:44Z vboxsync $ */
2/** @file
3 * Base class for Host DNS & Co services.
4 */
5
6/*
7 * Copyright (C) 2013-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 <VBox/com/array.h>
19#include <VBox/com/ptr.h>
20#include <VBox/com/string.h>
21
22#include <iprt/cpp/utils.h>
23
24#include "Logging.h"
25#include "VirtualBoxImpl.h"
26#include <iprt/thread.h>
27#include <iprt/semaphore.h>
28#include <iprt/critsect.h>
29
30#include <algorithm>
31#include <string>
32#include "HostDnsService.h"
33
34
35static HostDnsMonitor *g_monitor;
36
37static void dumpHostDnsInformation(const HostDnsInformation&);
38static void dumpHostDnsStrVector(const std::string&, const std::vector<std::string>&);
39
40/* Lockee */
41Lockee::Lockee()
42{
43 RTCritSectInit(&mLock);
44}
45
46Lockee::~Lockee()
47{
48 RTCritSectDelete(&mLock);
49}
50
51const RTCRITSECT* Lockee::lock() const
52{
53 return &mLock;
54}
55
56/* ALock */
57ALock::ALock(const Lockee *aLockee)
58 : lockee(aLockee)
59{
60 RTCritSectEnter(const_cast<PRTCRITSECT>(lockee->lock()));
61}
62
63ALock::~ALock()
64{
65 RTCritSectLeave(const_cast<PRTCRITSECT>(lockee->lock()));
66}
67
68/* HostDnsInformation */
69
70bool HostDnsInformation::equals(const HostDnsInformation &info) const
71{
72 return (servers == info.servers)
73 && (domain == info.domain)
74 && (searchList == info.searchList);
75}
76
77inline static void detachVectorOfString(const std::vector<std::string>& v,
78 std::vector<com::Utf8Str> &aArray)
79{
80 aArray.resize(v.size());
81 size_t i = 0;
82 for (std::vector<std::string>::const_iterator it = v.begin(); it != v.end(); ++it, ++i)
83 aArray[i] = Utf8Str(it->c_str());
84}
85
86struct HostDnsMonitor::Data
87{
88 Data(bool aThreaded) :
89 fThreaded(aThreaded)
90 {}
91
92 std::vector<PCHostDnsMonitorProxy> proxies;
93 HostDnsInformation info;
94 const bool fThreaded;
95 RTTHREAD hMonitoringThread;
96 RTSEMEVENT hDnsInitEvent;
97};
98
99struct HostDnsMonitorProxy::Data
100{
101 Data(const HostDnsMonitor *aMonitor, const VirtualBox *aParent)
102 : info(NULL)
103 , virtualbox(aParent)
104 , monitor(aMonitor)
105 , fModified(true)
106 {}
107
108 virtual ~Data()
109 {
110 if (info)
111 {
112 delete info;
113 info = NULL;
114 }
115 }
116
117 HostDnsInformation *info;
118 const VirtualBox *virtualbox;
119 const HostDnsMonitor *monitor;
120 bool fModified;
121};
122
123
124HostDnsMonitor::HostDnsMonitor(bool fThreaded)
125 : m(NULL)
126{
127 m = new HostDnsMonitor::Data(fThreaded);
128}
129
130HostDnsMonitor::~HostDnsMonitor()
131{
132 if (m)
133 {
134 delete m;
135 m = NULL;
136 }
137}
138
139const HostDnsMonitor *HostDnsMonitor::getHostDnsMonitor()
140{
141 /* XXX: Moved initialization from HostImpl.cpp */
142 if (!g_monitor)
143 {
144# if defined (RT_OS_DARWIN)
145 g_monitor = new HostDnsServiceDarwin();
146# elif defined(RT_OS_WINDOWS)
147 g_monitor = new HostDnsServiceWin();
148# elif defined(RT_OS_LINUX)
149 g_monitor = new HostDnsServiceLinux();
150# elif defined(RT_OS_SOLARIS)
151 g_monitor = new HostDnsServiceSolaris();
152# elif defined(RT_OS_FREEBSD)
153 g_monitor = new HostDnsServiceFreebsd();
154# elif defined(RT_OS_OS2)
155 g_monitor = new HostDnsServiceOs2();
156# else
157 g_monitor = new HostDnsService();
158# endif
159 g_monitor->init();
160 }
161
162 return g_monitor;
163}
164
165void HostDnsMonitor::addMonitorProxy(PCHostDnsMonitorProxy proxy) const
166{
167 ALock l(this);
168 m->proxies.push_back(proxy);
169 proxy->notify();
170}
171
172void HostDnsMonitor::releaseMonitorProxy(PCHostDnsMonitorProxy proxy) const
173{
174 ALock l(this);
175 std::vector<PCHostDnsMonitorProxy>::iterator it;
176 it = std::find(m->proxies.begin(), m->proxies.end(), proxy);
177
178 if (it == m->proxies.end())
179 return;
180
181 m->proxies.erase(it);
182}
183
184void HostDnsMonitor::shutdown()
185{
186 if (g_monitor)
187 {
188 delete g_monitor;
189 g_monitor = NULL;
190 }
191}
192
193const HostDnsInformation &HostDnsMonitor::getInfo() const
194{
195 return m->info;
196}
197
198void HostDnsMonitor::setInfo(const HostDnsInformation &info)
199{
200 ALock l(this);
201
202 if (info.equals(m->info))
203 return;
204
205 m->info = info;
206
207 std::vector<PCHostDnsMonitorProxy>::const_iterator it;
208 for (it = m->proxies.begin(); it != m->proxies.end(); ++it)
209 (*it)->notify();
210}
211
212HRESULT HostDnsMonitor::init()
213{
214 if (m->fThreaded)
215 {
216 int rc = RTSemEventCreate(&m->hDnsInitEvent);
217 AssertRCReturn(rc, E_FAIL);
218
219 rc = RTThreadCreate(&m->hMonitoringThread,
220 HostDnsMonitor::threadMonitoringRoutine,
221 this, 128 * _1K, RTTHREADTYPE_IO, 0, "dns-monitor");
222 AssertRCReturn(rc, E_FAIL);
223
224 RTSemEventWait(m->hDnsInitEvent, RT_INDEFINITE_WAIT);
225 }
226 return S_OK;
227}
228
229
230void HostDnsMonitor::monitorThreadInitializationDone()
231{
232 RTSemEventSignal(m->hDnsInitEvent);
233}
234
235
236int HostDnsMonitor::threadMonitoringRoutine(RTTHREAD, void *pvUser)
237{
238 HostDnsMonitor *pThis = static_cast<HostDnsMonitor *>(pvUser);
239 return pThis->monitorWorker();
240}
241
242/* HostDnsMonitorProxy */
243HostDnsMonitorProxy::HostDnsMonitorProxy()
244 : m(NULL)
245{
246}
247
248HostDnsMonitorProxy::~HostDnsMonitorProxy()
249{
250 if (m)
251 {
252 if (m->monitor)
253 m->monitor->releaseMonitorProxy(this);
254 delete m;
255 m = NULL;
256 }
257}
258
259void HostDnsMonitorProxy::init(const HostDnsMonitor *mon, const VirtualBox* aParent)
260{
261 m = new HostDnsMonitorProxy::Data(mon, aParent);
262 m->monitor->addMonitorProxy(this);
263 updateInfo();
264}
265
266void HostDnsMonitorProxy::notify() const
267{
268 LogRel(("HostDnsMonitorProxy::notify\n"));
269 m->fModified = true;
270 const_cast<VirtualBox *>(m->virtualbox)->i_onHostNameResolutionConfigurationChange();
271}
272
273HRESULT HostDnsMonitorProxy::GetNameServers(std::vector<com::Utf8Str> &aNameServers)
274{
275 AssertReturn(m && m->info, E_FAIL);
276 ALock l(this);
277
278 if (m->fModified)
279 updateInfo();
280
281 LogRel(("HostDnsMonitorProxy::GetNameServers:\n"));
282 dumpHostDnsStrVector("name server", m->info->servers);
283
284 detachVectorOfString(m->info->servers, aNameServers);
285
286 return S_OK;
287}
288
289HRESULT HostDnsMonitorProxy::GetDomainName(com::Utf8Str *pDomainName)
290{
291 AssertReturn(m && m->info, E_FAIL);
292 ALock l(this);
293
294 if (m->fModified)
295 updateInfo();
296
297 LogRel(("HostDnsMonitorProxy::GetDomainName: %s\n",
298 m->info->domain.empty() ? "no domain set" : m->info->domain.c_str()));
299
300 *pDomainName = m->info->domain.c_str();
301
302 return S_OK;
303}
304
305HRESULT HostDnsMonitorProxy::GetSearchStrings(std::vector<com::Utf8Str> &aSearchStrings)
306{
307 AssertReturn(m && m->info, E_FAIL);
308 ALock l(this);
309
310 if (m->fModified)
311 updateInfo();
312
313 LogRel(("HostDnsMonitorProxy::GetSearchStrings:\n"));
314 dumpHostDnsStrVector("search string", m->info->searchList);
315
316 detachVectorOfString(m->info->searchList, aSearchStrings);
317
318 return S_OK;
319}
320
321bool HostDnsMonitorProxy::operator==(PCHostDnsMonitorProxy& rhs)
322{
323 if (!m || !rhs->m)
324 return false;
325
326 /**
327 * we've assigned to the same instance of VirtualBox.
328 */
329 return m->virtualbox == rhs->m->virtualbox;
330}
331
332void HostDnsMonitorProxy::updateInfo()
333{
334 HostDnsInformation *info = new HostDnsInformation(m->monitor->getInfo());
335 HostDnsInformation *old = m->info;
336
337 LogRel(("HostDnsMonitorProxy: Host's DNS information updated:\n"));
338 dumpHostDnsInformation(*info);
339
340 m->info = info;
341 if (old)
342 {
343 LogRel(("HostDnsMonitorProxy: Old host information:\n"));
344 dumpHostDnsInformation(*old);
345
346 delete old;
347 }
348
349 m->fModified = false;
350}
351
352
353static void dumpHostDnsInformation(const HostDnsInformation& info)
354{
355 dumpHostDnsStrVector("server", info.servers);
356 dumpHostDnsStrVector("search string", info.searchList);
357
358 if (!info.domain.empty())
359 LogRel((" domain: %s\n", info.domain.c_str()));
360 else
361 LogRel((" no domain set\n"));
362}
363
364
365static void dumpHostDnsStrVector(const std::string& prefix, const std::vector<std::string>& v)
366{
367 int i = 1;
368 for (std::vector<std::string>::const_iterator it = v.begin();
369 it != v.end();
370 ++it, ++i)
371 LogRel((" %s %d: %s\n", prefix.c_str(), i, it->c_str()));
372 if (v.empty())
373 LogRel((" no %s entries\n", prefix.c_str()));
374}
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