diff options
Diffstat (limited to '')
-rw-r--r-- | src/lib/dhcpsrv/pool.h | 491 |
1 files changed, 491 insertions, 0 deletions
diff --git a/src/lib/dhcpsrv/pool.h b/src/lib/dhcpsrv/pool.h new file mode 100644 index 0000000..0015bb1 --- /dev/null +++ b/src/lib/dhcpsrv/pool.h @@ -0,0 +1,491 @@ +// Copyright (C) 2012-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/. + +#ifndef POOL_H +#define POOL_H + +#include <asiolink/io_address.h> +#include <cc/data.h> +#include <cc/user_context.h> +#include <dhcp/classify.h> +#include <dhcp/option6_pdexclude.h> +#include <dhcpsrv/allocation_state.h> +#include <dhcpsrv/cfg_option.h> +#include <dhcpsrv/lease.h> +#include <dhcpsrv/ip_range_permutation.h> +#include <util/bigints.h> + +#include <boost/shared_ptr.hpp> + +#include <vector> + +namespace isc { +namespace dhcp { + +/// @brief base class for Pool4 and Pool6 +/// +/// Stores information about pool of IPv4 or IPv6 addresses. +/// That is a basic component of a configuration. +class Pool : public isc::data::UserContext, public isc::data::CfgToElement { + +public: + /// @note: + /// PoolType enum was removed. Please use Lease::Type instead + + /// @brief Returns Pool-id + /// + /// Pool-id is an unique value that can be used to identify a pool within a + /// subnet or shared network. + /// + /// @return pool-id value + uint64_t getID() const { + return (id_); + } + + /// @brief Sets Pool-id + /// + /// Pool-id is an unique value that can be used to identify a pool within a + /// subnet or shared network. + /// + /// @param id value to be set + void setID(const uint64_t id) { + id_ = id; + } + + /// @brief Returns the first address in a pool. + /// + /// @return first address in a pool + const isc::asiolink::IOAddress& getFirstAddress() const { + return (first_); + } + + /// @brief Returns the last address in a pool. + /// @return last address in a pool + const isc::asiolink::IOAddress& getLastAddress() const { + return (last_); + } + + /// @brief Checks if a given address is in the range. + /// + /// @return true, if the address is in pool + bool inRange(const isc::asiolink::IOAddress& addr) const; + + /// @brief Returns pool type (v4, v6 non-temporary, v6 temp, v6 prefix) + /// @return returns pool type + Lease::Type getType() const { + return (type_); + } + + /// @brief returns textual representation of the pool + /// + /// @return textual representation + virtual std::string toText() const; + + /// @brief virtual destructor + /// + /// We need Pool to be a polymorphic class, so we could dynamic cast + /// from PoolPtr to Pool6Ptr if we need to. A class becomes polymorphic, + /// when there is at least one virtual method. + virtual ~Pool() = default; + + /// @brief Returns the number of all leases in this pool. + /// + /// Note that this is the upper bound, assuming that no leases are used + /// and there are no host reservations. This is just a theoretical calculation. + /// @return number of possible leases in this pool + isc::util::uint128_t getCapacity() const { + return (capacity_); + } + + /// @brief Returns pointer to the option data configuration for this pool. + CfgOptionPtr getCfgOption() { + return (cfg_option_); + } + + /// @brief Returns const pointer to the option data configuration for + /// this pool. + ConstCfgOptionPtr getCfgOption() const { + return (cfg_option_); + } + + /// @brief Checks whether this pool supports client that belongs to + /// specified classes. + /// + /// @todo: currently doing the same as network which needs improving. + /// + /// @param client_classes list of all classes the client belongs to + /// @return true if client can be supported, false otherwise + bool clientSupported(const ClientClasses& client_classes) const; + + /// @brief Sets the supported class to class class_name + /// + /// @param class_name client class to be supported by this pool + void allowClientClass(const ClientClass& class_name); + + /// @brief returns the client class + /// + /// @note The returned reference is only valid as long as the object + /// returned is valid. + /// + /// @return client class @ref client_class_ + const ClientClass& getClientClass() const { + return (client_class_); + } + + /// @brief Adds class class_name to classes required to be evaluated + /// + /// @param class_name client class required to be evaluated + void requireClientClass(const ClientClass& class_name) { + if (!required_classes_.contains(class_name)) { + required_classes_.insert(class_name); + } + } + + /// @brief Returns classes which are required to be evaluated + const ClientClasses& getRequiredClasses() const { + return (required_classes_); + } + + /// @brief Returns pool-specific allocation state. + /// + /// The actual type of the state depends on the allocator type. + /// + /// @return allocation state. + AllocationStatePtr getAllocationState() const { + return (allocation_state_); + } + + /// @brief Sets pool-specific allocation state. + /// + /// @param allocation_state allocation state instance. + void setAllocationState(const AllocationStatePtr& allocation_state) { + allocation_state_ = allocation_state; + } + + /// @brief Unparse a pool object. + /// + /// @return A pointer to unparsed pool configuration. + virtual data::ElementPtr toElement() const; + +protected: + + /// @brief protected constructor + /// + /// This constructor is protected to prevent anyone from instantiating + /// Pool class directly. Instances of Pool4 and Pool6 should be created + /// instead. + /// + /// @param type type of lease that will be served from this pool + /// @param first first address of a range + /// @param last last address of a range + Pool(Lease::Type type, + const isc::asiolink::IOAddress& first, + const isc::asiolink::IOAddress& last); + + /// @brief pool-id + /// + /// This id is an unique value that can be used to identify a pool within a + /// subnet or shared network. + uint64_t id_; + + /// @brief The first address in a pool + isc::asiolink::IOAddress first_; + + /// @brief The last address in a pool + isc::asiolink::IOAddress last_; + + /// @brief defines a lease type that will be served from this pool + Lease::Type type_; + + /// @brief Stores number of possible leases. + /// + /// This could be calculated on the fly, but the calculations are somewhat + /// involved, so it is more efficient to calculate it once and just store + /// the result. Note that for very large pools, the number is capped at + /// max value of uint64_t. + isc::util::uint128_t capacity_; + + /// @brief Pointer to the option data configuration for this pool. + CfgOptionPtr cfg_option_; + + /// @brief Optional definition of a client class + /// + /// @ref Network::client_class_ + ClientClass client_class_; + + /// @brief Required classes + /// + /// @ref isc::dhcp::Network::required_classes_ + ClientClasses required_classes_; + + /// @brief Pointer to the user context (may be NULL) + data::ConstElementPtr user_context_; + + /// @brief Holds pool-specific allocation state. + AllocationStatePtr allocation_state_; +}; + +class Pool4; + +/// @brief a pointer an IPv4 Pool +typedef boost::shared_ptr<Pool4> Pool4Ptr; + +/// @brief Pool information for IPv4 addresses +/// +/// It holds information about pool4, i.e. a range of IPv4 address space that +/// is configured for DHCP allocation. +class Pool4 : public Pool { +public: + /// @brief the constructor for Pool4 "min-max" style definition + /// + /// @param first the first address in a pool + /// @param last the last address in a pool + Pool4(const isc::asiolink::IOAddress& first, + const isc::asiolink::IOAddress& last); + + /// @brief the constructor for Pool4 "prefix/len" style definition + /// + /// @param prefix specifies prefix of the pool + /// @param prefix_len specifies length of the prefix of the pool + Pool4(const isc::asiolink::IOAddress& prefix, + uint8_t prefix_len); + + /// @brief Factory function for creating an instance of the @c Pool4. + /// + /// This function should be used to create an instance of the pool + /// within a hooks library in cases when the library may be unloaded + /// before the object is destroyed. This ensures that the ownership + /// of the object by the Kea process is retained. + /// + /// @param first the first address in a pool + /// @param last the last address in a pool + /// + /// @return Pointer to the @c Pool4 instance. + static Pool4Ptr create(const isc::asiolink::IOAddress& first, + const isc::asiolink::IOAddress& last); + + /// @brief Factory function for creating an instance of the @c Pool4. + /// + /// This function should be used to create an instance of the pool + /// within a hooks library in cases when the library may be unloaded + /// before the object is destroyed. This ensures that the ownership + /// of the object by the Kea process is retained. + /// + /// @param prefix specifies prefix of the pool. + /// @param prefix_len specifies length of the prefix of the pool. + /// + /// @return Pointer to the @c Pool4 instance. + static Pool4Ptr create(const isc::asiolink::IOAddress& prefix, + uint8_t prefix_len); + + /// @brief Unparse a Pool4 object. + /// + /// @return A pointer to unparsed Pool4 configuration. + virtual data::ElementPtr toElement() const; +}; + +class Pool6; + +/// @brief a pointer an IPv6 Pool +typedef boost::shared_ptr<Pool6> Pool6Ptr; + +/// @brief Pool information for IPv6 addresses and prefixes +/// +/// It holds information about pool6, i.e. a range of IPv6 address space that +/// is configured for DHCP allocation. +class Pool6 : public Pool { +public: + + /// @brief the constructor for Pool6 "min-max" style definition + /// + /// @throw BadValue if PD is define (PD can be only prefix/len) + /// + /// @param type type of the pool (IA or TA) + /// @param first the first address in a pool + /// @param last the last address in a pool + Pool6(Lease::Type type, const isc::asiolink::IOAddress& first, + const isc::asiolink::IOAddress& last); + + /// @brief the constructor for Pool6 "prefix/len" style definition + /// + /// For addressed, this is just a prefix/len definition. For prefixes, + /// there is one extra additional parameter delegated_len. It specifies + /// a size of delegated prefixes that the pool will be split into. For + /// example pool 2001:db8::/56, delegated_len=64 means that there is a + /// pool 2001:db8::/56. It will be split into 256 prefixes of length /64, + /// e.g. 2001:db8:0:1::/64, 2001:db8:0:2::/64 etc. + /// + /// Naming convention: + /// A smaller prefix length yields a shorter prefix which describes a larger + /// set of addresses. A larger length yields a longer prefix which describes + /// a smaller set of addresses. + /// + /// Obviously, prefix_len must define shorter or equal prefix length than + /// delegated_len, so prefix_len <= delegated_len. Note that it is slightly + /// confusing: bigger (larger) prefix actually has smaller prefix length, + /// e.g. /56 is a bigger prefix than /64, but has shorter (smaller) prefix + /// length. + /// + /// @throw BadValue if delegated_len is defined for non-PD types or + /// when delegated_len < prefix_len + /// + /// @param type type of the pool (IA, TA or PD) + /// @param prefix specifies prefix of the pool + /// @param prefix_len specifies prefix length of the pool + /// @param delegated_len specifies length of the delegated prefixes + Pool6(Lease::Type type, const isc::asiolink::IOAddress& prefix, + uint8_t prefix_len, uint8_t delegated_len = 128); + + /// @brief Constructor for DHCPv6 prefix pool with an excluded prefix. + /// + /// If @c excluded_prefix is equal to '::' and the @c excluded_prefix_len + /// is equal to 0, the excluded prefix is assumed to be unspecified for + /// the pool. In this case, the server will not send the Prefix Exclude + /// option to a client. + /// + /// @param prefix specified a prefix of the pool. + /// @param prefix_len specifies prefix length of the pool. + /// @param delegated_len specifies length of the delegated prefixes. + /// @param excluded_prefix specifies an excluded prefix as per RFC6603. + /// @param excluded_prefix_len specifies length of an excluded prefix. + Pool6(const asiolink::IOAddress& prefix, const uint8_t prefix_len, + const uint8_t delegated_len, + const asiolink::IOAddress& excluded_prefix, + const uint8_t excluded_prefix_len); + + /// @brief Factory function for creating an instance of the @c Pool6. + /// + /// This function should be used to create an instance of the pool + /// within a hooks library in cases when the library may be unloaded + /// before the object is destroyed. This ensures that the ownership + /// of the object by the Kea process is retained. + /// + /// @param type type of the pool (IA or TA) + /// @param first the first address in a pool + /// @param last the last address in a pool + /// + /// @return Pointer to the @c Pool6 instance. + static Pool6Ptr create(Lease::Type type, + const isc::asiolink::IOAddress& first, + const isc::asiolink::IOAddress& last); + + /// @brief Factory function for creating an instance of the @c Pool6. + /// + /// This function should be used to create an instance of the pool + /// within a hooks library in cases when the library may be unloaded + /// before the object is destroyed. This ensures that the ownership + /// of the object by the Kea process is retained. + /// + /// @param type type of the pool (IA, TA or PD) + /// @param prefix specifies prefix of the pool + /// @param prefix_len specifies prefix length of the pool + /// @param delegated_len specifies length of the delegated prefixes + /// + /// @return Pointer to the @c Pool6 instance. + static Pool6Ptr create(Lease::Type type, + const isc::asiolink::IOAddress& prefix, + uint8_t prefix_len, + uint8_t delegated_len = 128); + + /// @brief Factory function for creating an instance of the @c Pool6. + /// + /// If @c excluded_prefix is equal to '::' and the @c excluded_prefix_len + /// is equal to 0, the excluded prefix is assumed to be unspecified for + /// the pool. In this case, the server will not send the Prefix Exclude + /// option to a client. + /// + /// @param prefix specifies a prefix of the pool. + /// @param prefix_len specifies prefix length of the pool. + /// @param delegated_len specifies length of the delegated prefixes. + /// @param excluded_prefix specifies an excluded prefix as per RFC6603. + /// @param excluded_prefix_len specifies length of an excluded prefix. + /// + /// @return Pointer to the @c Pool6 instance. + static Pool6Ptr create(const asiolink::IOAddress& prefix, + const uint8_t prefix_len, + const uint8_t delegated_len, + const asiolink::IOAddress& excluded_prefix, + const uint8_t excluded_prefix_len); + + /// @brief returns pool type + /// + /// @return pool type + Lease::Type getType() const { + return (type_); + } + + /// @brief returns delegated prefix length + /// + /// This may be useful for "prefix/len" style definition for + /// addresses, but is mostly useful for prefix pools. + /// @return prefix length (1-128) + uint8_t getLength() const { + return (prefix_len_); + } + + /// @brief Returns instance of the pool specific Prefix Exclude option. + /// + /// @return An instance of the Prefix Exclude option (RFC 6603) or NULL + /// if such option hasn't been specified for the pool. + Option6PDExcludePtr getPrefixExcludeOption() const { + return (pd_exclude_option_); + } + + /// @brief Unparse a Pool6 object. + /// + /// @return A pointer to unparsed Pool6 configuration. + virtual data::ElementPtr toElement() const; + + /// @brief returns textual representation of the pool + /// + /// @return textual representation + virtual std::string toText() const; + +private: + + /// @brief Generic method initializing a DHCPv6 pool. + /// + /// This method should be called by the constructors to initialize + /// DHCPv6 pools. + /// + /// @param Lease/pool type. + /// @param prefix An address or delegated prefix (depending on the + /// pool type specified as @c type). + /// @param prefix_len Prefix length. If a pool is an address pool, + /// this value should be set to 128. + /// @param delegated_len Length of the delegated prefixes. If a pool + /// is an address pool, this value should be set to 128. + /// @param excluded_prefix An excluded prefix as per RFC6603. This + /// value should only be specified for prefix pools. The value of + /// '::' means "unspecified". + /// @param excluded_prefix_len Length of the excluded prefix. This + /// is only specified for prefix pools. The value of 0 should be + /// used when @c excluded_prefix is not specified. + void init(const Lease::Type& type, + const asiolink::IOAddress& prefix, + const uint8_t prefix_len, + const uint8_t delegated_len, + const asiolink::IOAddress& excluded_prefix, + const uint8_t excluded_prefix_len); + + /// @brief Defines prefix length (for TYPE_PD only) + uint8_t prefix_len_; + + /// @brief A pointer to the Prefix Exclude option (RFC 6603). + Option6PDExcludePtr pd_exclude_option_; + +}; + +/// @brief a pointer to either IPv4 or IPv6 Pool +typedef boost::shared_ptr<Pool> PoolPtr; + +/// @brief a container for either IPv4 or IPv6 Pools +typedef std::vector<PoolPtr> PoolCollection; + +} // end of isc::dhcp namespace +} // end of isc namespace + +#endif // POOL_H |