diff options
Diffstat (limited to 'src/lib/dhcp/pkt.cc')
-rw-r--r-- | src/lib/dhcp/pkt.cc | 314 |
1 files changed, 314 insertions, 0 deletions
diff --git a/src/lib/dhcp/pkt.cc b/src/lib/dhcp/pkt.cc new file mode 100644 index 0000000..2622460 --- /dev/null +++ b/src/lib/dhcp/pkt.cc @@ -0,0 +1,314 @@ +// Copyright (C) 2014-2023 Internet Systems Consortium, Inc. ("ISC") +// +// This Source Code Form is subject to the terms of the Mozilla Public +// License, v. 2.0. If a copy of the MPL was not distributed with this +// file, You can obtain one at http://mozilla.org/MPL/2.0/. + +#include <config.h> +#include <utility> +#include <dhcp/pkt.h> +#include <dhcp/iface_mgr.h> +#include <dhcp/hwaddr.h> +#include <vector> + +namespace isc { +namespace dhcp { + +Pkt::Pkt(uint32_t transid, const isc::asiolink::IOAddress& local_addr, + const isc::asiolink::IOAddress& remote_addr, uint16_t local_port, + uint16_t remote_port) + : transid_(transid), iface_(""), ifindex_(UNSET_IFINDEX), local_addr_(local_addr), + remote_addr_(remote_addr), local_port_(local_port), + remote_port_(remote_port), buffer_out_(0), copy_retrieved_options_(false) { +} + +Pkt::Pkt(const uint8_t* buf, uint32_t len, const isc::asiolink::IOAddress& local_addr, + const isc::asiolink::IOAddress& remote_addr, uint16_t local_port, + uint16_t remote_port) + : transid_(0), iface_(""), ifindex_(UNSET_IFINDEX), local_addr_(local_addr), + remote_addr_(remote_addr), local_port_(local_port), + remote_port_(remote_port), buffer_out_(0), copy_retrieved_options_(false) { + if (len != 0) { + if (buf == NULL) { + isc_throw(InvalidParameter, "data buffer passed to Pkt is NULL"); + } + data_.resize(len); + memcpy(&data_[0], buf, len); + } +} + +OptionCollection +Pkt::cloneOptions() { + OptionCollection options; + for (auto const& option : options_) { + options.emplace(std::make_pair(option.second->getType(), option.second->clone())); + } + return (options); +} + +void +Pkt::addOption(const OptionPtr& opt) { + options_.insert(std::pair<int, OptionPtr>(opt->getType(), opt)); +} + +OptionPtr +Pkt::getNonCopiedOption(const uint16_t type) const { + const auto& x = options_.find(type); + if (x != options_.end()) { + return (x->second); + } + return (OptionPtr()); +} + +OptionPtr +Pkt::getOption(const uint16_t type) { + const auto& x = options_.find(type); + if (x != options_.end()) { + if (copy_retrieved_options_) { + OptionPtr option_copy = x->second->clone(); + x->second = option_copy; + } + return (x->second); + } + return (OptionPtr()); // NULL +} + +OptionCollection +Pkt::getNonCopiedOptions(const uint16_t opt_type) const { + std::pair<OptionCollection::const_iterator, + OptionCollection::const_iterator> range = options_.equal_range(opt_type); + return (OptionCollection(range.first, range.second)); +} + +OptionCollection +Pkt::getOptions(const uint16_t opt_type) { + OptionCollection options_copy; + + std::pair<OptionCollection::iterator, + OptionCollection::iterator> range = options_.equal_range(opt_type); + // If options should be copied on retrieval, we should now iterate over + // matching options, copy them and replace the original ones with new + // instances. + if (copy_retrieved_options_) { + for (OptionCollection::iterator opt_it = range.first; + opt_it != range.second; ++opt_it) { + OptionPtr option_copy = opt_it->second->clone(); + opt_it->second = option_copy; + } + } + // Finally, return updated options. This can also be empty in some cases. + return (OptionCollection(range.first, range.second)); +} + +bool +Pkt::delOption(uint16_t type) { + const auto& x = options_.find(type); + if (x != options_.end()) { + options_.erase(x); + return (true); // delete successful + } else { + return (false); // can't find option to be deleted + } +} + +bool +Pkt::inClass(const ClientClass& client_class) { + return (classes_.contains(client_class)); +} + +void +Pkt::addClass(const ClientClass& client_class, bool required) { + ClientClasses& classes = !required ? classes_ : required_classes_; + if (!classes.contains(client_class)) { + classes.insert(client_class); + static_cast<void>(subclasses_.push_back(SubClassRelation(client_class, client_class))); + } +} + +void +Pkt::addSubClass(const ClientClass& class_def, const ClientClass& subclass) { + if (!classes_.contains(class_def)) { + classes_.insert(class_def); + static_cast<void>(subclasses_.push_back(SubClassRelation(class_def, subclass))); + } + if (!classes_.contains(subclass)) { + classes_.insert(subclass); + static_cast<void>(subclasses_.push_back(SubClassRelation(subclass, subclass))); + } +} + +void +Pkt::updateTimestamp() { + timestamp_ = boost::posix_time::microsec_clock::universal_time(); +} + +void Pkt::repack() { + if (!data_.empty()) { + buffer_out_.writeData(&data_[0], data_.size()); + } +} + +void +Pkt::setRemoteHWAddr(const uint8_t htype, const uint8_t hlen, + const std::vector<uint8_t>& hw_addr) { + setHWAddrMember(htype, hlen, hw_addr, remote_hwaddr_); +} + +void +Pkt::setRemoteHWAddr(const HWAddrPtr& hw_addr) { + if (!hw_addr) { + isc_throw(BadValue, "Setting remote HW address to NULL is" + << " forbidden."); + } + remote_hwaddr_ = hw_addr; +} + +void +Pkt::setHWAddrMember(const uint8_t htype, const uint8_t, + const std::vector<uint8_t>& hw_addr, + HWAddrPtr& storage) { + storage.reset(new HWAddr(hw_addr, htype)); +} + +HWAddrPtr +Pkt::getMAC(uint32_t hw_addr_src) { + HWAddrPtr mac; + + /// @todo: Implement an array of method pointers instead of set of ifs + + // Method 1: from raw sockets. + if (hw_addr_src & HWAddr::HWADDR_SOURCE_RAW) { + mac = getRemoteHWAddr(); + if (mac) { + mac->source_ = HWAddr::HWADDR_SOURCE_RAW; + return (mac); + } else if (hw_addr_src == HWAddr::HWADDR_SOURCE_RAW) { + // If we're interested only in RAW sockets as source of that info, + // there's no point in trying other options. + return (HWAddrPtr()); + } + } + + // Method 2: From client link-layer address option inserted by a relay + if (hw_addr_src & HWAddr::HWADDR_SOURCE_CLIENT_ADDR_RELAY_OPTION) { + mac = getMACFromIPv6RelayOpt(); + if (mac) { + return (mac); + } else if (hw_addr_src == HWAddr::HWADDR_SOURCE_CLIENT_ADDR_RELAY_OPTION) { + // If we're interested only in RFC6939 link layer address as source + // of that info, there's no point in trying other options. + return (HWAddrPtr()); + } + } + + // Method 3: Extracted from DUID-LLT or DUID-LL + if(hw_addr_src & HWAddr::HWADDR_SOURCE_DUID) { + mac = getMACFromDUID(); + if (mac) { + return (mac); + } else if (hw_addr_src == HWAddr::HWADDR_SOURCE_DUID) { + // If the only source allowed is DUID then we can skip the other + // methods. + return (HWAddrPtr()); + } + } + + // Method 4: Extracted from source IPv6 link-local address + if (hw_addr_src & HWAddr::HWADDR_SOURCE_IPV6_LINK_LOCAL) { + mac = getMACFromSrcLinkLocalAddr(); + if (mac) { + return (mac); + } else if (hw_addr_src == HWAddr::HWADDR_SOURCE_IPV6_LINK_LOCAL) { + // If we're interested only in link-local addr as source of that + // info, there's no point in trying other options. + return (HWAddrPtr()); + } + } + + // Method 5: From remote-id option inserted by a relay + if(hw_addr_src & HWAddr::HWADDR_SOURCE_REMOTE_ID) { + mac = getMACFromRemoteIdRelayOption(); + if (mac) { + return (mac); + } else if (hw_addr_src == HWAddr::HWADDR_SOURCE_REMOTE_ID) { + // If the only source allowed is remote-id option then we can skip + // the other methods. + return (HWAddrPtr()); + } + } + + // Method 6: From subscriber-id option inserted by a relay + + // Method 7: From docsis options + if (hw_addr_src & HWAddr::HWADDR_SOURCE_DOCSIS_CMTS) { + mac = getMACFromDocsisCMTS(); + if (mac) { + return (mac); + } else if (hw_addr_src == HWAddr::HWADDR_SOURCE_DOCSIS_CMTS) { + // If we're interested only in CMTS options as a source of that + // info, there's no point in trying other options. + return (HWAddrPtr()); + } + } + + // Method 8: From docsis options + if (hw_addr_src & HWAddr::HWADDR_SOURCE_DOCSIS_MODEM) { + mac = getMACFromDocsisModem(); + if (mac) { + return (mac); + } else if (hw_addr_src == HWAddr::HWADDR_SOURCE_DOCSIS_MODEM) { + // If we're interested only in CMTS options as a source of that + // info, there's no point in trying other options. + return (HWAddrPtr()); + } + } + + // Ok, none of the methods were suitable. Return NULL. + return (HWAddrPtr()); +} + +HWAddrPtr +Pkt::getMACFromIPv6(const isc::asiolink::IOAddress& addr) { + HWAddrPtr mac; + + if (addr.isV6LinkLocal()) { + std::vector<uint8_t> bin = addr.toBytes(); + + // Double check that it's of appropriate size + if ((bin.size() == isc::asiolink::V6ADDRESS_LEN) && + // Check that it's link-local (starts with fe80). + (bin[0] == 0xfe) && (bin[1] == 0x80) && + // Check that u bit is set and g is clear. + // See Section 2.5.1 of RFC2373 for details. + ((bin[8] & 3) == 2) && + // And that the IID is of EUI-64 type. + (bin[11] == 0xff) && (bin[12] == 0xfe)) { + + // Remove 8 most significant bytes + bin.erase(bin.begin(), bin.begin() + 8); + + // Ok, we're down to EUI-64 only now: XX:XX:XX:ff:fe:XX:XX:XX + bin.erase(bin.begin() + 3, bin.begin() + 5); + + // MAC-48 to EUI-64 involves inverting u bit (see explanation + // in Section 2.5.1 of RFC2373). We need to revert that. + bin[0] = bin[0] ^ 2; + + // Let's get the interface this packet was received on. + // We need it to get hardware type + IfacePtr iface = IfaceMgr::instance().getIface(iface_); + uint16_t hwtype = 0; // not specified + if (iface) { + hwtype = iface->getHWType(); + } + + mac.reset(new HWAddr(bin, hwtype)); + mac->source_ = HWAddr::HWADDR_SOURCE_IPV6_LINK_LOCAL; + } + } + + return (mac); +} + +} // end of namespace isc::dhcp +} // end of namespace isc |