VirtualBox

source: vbox/trunk/src/VBox/NetworkServices/DHCP/Config.cpp@ 61600

Last change on this file since 61600 was 58132, checked in by vboxsync, 9 years ago

*: Doxygen fixes.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 37.5 KB
Line 
1/* $Id: Config.cpp 58132 2015-10-09 00:09:37Z vboxsync $ */
2/** @file
3 * Configuration for DHCP.
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
19/**
20 * XXX: license.
21 */
22
23#include <iprt/asm.h>
24#include <iprt/getopt.h>
25#include <iprt/net.h>
26#include <iprt/time.h>
27
28#include <VBox/sup.h>
29#include <VBox/intnet.h>
30#include <VBox/intnetinline.h>
31#include <VBox/vmm/vmm.h>
32#include <VBox/version.h>
33
34#include <VBox/com/array.h>
35#include <VBox/com/string.h>
36
37#include <iprt/cpp/xml.h>
38
39#define BASE_SERVICES_ONLY
40#include "../NetLib/VBoxNetBaseService.h"
41#include "../NetLib/VBoxNetLib.h"
42#include "../NetLib/shared_ptr.h"
43
44#include <list>
45#include <vector>
46#include <map>
47#include <string>
48
49#include "Config.h"
50#include "ClientDataInt.h"
51
52bool operator== (const Lease& lhs, const Lease& rhs)
53{
54 return (lhs.m.get() == rhs.m.get());
55}
56
57
58bool operator!= (const Lease& lhs, const Lease& rhs)
59{
60 return !(lhs == rhs);
61}
62
63
64bool operator< (const Lease& lhs, const Lease& rhs)
65{
66 return ( (lhs.getAddress() < rhs.getAddress())
67 || (lhs.issued() < rhs.issued()));
68}
69/* consts */
70
71const NullConfigEntity *g_NullConfig = new NullConfigEntity();
72RootConfigEntity *g_RootConfig = new RootConfigEntity(std::string("ROOT"), 1200 /* 20 min. */);
73const ClientMatchCriteria *g_AnyClient = new AnyClientMatchCriteria();
74
75static ConfigurationManager *g_ConfigurationManager = ConfigurationManager::getConfigurationManager();
76
77NetworkManager *NetworkManager::g_NetworkManager;
78
79bool MACClientMatchCriteria::check(const Client& client) const
80{
81 return (client == m_mac);
82}
83
84
85int BaseConfigEntity::match(Client& client, BaseConfigEntity **cfg)
86{
87 int iMatch = (m_criteria && m_criteria->check(client)? m_MatchLevel: 0);
88 if (m_children.empty())
89 {
90 if (iMatch > 0)
91 {
92 *cfg = this;
93 return iMatch;
94 }
95 }
96 else
97 {
98 *cfg = this;
99 /* XXX: hack */
100 BaseConfigEntity *matching = this;
101 int matchingLevel = m_MatchLevel;
102
103 for (std::vector<BaseConfigEntity *>::iterator it = m_children.begin();
104 it != m_children.end();
105 ++it)
106 {
107 iMatch = (*it)->match(client, &matching);
108 if (iMatch > matchingLevel)
109 {
110 *cfg = matching;
111 matchingLevel = iMatch;
112 }
113 }
114 return matchingLevel;
115 }
116 return iMatch;
117}
118
119/* Client */
120/* Configs
121 NetworkConfigEntity(std::string name,
122 ConfigEntity* pCfg,
123 ClientMatchCriteria* criteria,
124 RTNETADDRIPV4& networkID,
125 RTNETADDRIPV4& networkMask)
126*/
127static const RTNETADDRIPV4 g_AnyIpv4 = {0};
128static const RTNETADDRIPV4 g_AllIpv4 = {0xffffffff};
129RootConfigEntity::RootConfigEntity(std::string name, uint32_t expPeriod):
130 NetworkConfigEntity(name, g_NullConfig, g_AnyClient, g_AnyIpv4, g_AllIpv4)
131{
132 m_MatchLevel = 2;
133 m_u32ExpirationPeriod = expPeriod;
134}
135
136/* Configuration Manager */
137struct ConfigurationManager::Data
138{
139 Data():fFileExists(false){}
140
141 MapLease2Ip4Address m_allocations;
142 Ipv4AddressContainer m_nameservers;
143 Ipv4AddressContainer m_routers;
144
145 std::string m_domainName;
146 VecClient m_clients;
147 com::Utf8Str m_leaseStorageFilename;
148 bool fFileExists;
149};
150
151ConfigurationManager *ConfigurationManager::getConfigurationManager()
152{
153 if (!g_ConfigurationManager)
154
155
156 {
157 g_ConfigurationManager = new ConfigurationManager();
158 g_ConfigurationManager->init();
159 }
160
161 return g_ConfigurationManager;
162}
163
164
165const std::string tagXMLLeases = "Leases";
166const std::string tagXMLLeasesAttributeVersion = "version";
167const std::string tagXMLLeasesVersion_1_0 = "1.0";
168const std::string tagXMLLease = "Lease";
169const std::string tagXMLLeaseAttributeMac = "mac";
170const std::string tagXMLLeaseAttributeNetwork = "network";
171const std::string tagXMLLeaseAddress = "Address";
172const std::string tagXMLAddressAttributeValue = "value";
173const std::string tagXMLLeaseTime = "Time";
174const std::string tagXMLTimeAttributeIssued = "issued";
175const std::string tagXMLTimeAttributeExpiration = "expiration";
176const std::string tagXMLLeaseOptions = "Options";
177
178/**
179 * @verbatim
180 <Leases version="1.0">
181 <Lease mac="" network=""/>
182 <Address value=""/>
183 <Time issued="" expiration=""/>
184 <options>
185 <option name="" type=""/>
186 </option>
187 </options>
188 </Lease>
189 </Leases>
190 @endverbatim
191 */
192int ConfigurationManager::loadFromFile(const com::Utf8Str& leaseStorageFileName)
193{
194 m->m_leaseStorageFilename = leaseStorageFileName;
195
196 xml::XmlFileParser parser;
197 xml::Document doc;
198
199 try {
200 parser.read(m->m_leaseStorageFilename.c_str(), doc);
201 }
202 catch (...)
203 {
204 return VINF_SUCCESS;
205 }
206
207 /* XML parsing */
208 xml::ElementNode *root = doc.getRootElement();
209
210 if (!root || !root->nameEquals(tagXMLLeases.c_str()))
211 {
212 m->fFileExists = false;
213 return VERR_NOT_FOUND;
214 }
215
216 com::Utf8Str version;
217 if (root)
218 root->getAttributeValue(tagXMLLeasesAttributeVersion.c_str(), version);
219
220 /* XXX: version check */
221 xml::NodesLoop leases(*root);
222
223 const xml::ElementNode *lease;
224 while ((lease = leases.forAllNodes()))
225 {
226 if (!lease->nameEquals(tagXMLLease.c_str()))
227 continue;
228
229 ClientData *data = new ClientData();
230 Lease l(data);
231 if (l.fromXML(lease))
232 {
233
234 m->m_allocations.insert(MapLease2Ip4AddressPair(l, l.getAddress()));
235
236
237 NetworkConfigEntity *pNetCfg = NULL;
238 Client c(data);
239 int rc = g_RootConfig->match(c, (BaseConfigEntity **)&pNetCfg);
240 Assert(rc >= 0 && pNetCfg);
241
242 l.setConfig(pNetCfg);
243
244 m->m_clients.push_back(c);
245 }
246 }
247
248 return VINF_SUCCESS;
249}
250
251
252int ConfigurationManager::saveToFile()
253{
254 if (m->m_leaseStorageFilename.isEmpty())
255 return VINF_SUCCESS;
256
257 xml::Document doc;
258
259 xml::ElementNode *root = doc.createRootElement(tagXMLLeases.c_str());
260 if (!root)
261 return VERR_INTERNAL_ERROR;
262
263 root->setAttribute(tagXMLLeasesAttributeVersion.c_str(), tagXMLLeasesVersion_1_0.c_str());
264
265 for(MapLease2Ip4AddressConstIterator it = m->m_allocations.begin();
266 it != m->m_allocations.end(); ++it)
267 {
268 xml::ElementNode *lease = root->createChild(tagXMLLease.c_str());
269 if (!it->first.toXML(lease))
270 {
271 /* XXX: todo logging + error handling */
272 }
273 }
274
275 try {
276 xml::XmlFileWriter writer(doc);
277 writer.write(m->m_leaseStorageFilename.c_str(), true);
278 } catch(...){}
279
280 return VINF_SUCCESS;
281}
282
283
284int ConfigurationManager::extractRequestList(PCRTNETBOOTP pDhcpMsg, size_t cbDhcpMsg, RawOption& rawOpt)
285{
286 return ConfigurationManager::findOption(RTNET_DHCP_OPT_PARAM_REQ_LIST, pDhcpMsg, cbDhcpMsg, rawOpt);
287}
288
289
290Client ConfigurationManager::getClientByDhcpPacket(const RTNETBOOTP *pDhcpMsg, size_t cbDhcpMsg)
291{
292
293 VecClientIterator it;
294 bool fDhcpValid = false;
295 uint8_t uMsgType = 0;
296
297 fDhcpValid = RTNetIPv4IsDHCPValid(NULL, pDhcpMsg, cbDhcpMsg, &uMsgType);
298 AssertReturn(fDhcpValid, Client::NullClient);
299
300 LogFlowFunc(("dhcp:mac:%RTmac\n", &pDhcpMsg->bp_chaddr.Mac));
301 /* 1st. client IDs */
302 for ( it = m->m_clients.begin();
303 it != m->m_clients.end();
304 ++it)
305 {
306 if ((*it) == pDhcpMsg->bp_chaddr.Mac)
307 {
308 LogFlowFunc(("client:mac:%RTmac\n", it->getMacAddress()));
309 /* check timestamp that request wasn't expired. */
310 return (*it);
311 }
312 }
313
314 if (it == m->m_clients.end())
315 {
316 /* We hasn't got any session for this client */
317 Client c;
318 c.initWithMac(pDhcpMsg->bp_chaddr.Mac);
319 m->m_clients.push_back(c);
320 return m->m_clients.back();
321 }
322
323 return Client::NullClient;
324}
325
326/**
327 * Finds an option.
328 *
329 * @returns On success, a pointer to the first byte in the option data (no none
330 * then it'll be the byte following the 0 size field) and *pcbOpt set
331 * to the option length.
332 * On failure, NULL is returned and *pcbOpt unchanged.
333 *
334 * @param uOption The option to search for.
335 * @param pDhcpMsg The DHCP message.
336 * that this is adjusted if the option length is larger
337 * than the message buffer.
338 */
339int
340ConfigurationManager::findOption(uint8_t uOption, PCRTNETBOOTP pDhcpMsg, size_t cbDhcpMsg, RawOption& opt)
341{
342 Assert(uOption != RTNET_DHCP_OPT_PAD);
343
344 /*
345 * Validate the DHCP bits and figure the max size of the options in the vendor field.
346 */
347 if (cbDhcpMsg <= RT_UOFFSETOF(RTNETBOOTP, bp_vend.Dhcp.dhcp_opts))
348 return VERR_INVALID_PARAMETER;
349
350 if (pDhcpMsg->bp_vend.Dhcp.dhcp_cookie != RT_H2N_U32_C(RTNET_DHCP_COOKIE))
351 return VERR_INVALID_PARAMETER;
352
353 size_t cbLeft = cbDhcpMsg - RT_UOFFSETOF(RTNETBOOTP, bp_vend.Dhcp.dhcp_opts);
354 if (cbLeft > RTNET_DHCP_OPT_SIZE)
355 cbLeft = RTNET_DHCP_OPT_SIZE;
356
357 /*
358 * Search the vendor field.
359 */
360 bool fExtended = false;
361 uint8_t const *pb = &pDhcpMsg->bp_vend.Dhcp.dhcp_opts[0];
362 while (pb && cbLeft > 0)
363 {
364 uint8_t uCur = *pb;
365 if (uCur == RTNET_DHCP_OPT_PAD)
366 {
367 cbLeft--;
368 pb++;
369 }
370 else if (cbLeft <= 1)
371 break;
372 else
373 {
374 size_t cbCur = pb[1];
375 if (cbCur > cbLeft - 2)
376 cbCur = cbLeft - 2;
377 if (uCur == uOption)
378 {
379 opt.u8OptId = uCur;
380 memcpy(opt.au8RawOpt, pb+2, cbCur);
381 opt.cbRawOpt = cbCur;
382 return VINF_SUCCESS;
383 }
384 pb += cbCur + 2;
385 cbLeft -= cbCur - 2;
386 }
387 }
388
389 /** @todo search extended dhcp option field(s) when present */
390
391 return VERR_NOT_FOUND;
392}
393
394
395/**
396 * We bind lease for client till it continue with it on DHCPREQUEST.
397 */
398Lease ConfigurationManager::allocateLease4Client(const Client& client, PCRTNETBOOTP pDhcpMsg, size_t cbDhcpMsg)
399{
400 {
401 /**
402 * This mean that client has already bound or commited lease.
403 * If we've it happens it means that we received DHCPDISCOVER twice.
404 */
405 const Lease l = client.lease();
406 if (l != Lease::NullLease)
407 {
408 /* Here we should take lease from the m_allocation which was feed with leases
409 * on start
410 */
411 if (l.isExpired())
412 {
413 expireLease4Client(const_cast<Client&>(client));
414 if (!l.isExpired())
415 return l;
416 }
417 else
418 {
419 AssertReturn(l.getAddress().u != 0, Lease::NullLease);
420 return l;
421 }
422 }
423 }
424
425 RTNETADDRIPV4 hintAddress;
426 RawOption opt;
427 NetworkConfigEntity *pNetCfg;
428
429 Client cl(client);
430 AssertReturn(g_RootConfig->match(cl, (BaseConfigEntity **)&pNetCfg) > 0, Lease::NullLease);
431
432 /* DHCPDISCOVER MAY contain request address */
433 hintAddress.u = 0;
434 int rc = findOption(RTNET_DHCP_OPT_REQ_ADDR, pDhcpMsg, cbDhcpMsg, opt);
435 if (RT_SUCCESS(rc))
436 {
437 hintAddress.u = *(uint32_t *)opt.au8RawOpt;
438 if ( RT_H2N_U32(hintAddress.u) < RT_H2N_U32(pNetCfg->lowerIp().u)
439 || RT_H2N_U32(hintAddress.u) > RT_H2N_U32(pNetCfg->upperIp().u))
440 hintAddress.u = 0; /* clear hint */
441 }
442
443 if ( hintAddress.u
444 && !isAddressTaken(hintAddress))
445 {
446 Lease l(cl);
447 l.setConfig(pNetCfg);
448 l.setAddress(hintAddress);
449 m->m_allocations.insert(MapLease2Ip4AddressPair(l, hintAddress));
450 return l;
451 }
452
453 uint32_t u32 = 0;
454 for(u32 = RT_H2N_U32(pNetCfg->lowerIp().u);
455 u32 <= RT_H2N_U32(pNetCfg->upperIp().u);
456 ++u32)
457 {
458 RTNETADDRIPV4 address;
459 address.u = RT_H2N_U32(u32);
460 if (!isAddressTaken(address))
461 {
462 Lease l(cl);
463 l.setConfig(pNetCfg);
464 l.setAddress(address);
465 m->m_allocations.insert(MapLease2Ip4AddressPair(l, address));
466 return l;
467 }
468 }
469
470 return Lease::NullLease;
471}
472
473
474int ConfigurationManager::commitLease4Client(Client& client)
475{
476 Lease l = client.lease();
477 AssertReturn(l != Lease::NullLease, VERR_INTERNAL_ERROR);
478
479 l.bindingPhase(false);
480 const NetworkConfigEntity *pCfg = l.getConfig();
481
482 AssertPtr(pCfg);
483 l.setExpiration(pCfg->expirationPeriod());
484 l.phaseStart(RTTimeMilliTS());
485
486 saveToFile();
487
488 return VINF_SUCCESS;
489}
490
491
492int ConfigurationManager::expireLease4Client(Client& client)
493{
494 Lease l = client.lease();
495 AssertReturn(l != Lease::NullLease, VERR_INTERNAL_ERROR);
496
497 if (l.isInBindingPhase())
498 {
499
500 MapLease2Ip4AddressIterator it = m->m_allocations.find(l);
501 AssertReturn(it != m->m_allocations.end(), VERR_NOT_FOUND);
502
503 /*
504 * XXX: perhaps it better to keep this allocation ????
505 */
506 m->m_allocations.erase(it);
507
508 l.expire();
509 return VINF_SUCCESS;
510 }
511
512 l = Lease(client); /* re-new */
513 return VINF_SUCCESS;
514}
515
516
517bool ConfigurationManager::isAddressTaken(const RTNETADDRIPV4& addr, Lease& lease)
518{
519 MapLease2Ip4AddressIterator it;
520
521 for (it = m->m_allocations.begin();
522 it != m->m_allocations.end();
523 ++it)
524 {
525 if (it->second.u == addr.u)
526 {
527 if (lease != Lease::NullLease)
528 lease = it->first;
529
530 return true;
531 }
532 }
533 lease = Lease::NullLease;
534 return false;
535}
536
537
538bool ConfigurationManager::isAddressTaken(const RTNETADDRIPV4& addr)
539{
540 Lease ignore;
541 return isAddressTaken(addr, ignore);
542}
543
544
545NetworkConfigEntity *ConfigurationManager::addNetwork(NetworkConfigEntity *,
546 const RTNETADDRIPV4& networkId,
547 const RTNETADDRIPV4& netmask,
548 RTNETADDRIPV4& LowerAddress,
549 RTNETADDRIPV4& UpperAddress)
550{
551 static int id;
552 char name[64];
553
554 RTStrPrintf(name, RT_ELEMENTS(name), "network-%d", id);
555 std::string strname(name);
556 id++;
557
558
559 if (!LowerAddress.u)
560 LowerAddress = networkId;
561
562 if (!UpperAddress.u)
563 UpperAddress.u = networkId.u | (~netmask.u);
564
565 return new NetworkConfigEntity(strname,
566 g_RootConfig,
567 g_AnyClient,
568 5,
569 networkId,
570 netmask,
571 LowerAddress,
572 UpperAddress);
573}
574
575HostConfigEntity *ConfigurationManager::addHost(NetworkConfigEntity* pCfg,
576 const RTNETADDRIPV4& address,
577 ClientMatchCriteria *criteria)
578{
579 static int id;
580 char name[64];
581
582 RTStrPrintf(name, RT_ELEMENTS(name), "host-%d", id);
583 std::string strname(name);
584 id++;
585
586 return new HostConfigEntity(address, strname, pCfg, criteria);
587}
588
589int ConfigurationManager::addToAddressList(uint8_t u8OptId, RTNETADDRIPV4& address)
590{
591 switch(u8OptId)
592 {
593 case RTNET_DHCP_OPT_DNS:
594 m->m_nameservers.push_back(address);
595 break;
596 case RTNET_DHCP_OPT_ROUTERS:
597 m->m_routers.push_back(address);
598 break;
599 default:
600 Log(("dhcp-opt: list (%d) unsupported\n", u8OptId));
601 }
602 return VINF_SUCCESS;
603}
604
605
606int ConfigurationManager::flushAddressList(uint8_t u8OptId)
607{
608 switch(u8OptId)
609 {
610 case RTNET_DHCP_OPT_DNS:
611 m->m_nameservers.clear();
612 break;
613 case RTNET_DHCP_OPT_ROUTERS:
614 m->m_routers.clear();
615 break;
616 default:
617 Log(("dhcp-opt: list (%d) unsupported\n", u8OptId));
618 }
619 return VINF_SUCCESS;
620}
621
622
623const Ipv4AddressContainer& ConfigurationManager::getAddressList(uint8_t u8OptId)
624{
625 switch(u8OptId)
626 {
627 case RTNET_DHCP_OPT_DNS:
628 return m->m_nameservers;
629
630 case RTNET_DHCP_OPT_ROUTERS:
631 return m->m_routers;
632
633 }
634 /* XXX: Grrr !!! */
635 return m_empty;
636}
637
638
639int ConfigurationManager::setString(uint8_t u8OptId, const std::string& str)
640{
641 switch (u8OptId)
642 {
643 case RTNET_DHCP_OPT_DOMAIN_NAME:
644 m->m_domainName = str;
645 break;
646 default:
647 break;
648 }
649
650 return VINF_SUCCESS;
651}
652
653
654const std::string& ConfigurationManager::getString(uint8_t u8OptId)
655{
656 switch (u8OptId)
657 {
658 case RTNET_DHCP_OPT_DOMAIN_NAME:
659 if (m->m_domainName.length())
660 return m->m_domainName;
661 else
662 return m_noString;
663 default:
664 break;
665 }
666
667 return m_noString;
668}
669
670
671void ConfigurationManager::init()
672{
673 m = new ConfigurationManager::Data();
674}
675
676
677ConfigurationManager::~ConfigurationManager() { if (m) delete m; }
678
679/**
680 * Network manager
681 */
682struct NetworkManager::Data
683{
684 Data()
685 {
686 RT_ZERO(BootPReplyMsg);
687 cbBooPReplyMsg = 0;
688
689 m_OurAddress.u = 0;
690 m_OurNetmask.u = 0;
691 RT_ZERO(m_OurMac);
692 }
693
694 union {
695 RTNETBOOTP BootPHeader;
696 uint8_t au8Storage[1024];
697 } BootPReplyMsg;
698 int cbBooPReplyMsg;
699
700 RTNETADDRIPV4 m_OurAddress;
701 RTNETADDRIPV4 m_OurNetmask;
702 RTMAC m_OurMac;
703
704 ComPtr<IDHCPServer> m_DhcpServer;
705 const VBoxNetHlpUDPService *m_service;
706};
707
708
709NetworkManager::NetworkManager():m(NULL)
710{
711 m = new NetworkManager::Data();
712}
713
714
715NetworkManager::~NetworkManager()
716{
717 delete m;
718 m = NULL;
719}
720
721
722NetworkManager *NetworkManager::getNetworkManager(ComPtr<IDHCPServer> aDhcpServer)
723{
724 if (!g_NetworkManager)
725 {
726 g_NetworkManager = new NetworkManager();
727 g_NetworkManager->m->m_DhcpServer = aDhcpServer;
728 }
729
730 return g_NetworkManager;
731}
732
733
734const RTNETADDRIPV4& NetworkManager::getOurAddress() const
735{
736 return m->m_OurAddress;
737}
738
739
740const RTNETADDRIPV4& NetworkManager::getOurNetmask() const
741{
742 return m->m_OurNetmask;
743}
744
745
746const RTMAC& NetworkManager::getOurMac() const
747{
748 return m->m_OurMac;
749}
750
751
752void NetworkManager::setOurAddress(const RTNETADDRIPV4& aAddress)
753{
754 m->m_OurAddress = aAddress;
755}
756
757
758void NetworkManager::setOurNetmask(const RTNETADDRIPV4& aNetmask)
759{
760 m->m_OurNetmask = aNetmask;
761}
762
763
764void NetworkManager::setOurMac(const RTMAC& aMac)
765{
766 m->m_OurMac = aMac;
767}
768
769
770void NetworkManager::setService(const VBoxNetHlpUDPService *srv)
771{
772 m->m_service = srv;
773}
774
775/**
776 * Network manager creates DHCPOFFER datagramm
777 */
778int NetworkManager::offer4Client(const Client& client, uint32_t u32Xid,
779 uint8_t *pu8ReqList, int cReqList)
780{
781 Lease l(client); /* XXX: oh, it looks badly, but now we have lease */
782 prepareReplyPacket4Client(client, u32Xid);
783
784 RTNETADDRIPV4 address = l.getAddress();
785 m->BootPReplyMsg.BootPHeader.bp_yiaddr = address;
786
787 /* Ubuntu ???*/
788 m->BootPReplyMsg.BootPHeader.bp_ciaddr = address;
789
790 /* options:
791 * - IP lease time
792 * - message type
793 * - server identifier
794 */
795 RawOption opt;
796 RT_ZERO(opt);
797
798 std::vector<RawOption> extra;
799 opt.u8OptId = RTNET_DHCP_OPT_MSG_TYPE;
800 opt.au8RawOpt[0] = RTNET_DHCP_MT_OFFER;
801 opt.cbRawOpt = 1;
802 extra.push_back(opt);
803
804 opt.u8OptId = RTNET_DHCP_OPT_LEASE_TIME;
805
806 const NetworkConfigEntity *pCfg = l.getConfig();
807 AssertPtr(pCfg);
808
809 *(uint32_t *)opt.au8RawOpt = RT_H2N_U32(pCfg->expirationPeriod());
810 opt.cbRawOpt = sizeof(RTNETADDRIPV4);
811
812 extra.push_back(opt);
813
814 processParameterReqList(client, pu8ReqList, cReqList, extra);
815
816 return doReply(client, extra);
817}
818
819/**
820 * Network manager creates DHCPACK
821 */
822int NetworkManager::ack(const Client& client, uint32_t u32Xid,
823 uint8_t *pu8ReqList, int cReqList)
824{
825 RTNETADDRIPV4 address;
826
827 prepareReplyPacket4Client(client, u32Xid);
828
829 Lease l = client.lease();
830 address = l.getAddress();
831 m->BootPReplyMsg.BootPHeader.bp_ciaddr = address;
832
833
834 /* rfc2131 4.3.1 is about DHCPDISCOVER and this value is equal to ciaddr from
835 * DHCPREQUEST or 0 ...
836 * XXX: Using addressHint is not correct way to initialize [cy]iaddress...
837 */
838 m->BootPReplyMsg.BootPHeader.bp_ciaddr = address;
839 m->BootPReplyMsg.BootPHeader.bp_yiaddr = address;
840
841 Assert(m->BootPReplyMsg.BootPHeader.bp_yiaddr.u);
842
843 /* options:
844 * - IP address lease time (if DHCPREQUEST)
845 * - message type
846 * - server identifier
847 */
848 RawOption opt;
849 RT_ZERO(opt);
850
851 std::vector<RawOption> extra;
852 opt.u8OptId = RTNET_DHCP_OPT_MSG_TYPE;
853 opt.au8RawOpt[0] = RTNET_DHCP_MT_ACK;
854 opt.cbRawOpt = 1;
855 extra.push_back(opt);
856
857 /*
858 * XXX: lease time should be conditional. If on dhcprequest then tim should be provided,
859 * else on dhcpinform it mustn't.
860 */
861 opt.u8OptId = RTNET_DHCP_OPT_LEASE_TIME;
862 *(uint32_t *)opt.au8RawOpt = RT_H2N_U32(l.getExpiration());
863 opt.cbRawOpt = sizeof(RTNETADDRIPV4);
864 extra.push_back(opt);
865
866 processParameterReqList(client, pu8ReqList, cReqList, extra);
867
868 return doReply(client, extra);
869}
870
871/**
872 * Network manager creates DHCPNAK
873 */
874int NetworkManager::nak(const Client& client, uint32_t u32Xid)
875{
876
877 Lease l = client.lease();
878 if (l == Lease::NullLease)
879 return VERR_INTERNAL_ERROR;
880
881 prepareReplyPacket4Client(client, u32Xid);
882
883 /* this field filed in prepareReplyPacket4Session, and
884 * RFC 2131 require to have it zero fo NAK.
885 */
886 m->BootPReplyMsg.BootPHeader.bp_yiaddr.u = 0;
887
888 /* options:
889 * - message type (if DHCPREQUEST)
890 * - server identifier
891 */
892 RawOption opt;
893 std::vector<RawOption> extra;
894
895 opt.u8OptId = RTNET_DHCP_OPT_MSG_TYPE;
896 opt.au8RawOpt[0] = RTNET_DHCP_MT_NAC;
897 opt.cbRawOpt = 1;
898 extra.push_back(opt);
899
900 return doReply(client, extra);
901}
902
903/**
904 *
905 */
906int NetworkManager::prepareReplyPacket4Client(const Client& client, uint32_t u32Xid)
907{
908 RT_ZERO(m->BootPReplyMsg);
909
910 m->BootPReplyMsg.BootPHeader.bp_op = RTNETBOOTP_OP_REPLY;
911 m->BootPReplyMsg.BootPHeader.bp_htype = RTNET_ARP_ETHER;
912 m->BootPReplyMsg.BootPHeader.bp_hlen = sizeof(RTMAC);
913 m->BootPReplyMsg.BootPHeader.bp_hops = 0;
914 m->BootPReplyMsg.BootPHeader.bp_xid = u32Xid;
915 m->BootPReplyMsg.BootPHeader.bp_secs = 0;
916 /* XXX: bp_flags should be processed specially */
917 m->BootPReplyMsg.BootPHeader.bp_flags = 0;
918 m->BootPReplyMsg.BootPHeader.bp_ciaddr.u = 0;
919 m->BootPReplyMsg.BootPHeader.bp_giaddr.u = 0;
920
921 m->BootPReplyMsg.BootPHeader.bp_chaddr.Mac = client.getMacAddress();
922
923 const Lease l = client.lease();
924 m->BootPReplyMsg.BootPHeader.bp_yiaddr = l.getAddress();
925 m->BootPReplyMsg.BootPHeader.bp_siaddr.u = 0;
926
927
928 m->BootPReplyMsg.BootPHeader.bp_vend.Dhcp.dhcp_cookie = RT_H2N_U32_C(RTNET_DHCP_COOKIE);
929
930 memset(&m->BootPReplyMsg.BootPHeader.bp_vend.Dhcp.dhcp_opts[0],
931 '\0',
932 RTNET_DHCP_OPT_SIZE);
933
934 return VINF_SUCCESS;
935}
936
937
938int NetworkManager::doReply(const Client& client, const std::vector<RawOption>& extra)
939{
940 int rc;
941
942 /*
943 Options....
944 */
945 VBoxNetDhcpWriteCursor Cursor(&m->BootPReplyMsg.BootPHeader, RTNET_DHCP_NORMAL_SIZE);
946
947 /* The basics */
948
949 Cursor.optIPv4Addr(RTNET_DHCP_OPT_SERVER_ID, m->m_OurAddress);
950
951 const Lease l = client.lease();
952 const std::map<uint8_t, RawOption>& options = l.options();
953
954 for(std::vector<RawOption>::const_iterator it = extra.begin();
955 it != extra.end(); ++it)
956 {
957 if (!Cursor.begin(it->u8OptId, it->cbRawOpt))
958 break;
959 Cursor.put(it->au8RawOpt, it->cbRawOpt);
960
961 }
962
963 for(std::map<uint8_t, RawOption>::const_iterator it = options.begin();
964 it != options.end(); ++it)
965 {
966 if (!Cursor.begin(it->second.u8OptId, it->second.cbRawOpt))
967 break;
968 Cursor.put(it->second.au8RawOpt, it->second.cbRawOpt);
969
970 }
971
972 Cursor.optEnd();
973
974 /*
975 */
976#if 0
977 /** @todo need to see someone set this flag to check that it's correct. */
978 if (!(pDhcpMsg->bp_flags & RTNET_DHCP_FLAGS_NO_BROADCAST))
979 {
980 rc = VBoxNetUDPUnicast(m_pSession,
981 m_hIf,
982 m_pIfBuf,
983 m_OurAddress,
984 &m_OurMac,
985 RTNETIPV4_PORT_BOOTPS, /* sender */
986 IPv4AddrBrdCast,
987 &BootPReplyMsg.BootPHeader->bp_chaddr.Mac,
988 RTNETIPV4_PORT_BOOTPC, /* receiver */
989 &BootPReplyMsg, cbBooPReplyMsg);
990 }
991 else
992#endif
993 rc = m->m_service->hlpUDPBroadcast(RTNETIPV4_PORT_BOOTPS, /* sender */
994 RTNETIPV4_PORT_BOOTPC,
995 &m->BootPReplyMsg,
996 RTNET_DHCP_NORMAL_SIZE);
997
998 AssertRCReturn(rc,rc);
999
1000 return VINF_SUCCESS;
1001}
1002
1003
1004/*
1005 * XXX: TODO: Share decoding code with DHCPServer::addOption.
1006 */
1007static int parseDhcpOptionText(const char *pszText,
1008 int *pOptCode, char **ppszOptText, int *pOptEncoding)
1009{
1010 uint8_t u8Code;
1011 uint32_t u32Enc;
1012 char *pszNext;
1013 int rc;
1014
1015 rc = RTStrToUInt8Ex(pszText, &pszNext, 10, &u8Code);
1016 if (!RT_SUCCESS(rc))
1017 return VERR_PARSE_ERROR;
1018
1019 switch (*pszNext)
1020 {
1021 case ':': /* support legacy format too */
1022 {
1023 u32Enc = 0;
1024 break;
1025 }
1026
1027 case '=':
1028 {
1029 u32Enc = 1;
1030 break;
1031 }
1032
1033 case '@':
1034 {
1035 rc = RTStrToUInt32Ex(pszNext + 1, &pszNext, 10, &u32Enc);
1036 if (!RT_SUCCESS(rc))
1037 return VERR_PARSE_ERROR;
1038 if (*pszNext != '=')
1039 return VERR_PARSE_ERROR;
1040 break;
1041 }
1042
1043 default:
1044 return VERR_PARSE_ERROR;
1045 }
1046
1047 *pOptCode = u8Code;
1048 *ppszOptText = pszNext + 1;
1049 *pOptEncoding = (int)u32Enc;
1050
1051 return VINF_SUCCESS;
1052}
1053
1054
1055static int fillDhcpOption(RawOption &opt, const std::string &OptText, int OptEncoding)
1056{
1057 int rc;
1058
1059 if (OptEncoding == DhcpOptEncoding_Hex)
1060 {
1061 if (OptText.empty())
1062 return VERR_INVALID_PARAMETER;
1063
1064 size_t cbRawOpt = 0;
1065 char *pszNext = const_cast<char *>(OptText.c_str());
1066 while (*pszNext != '\0')
1067 {
1068 if (cbRawOpt >= RT_ELEMENTS(opt.au8RawOpt))
1069 return VERR_INVALID_PARAMETER;
1070
1071 uint8_t u8Byte;
1072 rc = RTStrToUInt8Ex(pszNext, &pszNext, 16, &u8Byte);
1073 if (!RT_SUCCESS(rc))
1074 return rc;
1075
1076 if (*pszNext == ':')
1077 ++pszNext;
1078 else if (*pszNext != '\0')
1079 return VERR_PARSE_ERROR;
1080
1081 opt.au8RawOpt[cbRawOpt] = u8Byte;
1082 ++cbRawOpt;
1083 }
1084 opt.cbRawOpt = (uint8_t)cbRawOpt;
1085 }
1086 else if (OptEncoding == DhcpOptEncoding_Legacy)
1087 {
1088 /*
1089 * XXX: TODO: encode "known" option opt.u8OptId
1090 */
1091 return VERR_INVALID_PARAMETER;
1092 }
1093
1094 return VINF_SUCCESS;
1095}
1096
1097
1098int NetworkManager::processParameterReqList(const Client& client, const uint8_t *pu8ReqList,
1099 int cReqList, std::vector<RawOption>& extra)
1100{
1101 int rc;
1102
1103 const Lease l = client.lease();
1104
1105 const NetworkConfigEntity *pNetCfg = l.getConfig();
1106
1107 /*
1108 * XXX: Brute-force. Unfortunately, there's no notification event
1109 * for changes. Should at least cache the options for a short
1110 * time, enough to last discover/offer/request/ack cycle.
1111 */
1112 typedef std::map< int, std::pair<std::string, int> > DhcpOptionMap;
1113 DhcpOptionMap OptMap;
1114
1115 if (!m->m_DhcpServer.isNull())
1116 {
1117 com::SafeArray<BSTR> strings;
1118 com::Bstr str;
1119 HRESULT hrc;
1120 int OptCode, OptEncoding;
1121 char *pszOptText;
1122
1123 strings.setNull();
1124 hrc = m->m_DhcpServer->COMGETTER(GlobalOptions)(ComSafeArrayAsOutParam(strings));
1125 AssertComRC(hrc);
1126 for (size_t i = 0; i < strings.size(); ++i)
1127 {
1128 com::Utf8Str encoded(strings[i]);
1129 rc = parseDhcpOptionText(encoded.c_str(),
1130 &OptCode, &pszOptText, &OptEncoding);
1131 if (!RT_SUCCESS(rc))
1132 continue;
1133
1134 OptMap[OptCode] = std::make_pair(pszOptText, OptEncoding);
1135 }
1136
1137 const RTMAC &mac = client.getMacAddress();
1138 char strMac[6*2+1] = "";
1139 RTStrPrintf(strMac, sizeof(strMac), "%02x%02x%02x%02x%02x%02x",
1140 mac.au8[0], mac.au8[1], mac.au8[2],
1141 mac.au8[3], mac.au8[4], mac.au8[5]);
1142
1143 strings.setNull();
1144 hrc = m->m_DhcpServer->GetMacOptions(com::Bstr(strMac).raw(),
1145 ComSafeArrayAsOutParam(strings));
1146 AssertComRC(hrc);
1147 for (size_t i = 0; i < strings.size(); ++i)
1148 {
1149 com::Utf8Str text(strings[i]);
1150 rc = parseDhcpOptionText(text.c_str(),
1151 &OptCode, &pszOptText, &OptEncoding);
1152 if (!RT_SUCCESS(rc))
1153 continue;
1154
1155 OptMap[OptCode] = std::make_pair(pszOptText, OptEncoding);
1156 }
1157 }
1158
1159 /* request parameter list */
1160 RawOption opt;
1161 bool fIgnore;
1162 uint8_t u8Req;
1163 for (int idxParam = 0; idxParam < cReqList; ++idxParam)
1164 {
1165 fIgnore = false;
1166 RT_ZERO(opt);
1167 u8Req = opt.u8OptId = pu8ReqList[idxParam];
1168
1169 switch(u8Req)
1170 {
1171 case RTNET_DHCP_OPT_SUBNET_MASK:
1172 ((PRTNETADDRIPV4)opt.au8RawOpt)->u = pNetCfg->netmask().u;
1173 opt.cbRawOpt = sizeof(RTNETADDRIPV4);
1174
1175 break;
1176
1177 case RTNET_DHCP_OPT_ROUTERS:
1178 case RTNET_DHCP_OPT_DNS:
1179 {
1180 const Ipv4AddressContainer lst =
1181 g_ConfigurationManager->getAddressList(u8Req);
1182 PRTNETADDRIPV4 pAddresses = (PRTNETADDRIPV4)&opt.au8RawOpt[0];
1183
1184 for (Ipv4AddressConstIterator it = lst.begin();
1185 it != lst.end();
1186 ++it)
1187 {
1188 *pAddresses = (*it);
1189 pAddresses++;
1190 opt.cbRawOpt += sizeof(RTNETADDRIPV4);
1191 }
1192
1193 if (lst.empty())
1194 fIgnore = true;
1195 }
1196 break;
1197 case RTNET_DHCP_OPT_DOMAIN_NAME:
1198 {
1199 std::string domainName = g_ConfigurationManager->getString(u8Req);
1200 if (domainName == g_ConfigurationManager->m_noString)
1201 {
1202 fIgnore = true;
1203 break;
1204 }
1205
1206 char *pszDomainName = (char *)&opt.au8RawOpt[0];
1207
1208 strcpy(pszDomainName, domainName.c_str());
1209 opt.cbRawOpt = domainName.length();
1210 }
1211 break;
1212 default:
1213 {
1214 DhcpOptionMap::const_iterator it = OptMap.find((int)u8Req);
1215 if (it == OptMap.end())
1216 {
1217 Log(("opt: %d is ignored\n", u8Req));
1218 fIgnore = true;
1219 }
1220 else
1221 {
1222 std::string OptText((*it).second.first);
1223 int OptEncoding((*it).second.second);
1224
1225 rc = fillDhcpOption(opt, OptText, OptEncoding);
1226 if (!RT_SUCCESS(rc))
1227 {
1228 fIgnore = true;
1229 break;
1230 }
1231 }
1232 }
1233 break;
1234 }
1235
1236 if (!fIgnore)
1237 extra.push_back(opt);
1238
1239 }
1240
1241 return VINF_SUCCESS;
1242}
1243
1244/* Client */
1245Client::Client()
1246{
1247 m = SharedPtr<ClientData>();
1248}
1249
1250
1251void Client::initWithMac(const RTMAC& mac)
1252{
1253 m = SharedPtr<ClientData>(new ClientData());
1254 m->m_mac = mac;
1255}
1256
1257
1258bool Client::operator== (const RTMAC& mac) const
1259{
1260 return (m.get() && m->m_mac == mac);
1261}
1262
1263
1264const RTMAC& Client::getMacAddress() const
1265{
1266 return m->m_mac;
1267}
1268
1269
1270Lease Client::lease()
1271{
1272 if (!m.get()) return Lease::NullLease;
1273
1274 if (m->fHasLease)
1275 return Lease(*this);
1276 else
1277 return Lease::NullLease;
1278}
1279
1280
1281const Lease Client::lease() const
1282{
1283 return const_cast<Client *>(this)->lease();
1284}
1285
1286
1287Client::Client(ClientData *data):m(SharedPtr<ClientData>(data)){}
1288
1289/* Lease */
1290Lease::Lease()
1291{
1292 m = SharedPtr<ClientData>();
1293}
1294
1295
1296Lease::Lease (const Client& c)
1297{
1298 m = SharedPtr<ClientData>(c.m);
1299 if ( !m->fHasLease
1300 || ( isExpired()
1301 && !isInBindingPhase()))
1302 {
1303 m->fHasLease = true;
1304 m->fBinding = true;
1305 phaseStart(RTTimeMilliTS());
1306 }
1307}
1308
1309
1310bool Lease::isExpired() const
1311{
1312 AssertPtrReturn(m.get(), false);
1313
1314 if (!m->fBinding)
1315 return (ASMDivU64ByU32RetU32(RTTimeMilliTS() - m->u64TimestampLeasingStarted, 1000)
1316 > m->u32LeaseExpirationPeriod);
1317 else
1318 return (ASMDivU64ByU32RetU32(RTTimeMilliTS() - m->u64TimestampBindingStarted, 1000)
1319 > m->u32BindExpirationPeriod);
1320}
1321
1322
1323void Lease::expire()
1324{
1325 /* XXX: TODO */
1326}
1327
1328
1329void Lease::phaseStart(uint64_t u64Start)
1330{
1331 if (m->fBinding)
1332 m->u64TimestampBindingStarted = u64Start;
1333 else
1334 m->u64TimestampLeasingStarted = u64Start;
1335}
1336
1337
1338void Lease::bindingPhase(bool fOnOff)
1339{
1340 m->fBinding = fOnOff;
1341}
1342
1343
1344bool Lease::isInBindingPhase() const
1345{
1346 return m->fBinding;
1347}
1348
1349
1350uint64_t Lease::issued() const
1351{
1352 return m->u64TimestampLeasingStarted;
1353}
1354
1355
1356void Lease::setExpiration(uint32_t exp)
1357{
1358 if (m->fBinding)
1359 m->u32BindExpirationPeriod = exp;
1360 else
1361 m->u32LeaseExpirationPeriod = exp;
1362}
1363
1364
1365uint32_t Lease::getExpiration() const
1366{
1367 if (m->fBinding)
1368 return m->u32BindExpirationPeriod;
1369 else
1370 return m->u32LeaseExpirationPeriod;
1371}
1372
1373
1374RTNETADDRIPV4 Lease::getAddress() const
1375{
1376 return m->m_address;
1377}
1378
1379
1380void Lease::setAddress(RTNETADDRIPV4 address)
1381{
1382 m->m_address = address;
1383}
1384
1385
1386const NetworkConfigEntity *Lease::getConfig() const
1387{
1388 return m->pCfg;
1389}
1390
1391
1392void Lease::setConfig(NetworkConfigEntity *pCfg)
1393{
1394 m->pCfg = pCfg;
1395}
1396
1397
1398const MapOptionId2RawOption& Lease::options() const
1399{
1400 return m->options;
1401}
1402
1403
1404Lease::Lease(ClientData *pd):m(SharedPtr<ClientData>(pd)){}
1405
1406
1407bool Lease::toXML(xml::ElementNode *node) const
1408{
1409 xml::AttributeNode *pAttribNode = node->setAttribute(tagXMLLeaseAttributeMac.c_str(),
1410 com::Utf8StrFmt("%RTmac", &m->m_mac));
1411 if (!pAttribNode)
1412 return false;
1413
1414 pAttribNode = node->setAttribute(tagXMLLeaseAttributeNetwork.c_str(),
1415 com::Utf8StrFmt("%RTnaipv4", m->m_network));
1416 if (!pAttribNode)
1417 return false;
1418
1419 xml::ElementNode *pLeaseAddress = node->createChild(tagXMLLeaseAddress.c_str());
1420 if (!pLeaseAddress)
1421 return false;
1422
1423 pAttribNode = pLeaseAddress->setAttribute(tagXMLAddressAttributeValue.c_str(),
1424 com::Utf8StrFmt("%RTnaipv4", m->m_address));
1425 if (!pAttribNode)
1426 return false;
1427
1428 xml::ElementNode *pLeaseTime = node->createChild(tagXMLLeaseTime.c_str());
1429 if (!pLeaseTime)
1430 return false;
1431
1432 pAttribNode = pLeaseTime->setAttribute(tagXMLTimeAttributeIssued.c_str(),
1433 m->u64TimestampLeasingStarted);
1434 if (!pAttribNode)
1435 return false;
1436
1437 pAttribNode = pLeaseTime->setAttribute(tagXMLTimeAttributeExpiration.c_str(),
1438 m->u32LeaseExpirationPeriod);
1439 if (!pAttribNode)
1440 return false;
1441
1442 return true;
1443}
1444
1445
1446bool Lease::fromXML(const xml::ElementNode *node)
1447{
1448 com::Utf8Str mac;
1449 bool valueExists = node->getAttributeValue(tagXMLLeaseAttributeMac.c_str(), mac);
1450 if (!valueExists) return false;
1451 int rc = RTNetStrToMacAddr(mac.c_str(), &m->m_mac);
1452 if (RT_FAILURE(rc)) return false;
1453
1454 com::Utf8Str network;
1455 valueExists = node->getAttributeValue(tagXMLLeaseAttributeNetwork.c_str(), network);
1456 if (!valueExists) return false;
1457 rc = RTNetStrToIPv4Addr(network.c_str(), &m->m_network);
1458 if (RT_FAILURE(rc)) return false;
1459
1460 /* Address */
1461 const xml::ElementNode *address = node->findChildElement(tagXMLLeaseAddress.c_str());
1462 if (!address) return false;
1463 com::Utf8Str addressValue;
1464 valueExists = address->getAttributeValue(tagXMLAddressAttributeValue.c_str(), addressValue);
1465 if (!valueExists) return false;
1466 rc = RTNetStrToIPv4Addr(addressValue.c_str(), &m->m_address);
1467
1468 /* Time */
1469 const xml::ElementNode *time = node->findChildElement(tagXMLLeaseTime.c_str());
1470 if (!time) return false;
1471
1472 valueExists = time->getAttributeValue(tagXMLTimeAttributeIssued.c_str(),
1473 &m->u64TimestampLeasingStarted);
1474 if (!valueExists) return false;
1475 m->fBinding = false;
1476
1477 valueExists = time->getAttributeValue(tagXMLTimeAttributeExpiration.c_str(),
1478 &m->u32LeaseExpirationPeriod);
1479 if (!valueExists) return false;
1480
1481 m->fHasLease = true;
1482 return true;
1483}
1484
1485
1486const Lease Lease::NullLease;
1487
1488const Client Client::NullClient;
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