VirtualBox

source: vbox/trunk/src/VBox/NetworkServices/Dhcpd/Db.cpp@ 71353

Last change on this file since 71353 was 70836, checked in by vboxsync, 7 years ago

NetworkServices/Dhcpd: export fix

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 19.2 KB
Line 
1/* $Id: Db.cpp 70836 2018-01-31 14:55:44Z vboxsync $ */
2/** @file
3 * DHCP server - address database
4 */
5
6/*
7 * Copyright (C) 2017-2018 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 <iprt/err.h>
19#include <iprt/stream.h>
20
21#include <algorithm>
22#include <functional>
23
24#include "Db.h"
25
26
27Db::~Db()
28{
29 /** @todo free bindings */
30}
31
32
33int Db::init(const Config *pConfig)
34{
35 Binding::registerFormat();
36
37 m_pConfig = pConfig;
38
39 m_pool.init(pConfig->getIPv4PoolFirst(),
40 pConfig->getIPv4PoolLast());
41
42 return VINF_SUCCESS;
43}
44
45
46bool Binding::g_fFormatRegistered = false;
47
48
49void Binding::registerFormat()
50{
51 if (g_fFormatRegistered)
52 return;
53
54 int rc = RTStrFormatTypeRegister("binding", rtStrFormat, NULL);
55 AssertRC(rc);
56
57 g_fFormatRegistered = true;
58}
59
60
61DECLCALLBACK(size_t)
62Binding::rtStrFormat(PFNRTSTROUTPUT pfnOutput, void *pvArgOutput,
63 const char *pszType, void const *pvValue,
64 int cchWidth, int cchPrecision, unsigned fFlags,
65 void *pvUser)
66{
67 const Binding *b = static_cast<const Binding *>(pvValue);
68 size_t cb = 0;
69
70 AssertReturn(strcmp(pszType, "binding") == 0, 0);
71 RT_NOREF(pszType);
72
73 RT_NOREF(cchWidth, cchPrecision, fFlags);
74 RT_NOREF(pvUser);
75
76 if (b == NULL)
77 {
78 return RTStrFormat(pfnOutput, pvArgOutput, NULL, 0,
79 "<NULL>");
80 }
81
82 cb += RTStrFormat(pfnOutput, pvArgOutput, NULL, 0,
83 "%RTnaipv4", b->m_addr.u);
84
85 if (b->m_state == Binding::FREE)
86 {
87 cb += RTStrFormat(pfnOutput, pvArgOutput, NULL, 0,
88 " free");
89 }
90 else
91 {
92 cb += RTStrFormat(pfnOutput, pvArgOutput, NULL, 0,
93 " to %R[id], %s, valid from ",
94 &b->m_id, b->stateName());
95
96 TimeStamp tsIssued = b->issued();
97 cb += tsIssued.absStrFormat(pfnOutput, pvArgOutput);
98
99 cb += RTStrFormat(pfnOutput, pvArgOutput, NULL, 0,
100 " for %ds until ",
101 b->leaseTime());
102
103 TimeStamp tsValid = b->issued();
104 tsValid.addSeconds(b->leaseTime());
105 cb += tsValid.absStrFormat(pfnOutput, pvArgOutput);
106 }
107
108 return cb;
109}
110
111const char *Binding::stateName() const
112{
113 switch (m_state) {
114 case FREE:
115 return "free";
116 case RELEASED:
117 return "released";
118 case EXPIRED:
119 return "expired";
120 case OFFERED:
121 return "offered";
122 case ACKED:
123 return "acked";
124 default:
125 return "released";
126 }
127}
128
129
130Binding &Binding::setState(const char *pszStateName)
131{
132 if (strcmp(pszStateName, "free") == 0)
133 m_state = Binding::FREE;
134 else if (strcmp(pszStateName, "released") == 0)
135 m_state = Binding::RELEASED;
136 else if (strcmp(pszStateName, "expired") == 0)
137 m_state = Binding::EXPIRED;
138 else if (strcmp(pszStateName, "offered") == 0)
139 m_state = Binding::OFFERED;
140 else if (strcmp(pszStateName, "acked") == 0)
141 m_state = Binding::ACKED;
142 else
143 m_state = Binding::RELEASED;
144
145 return *this;
146}
147
148
149bool Binding::expire(TimeStamp deadline)
150{
151 if (m_state <= Binding::EXPIRED)
152 return false;
153
154 TimeStamp t = m_issued;
155 t.addSeconds(m_secLease);
156
157 if (t < deadline)
158 {
159 if (m_state == Binding::OFFERED)
160 setState(Binding::FREE);
161 else
162 setState(Binding::EXPIRED);
163 }
164 return true;
165}
166
167
168int Binding::toXML(xml::ElementNode *ndParent) const
169{
170 int rc;
171
172 /*
173 * Lease
174 */
175 xml::ElementNode *ndLease = ndParent->createChild("Lease");
176 if (ndLease == NULL)
177 return VERR_GENERAL_FAILURE;
178
179 /* XXX: arrange for lease to get deleted if anything below fails */
180
181
182 ndLease->setAttribute("mac", RTCStringFmt("%RTmac", &m_id.mac()));
183 if (m_id.id().present())
184 {
185 /* I'd prefer RTSTRPRINTHEXBYTES_F_SEP_COLON but there's no decoder */
186 size_t cbStrId = m_id.id().value().size() * 2 + 1;
187 char *pszId = new char[cbStrId];
188 rc = RTStrPrintHexBytes(pszId, cbStrId,
189 &m_id.id().value().front(), m_id.id().value().size(),
190 0);
191 ndLease->setAttribute("id", pszId);
192 delete[] pszId;
193 }
194
195 /* unused but we need it to keep the old code happy */
196 ndLease->setAttribute("network", "0.0.0.0");
197
198 ndLease->setAttribute("state", stateName());
199
200
201 /*
202 * Lease/Address
203 */
204 xml::ElementNode *ndAddr = ndLease->createChild("Address");
205 ndAddr->setAttribute("value", RTCStringFmt("%RTnaipv4", m_addr.u));
206
207
208 /*
209 * Lease/Time
210 */
211 xml::ElementNode *ndTime = ndLease->createChild("Time");
212 ndTime->setAttribute("issued", m_issued.getAbsSeconds());
213 ndTime->setAttribute("expiration", m_secLease);
214
215 return VINF_SUCCESS;
216}
217
218
219Binding *Binding::fromXML(const xml::ElementNode *ndLease)
220{
221 int rc;
222
223 /* Lease/@network seems to always have bogus value, ignore it. */
224
225 /*
226 * Lease/@mac
227 */
228 RTCString strMac;
229 bool fHasMac = ndLease->getAttributeValue("mac", &strMac);
230 if (!fHasMac)
231 return NULL;
232
233 RTMAC mac;
234 rc = RTNetStrToMacAddr(strMac.c_str(), &mac);
235 if (RT_FAILURE(rc))
236 return NULL;
237
238 OptClientId id;
239 RTCString strId;
240 bool fHasId = ndLease->getAttributeValue("id", &strId);
241 if (fHasId)
242 {
243 /*
244 * Decode from "de:ad:be:ef".
245 * XXX: RTStrConvertHexBytes() doesn't grok colons
246 */
247 size_t cbBytes = strId.length() / 2;
248 uint8_t *pBytes = new uint8_t[cbBytes];
249 rc = RTStrConvertHexBytes(strId.c_str(), pBytes, cbBytes, 0);
250 if (RT_SUCCESS(rc))
251 {
252 std::vector<uint8_t> rawopt(pBytes, pBytes + cbBytes);
253 id = OptClientId(rawopt);
254 }
255 }
256
257 /*
258 * Lease/@state - not present in old leases file. We will try to
259 * infer from lease time below.
260 */
261 RTCString strState;
262 bool fHasState = ndLease->getAttributeValue("state", &strState);
263
264 /*
265 * Lease/Address
266 */
267 const xml::ElementNode *ndAddress = ndLease->findChildElement("Address");
268 if (ndAddress == NULL)
269 return NULL;
270
271 /*
272 * Lease/Address/@value
273 */
274 RTCString strAddress;
275 bool fHasValue = ndAddress->getAttributeValue("value", &strAddress);
276 if (!fHasValue)
277 return NULL;
278
279 RTNETADDRIPV4 addr;
280 rc = RTNetStrToIPv4Addr(strAddress.c_str(), &addr);
281 if (RT_FAILURE(rc))
282 return NULL;
283
284 /*
285 * Lease/Time
286 */
287 const xml::ElementNode *ndTime = ndLease->findChildElement("Time");
288 if (time == NULL)
289 return NULL;
290
291 /*
292 * Lease/Time/@issued
293 */
294 int64_t issued;
295 bool fHasIssued = ndTime->getAttributeValue("issued", &issued);
296 if (!fHasIssued)
297 return NULL;
298
299 /*
300 * Lease/Time/@expiration
301 */
302 uint32_t duration;
303 bool fHasExpiration = ndTime->getAttributeValue("expiration", &duration);
304 if (!fHasExpiration)
305 return NULL;
306
307 std::unique_ptr<Binding> b(new Binding(addr));
308 b->m_id = ClientId(mac, id);
309
310 if (fHasState)
311 {
312 b->m_issued = TimeStamp::absSeconds(issued);
313 b->m_secLease = duration;
314 b->setState(strState.c_str());
315 }
316 else
317 { /* XXX: old code wrote timestamps instead of absolute time. */
318 /* pretend that lease has just ended */
319 TimeStamp fakeIssued = TimeStamp::now();
320 fakeIssued.subSeconds(duration);
321 b->m_issued = fakeIssued;
322 b->m_secLease = duration;
323 b->m_state = Binding::EXPIRED;
324 }
325
326 return b.release();
327}
328
329
330void Db::expire()
331{
332 const TimeStamp now = TimeStamp::now();
333
334 for (bindings_t::iterator it = m_bindings.begin();
335 it != m_bindings.end(); ++it)
336 {
337 Binding *b = *it;
338 b->expire(now);
339 }
340}
341
342
343Binding *Db::bindingById(const ClientId &id) const
344{
345 struct ClientMatch : public Binding::Match {
346 const ClientId &m_id;
347 ClientMatch(const ClientId &id) : m_id(id) {}
348
349 bool operator()(const Binding *b)
350 {
351 return b->m_id == m_id;
352 }
353 };
354
355 bindings_t::const_iterator found =
356 std::find_if(m_bindings.begin(), m_bindings.end(),
357 ClientMatch(id));
358
359 if (found == m_bindings.end())
360 return NULL;
361
362 Binding *b = *found;
363 return b;
364}
365
366
367Binding *Db::bindingByAddr(RTNETADDRIPV4 addr) const
368{
369 struct AddrMatch : public Binding::Match {
370 const RTNETADDRIPV4 m_addr;
371 AddrMatch(RTNETADDRIPV4 addr) : m_addr(addr) {}
372
373 bool operator()(const Binding *b)
374 {
375 return b->m_addr.u == m_addr.u;
376 }
377 };
378
379 bindings_t::const_iterator found =
380 std::find_if(m_bindings.begin(), m_bindings.end(),
381 AddrMatch(addr));
382
383 if (found == m_bindings.end())
384 return NULL;
385
386 Binding *b = *found;
387 return b;
388}
389
390
391
392Binding *Db::createBinding(const ClientId &id)
393{
394 RTNETADDRIPV4 addr = m_pool.allocate();
395 if (addr.u == 0)
396 return NULL;
397
398 Binding *b = new Binding(addr, id);
399 m_bindings.push_front(b);
400 return b;
401}
402
403
404Binding *Db::createBinding(RTNETADDRIPV4 addr, const ClientId &id)
405{
406 bool fAvailable = m_pool.allocate(addr);
407 if (!fAvailable)
408 {
409 /*
410 * XXX: this should not happen. If the address is from the
411 * pool, which we have verified before, then either it's in
412 * the free pool or there's an binding (possibly free) for it.
413 */
414 return NULL;
415 }
416
417 Binding *b = new Binding(addr, id);
418 m_bindings.push_front(b);
419 return b;
420}
421
422
423Binding *Db::allocateAddress(const ClientId &id, RTNETADDRIPV4 addr)
424{
425 Assert(addr.u == 0 || addressBelongs(addr));
426
427 Binding *addrBinding = NULL;
428 Binding *freeBinding = NULL;
429 Binding *reuseBinding = NULL;
430
431 if (addr.u != 0)
432 LogDHCP(("> allocateAddress %RTnaipv4 to client %R[id]\n", addr.u, &id));
433 else
434 LogDHCP(("> allocateAddress to client %R[id]\n", &id));
435
436 /*
437 * Allocate existing address if client has one. Ignore requested
438 * address in that case. While here, look for free addresses and
439 * addresses that can be reused.
440 */
441 const TimeStamp now = TimeStamp::now();
442 for (bindings_t::iterator it = m_bindings.begin();
443 it != m_bindings.end(); ++it)
444 {
445 Binding *b = *it;
446 b->expire(now);
447
448 /*
449 * We've already seen this client, give it its old binding.
450 */
451 if (b->m_id == id)
452 {
453 LogDHCP(("> ... found existing binding %R[binding]\n", b));
454 return b;
455 }
456
457 if (addr.u != 0 && b->m_addr.u == addr.u)
458 {
459 Assert(addrBinding == NULL);
460 addrBinding = b;
461 LogDHCP(("> .... noted existing binding %R[binding]\n", addrBinding));
462 }
463
464 /* if we haven't found a free binding yet, keep looking */
465 if (freeBinding == NULL)
466 {
467 if (b->m_state == Binding::FREE)
468 {
469 freeBinding = b;
470 LogDHCP(("> .... noted free binding %R[binding]\n", freeBinding));
471 continue;
472 }
473
474 /* still no free binding, can this one be reused? */
475 if (b->m_state == Binding::RELEASED)
476 {
477 if ( reuseBinding == NULL
478 /* released binding is better than an expired one */
479 || reuseBinding->m_state == Binding::EXPIRED)
480 {
481 reuseBinding = b;
482 LogDHCP(("> .... noted released binding %R[binding]\n", reuseBinding));
483 }
484 }
485 else if (b->m_state == Binding::EXPIRED)
486 {
487 if ( reuseBinding == NULL
488 /* long expired binding is bettern than a recent one */
489 /* || (reuseBinding->m_state == Binding::EXPIRED && b->olderThan(reuseBinding)) */)
490 {
491 reuseBinding = b;
492 LogDHCP(("> .... noted expired binding %R[binding]\n", reuseBinding));
493 }
494 }
495 }
496 }
497
498 /*
499 * Allocate requested address if we can.
500 */
501 if (addr.u != 0)
502 {
503 if (addrBinding == NULL)
504 {
505 addrBinding = createBinding(addr, id);
506 Assert(addrBinding != NULL);
507 LogDHCP(("> .... creating new binding for this address %R[binding]\n",
508 addrBinding));
509 return addrBinding;
510 }
511
512 if (addrBinding->m_state <= Binding::EXPIRED) /* not in use */
513 {
514 LogDHCP(("> .... reusing %s binding for this address\n",
515 addrBinding->stateName()));
516 addrBinding->giveTo(id);
517 return addrBinding;
518 }
519 else
520 {
521 LogDHCP(("> .... cannot reuse %s binding for this address\n",
522 addrBinding->stateName()));
523 }
524 }
525
526 /*
527 * Allocate new (or reuse).
528 */
529 Binding *idBinding = NULL;
530 if (freeBinding != NULL)
531 {
532 idBinding = freeBinding;
533 LogDHCP(("> .... reusing free binding\n"));
534 }
535 else
536 {
537 idBinding = createBinding();
538 if (idBinding != NULL)
539 {
540 LogDHCP(("> .... creating new binding\n"));
541 }
542 else
543 {
544 idBinding = reuseBinding;
545 LogDHCP(("> .... reusing %s binding %R[binding]\n",
546 reuseBinding->stateName()));
547 }
548 }
549
550 if (idBinding == NULL)
551 {
552 LogDHCP(("> .... failed to allocate binding\n"));
553 return NULL;
554 }
555
556 idBinding->giveTo(id);
557 LogDHCP(("> .... allocated %R[binding]\n", idBinding));
558
559 return idBinding;
560}
561
562
563
564Binding *Db::allocateBinding(const DhcpClientMessage &req)
565{
566 /** @todo XXX: handle fixed address assignments */
567 OptRequestedAddress reqAddr(req);
568 if (reqAddr.present() && !addressBelongs(reqAddr.value()))
569 {
570 if (req.messageType() == RTNET_DHCP_MT_DISCOVER)
571 {
572 LogDHCP(("DISCOVER: ignoring invalid requested address\n"));
573 reqAddr = OptRequestedAddress();
574 }
575 else
576 {
577 LogDHCP(("rejecting invalid requested address\n"));
578 return NULL;
579 }
580 }
581
582 const ClientId &id(req.clientId());
583
584 Binding *b = allocateAddress(id, reqAddr.value());
585 if (b == NULL)
586 return NULL;
587
588 Assert(b->id() == id);
589 Assert(b->state() == Binding::FREE || b->state() >= Binding::OFFERED);
590
591 /*
592 * XXX: handle requests for specific lease time!
593 * XXX: old lease might not have expired yet?
594 */
595 // OptLeaseTime reqLeaseTime(req);
596 b->setLeaseTime(1200);
597 return b;
598}
599
600
601int Db::addBinding(Binding *newb)
602{
603 if (!addressBelongs(newb->m_addr))
604 {
605 LogDHCP(("Binding for out of range address %RTnaipv4 ignored\n",
606 newb->m_addr.u));
607 return VERR_INVALID_PARAMETER;
608 }
609
610 for (bindings_t::iterator it = m_bindings.begin();
611 it != m_bindings.end(); ++it)
612 {
613 Binding *b = *it;
614
615 if (newb->m_addr.u == b->m_addr.u)
616 {
617 LogDHCP(("> ADD: %R[binding]\n", newb));
618 LogDHCP(("> .... duplicate ip: %R[binding]\n", b));
619 return VERR_INVALID_PARAMETER;
620 }
621
622 if (newb->m_id == b->m_id)
623 {
624 LogDHCP(("> ADD: %R[binding]\n", newb));
625 LogDHCP(("> .... duplicate id: %R[binding]\n", b));
626 return VERR_INVALID_PARAMETER;
627 }
628 }
629
630 bool ok = m_pool.allocate(newb->m_addr);
631 if (!ok)
632 {
633 LogDHCP(("> ADD: failed to claim IP %R[binding]\n", newb));
634 return VERR_INVALID_PARAMETER;
635 }
636
637 m_bindings.push_back(newb);
638 return VINF_SUCCESS;
639}
640
641
642void Db::cancelOffer(const DhcpClientMessage &req)
643{
644 const OptRequestedAddress reqAddr(req);
645 if (!reqAddr.present())
646 return;
647
648 const RTNETADDRIPV4 addr = reqAddr.value();
649 const ClientId &id(req.clientId());
650
651 for (bindings_t::iterator it = m_bindings.begin();
652 it != m_bindings.end(); ++it)
653 {
654 Binding *b = *it;
655
656 if (b->addr().u == addr.u && b->id() == id)
657 {
658 if (b->state() == Binding::OFFERED)
659 {
660 b->setLeaseTime(0);
661 b->setState(Binding::RELEASED);
662 }
663 return;
664 }
665 }
666}
667
668
669bool Db::releaseBinding(const DhcpClientMessage &req)
670{
671 const RTNETADDRIPV4 addr = req.ciaddr();
672 const ClientId &id(req.clientId());
673
674 for (bindings_t::iterator it = m_bindings.begin();
675 it != m_bindings.end(); ++it)
676 {
677 Binding *b = *it;
678
679 if (b->addr().u == addr.u && b->id() == id)
680 {
681 b->setState(Binding::RELEASED);
682 return true;
683 }
684 }
685
686 return false;
687}
688
689
690int Db::writeLeases(const std::string &strFileName) const
691{
692 LogDHCP(("writing leases to %s\n", strFileName.c_str()));
693
694 xml::Document doc;
695
696 xml::ElementNode *root = doc.createRootElement("Leases");
697 if (root == NULL)
698 return VERR_INTERNAL_ERROR;
699
700 root->setAttribute("version", "1.0");
701
702 for (bindings_t::const_iterator it = m_bindings.begin();
703 it != m_bindings.end(); ++it)
704 {
705 const Binding *b = *it;
706 b->toXML(root);
707 }
708
709 try {
710 xml::XmlFileWriter writer(doc);
711 writer.write(strFileName.c_str(), true);
712 }
713 catch (const xml::EIPRTFailure &e)
714 {
715 LogDHCP(("%s\n", e.what()));
716 return e.rc();
717 }
718 catch (const RTCError &e)
719 {
720 LogDHCP(("%s\n", e.what()));
721 return VERR_GENERAL_FAILURE;
722 }
723 catch (...)
724 {
725 LogDHCP(("Unknown exception while writing '%s'\n",
726 strFileName.c_str()));
727 return VERR_GENERAL_FAILURE;
728 }
729
730 return VINF_SUCCESS;
731}
732
733
734int Db::loadLeases(const std::string &strFileName)
735{
736 LogDHCP(("loading leases from %s\n", strFileName.c_str()));
737
738 xml::Document doc;
739 try
740 {
741 xml::XmlFileParser parser;
742 parser.read(strFileName.c_str(), doc);
743 }
744 catch (const xml::EIPRTFailure &e)
745 {
746 LogDHCP(("%s\n", e.what()));
747 return e.rc();
748 }
749 catch (const RTCError &e)
750 {
751 LogDHCP(("%s\n", e.what()));
752 return VERR_GENERAL_FAILURE;
753 }
754 catch (...)
755 {
756 LogDHCP(("Unknown exception while reading and parsing '%s'\n",
757 strFileName.c_str()));
758 return VERR_GENERAL_FAILURE;
759 }
760
761 xml::ElementNode *ndRoot = doc.getRootElement();
762 if (ndRoot == NULL || !ndRoot->nameEquals("Leases"))
763 {
764 return VERR_NOT_FOUND;
765 }
766
767 xml::NodesLoop it(*ndRoot);
768 const xml::ElementNode *node;
769 while ((node = it.forAllNodes()) != NULL)
770 {
771 if (!node->nameEquals("Lease"))
772 continue;
773
774 loadLease(node);
775 }
776
777 return VINF_SUCCESS;
778}
779
780
781void Db::loadLease(const xml::ElementNode *ndLease)
782{
783 Binding *b = Binding::fromXML(ndLease);
784 bool expired = b->expire();
785
786 if (!expired)
787 LogDHCP(("> LOAD: lease %R[binding]\n", b));
788 else
789 LogDHCP(("> LOAD: EXPIRED lease %R[binding]\n", b));
790
791 addBinding(b);
792}
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