2023-12-07 14:23:00 +00:00
|
|
|
package dhcpsvc
|
|
|
|
|
|
|
|
import (
|
2024-07-03 13:38:37 +01:00
|
|
|
"context"
|
2023-12-07 14:23:00 +00:00
|
|
|
"fmt"
|
2024-07-03 13:38:37 +01:00
|
|
|
"log/slog"
|
2023-12-07 14:23:00 +00:00
|
|
|
"net/netip"
|
2024-03-12 14:45:11 +00:00
|
|
|
"slices"
|
2023-12-07 14:23:00 +00:00
|
|
|
"time"
|
|
|
|
|
2024-07-03 13:38:37 +01:00
|
|
|
"github.com/AdguardTeam/golibs/errors"
|
2024-03-12 14:45:11 +00:00
|
|
|
"github.com/AdguardTeam/golibs/netutil"
|
2023-12-07 14:23:00 +00:00
|
|
|
"github.com/google/gopacket/layers"
|
|
|
|
)
|
|
|
|
|
|
|
|
// IPv6Config is the interface-specific configuration for DHCPv6.
|
|
|
|
type IPv6Config struct {
|
|
|
|
// RangeStart is the first address in the range to assign to DHCP clients.
|
|
|
|
RangeStart netip.Addr
|
|
|
|
|
2024-01-30 15:43:51 +00:00
|
|
|
// Options is the list of DHCP options to send to DHCP clients. The options
|
|
|
|
// with zero length are treated as deletions of the corresponding options,
|
|
|
|
// either implicit or explicit.
|
|
|
|
Options layers.DHCPv6Options
|
2023-12-07 14:23:00 +00:00
|
|
|
|
|
|
|
// LeaseDuration is the TTL of a DHCP lease.
|
|
|
|
LeaseDuration time.Duration
|
|
|
|
|
|
|
|
// RASlaacOnly defines whether the DHCP clients should only use SLAAC for
|
|
|
|
// address assignment.
|
|
|
|
RASLAACOnly bool
|
|
|
|
|
|
|
|
// RAAllowSlaac defines whether the DHCP clients may use SLAAC for address
|
|
|
|
// assignment.
|
|
|
|
RAAllowSLAAC bool
|
|
|
|
|
|
|
|
// Enabled is the state of the DHCPv6 service, whether it is enabled or not
|
|
|
|
// on the specific interface.
|
|
|
|
Enabled bool
|
|
|
|
}
|
|
|
|
|
|
|
|
// validate returns an error in conf if any.
|
2024-07-03 13:38:37 +01:00
|
|
|
func (c *IPv6Config) validate() (err error) {
|
|
|
|
if c == nil {
|
2023-12-07 14:23:00 +00:00
|
|
|
return errNilConfig
|
2024-07-03 13:38:37 +01:00
|
|
|
} else if !c.Enabled {
|
2023-12-07 14:23:00 +00:00
|
|
|
return nil
|
|
|
|
}
|
2024-03-12 14:45:11 +00:00
|
|
|
|
2024-07-03 13:38:37 +01:00
|
|
|
var errs []error
|
2024-03-12 14:45:11 +00:00
|
|
|
|
2024-07-03 13:38:37 +01:00
|
|
|
if !c.RangeStart.Is6() {
|
|
|
|
err = fmt.Errorf("range start %s should be a valid ipv6", c.RangeStart)
|
|
|
|
errs = append(errs, err)
|
2024-03-12 14:45:11 +00:00
|
|
|
}
|
|
|
|
|
2024-07-03 13:38:37 +01:00
|
|
|
if c.LeaseDuration <= 0 {
|
|
|
|
err = fmt.Errorf("lease duration %s must be positive", c.LeaseDuration)
|
|
|
|
errs = append(errs, err)
|
|
|
|
}
|
2024-03-12 14:45:11 +00:00
|
|
|
|
2024-07-03 13:38:37 +01:00
|
|
|
return errors.Join(errs...)
|
2024-03-12 14:45:11 +00:00
|
|
|
}
|
|
|
|
|
2024-09-30 18:17:20 +01:00
|
|
|
// dhcpInterfaceV6 is a DHCP interface for IPv6 address family.
|
|
|
|
type dhcpInterfaceV6 struct {
|
|
|
|
// common is the common part of any network interface within the DHCP
|
|
|
|
// server.
|
|
|
|
common *netInterface
|
|
|
|
|
2023-12-07 14:23:00 +00:00
|
|
|
// rangeStart is the first IP address in the range.
|
|
|
|
rangeStart netip.Addr
|
|
|
|
|
2024-01-30 15:43:51 +00:00
|
|
|
// implicitOpts are the DHCPv6 options listed in RFC 8415 (and others) and
|
|
|
|
// initialized with default values. It must not have intersections with
|
|
|
|
// explicitOpts.
|
|
|
|
implicitOpts layers.DHCPv6Options
|
|
|
|
|
|
|
|
// explicitOpts are the user-configured options. It must not have
|
|
|
|
// intersections with implicitOpts.
|
|
|
|
explicitOpts layers.DHCPv6Options
|
|
|
|
|
2023-12-07 14:23:00 +00:00
|
|
|
// raSLAACOnly defines if DHCP should send ICMPv6.RA packets without MO
|
|
|
|
// flags.
|
|
|
|
raSLAACOnly bool
|
|
|
|
|
|
|
|
// raAllowSLAAC defines if DHCP should send ICMPv6.RA packets with MO flags.
|
|
|
|
raAllowSLAAC bool
|
|
|
|
}
|
|
|
|
|
2024-09-30 18:17:20 +01:00
|
|
|
// newDHCPInterfaceV6 creates a new DHCP interface for IPv6 address family with
|
2024-03-12 14:45:11 +00:00
|
|
|
// the given configuration.
|
2023-12-07 14:23:00 +00:00
|
|
|
//
|
|
|
|
// TODO(e.burkov): Validate properly.
|
2024-09-30 18:17:20 +01:00
|
|
|
func newDHCPInterfaceV6(
|
2024-07-03 13:38:37 +01:00
|
|
|
ctx context.Context,
|
|
|
|
l *slog.Logger,
|
|
|
|
name string,
|
|
|
|
conf *IPv6Config,
|
2024-09-30 18:17:20 +01:00
|
|
|
) (i *dhcpInterfaceV6) {
|
2024-07-03 13:38:37 +01:00
|
|
|
l = l.With(keyInterface, name, keyFamily, netutil.AddrFamilyIPv6)
|
2023-12-07 14:23:00 +00:00
|
|
|
if !conf.Enabled {
|
2024-07-03 13:38:37 +01:00
|
|
|
l.DebugContext(ctx, "disabled")
|
|
|
|
|
2023-12-07 14:23:00 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2024-09-30 18:17:20 +01:00
|
|
|
i = &dhcpInterfaceV6{
|
|
|
|
rangeStart: conf.RangeStart,
|
|
|
|
common: newNetInterface(name, l, conf.LeaseDuration),
|
2023-12-07 14:23:00 +00:00
|
|
|
raSLAACOnly: conf.RASLAACOnly,
|
|
|
|
raAllowSLAAC: conf.RAAllowSLAAC,
|
|
|
|
}
|
2024-07-03 13:38:37 +01:00
|
|
|
i.implicitOpts, i.explicitOpts = conf.options(ctx, l)
|
2024-01-30 15:43:51 +00:00
|
|
|
|
|
|
|
return i
|
|
|
|
}
|
|
|
|
|
2024-09-30 18:17:20 +01:00
|
|
|
// dhcpInterfacesV6 is a slice of network interfaces of IPv6 address family.
|
|
|
|
type dhcpInterfacesV6 []*dhcpInterfaceV6
|
2024-03-12 14:45:11 +00:00
|
|
|
|
|
|
|
// find returns the first network interface within ifaces containing ip. It
|
|
|
|
// returns false if there is no such interface.
|
2024-09-30 18:17:20 +01:00
|
|
|
func (ifaces dhcpInterfacesV6) find(ip netip.Addr) (iface6 *netInterface, ok bool) {
|
2024-03-12 14:45:11 +00:00
|
|
|
// prefLen is the length of prefix to match ip against.
|
|
|
|
//
|
|
|
|
// TODO(e.burkov): DHCPv6 inherits the weird behavior of legacy
|
|
|
|
// implementation where the allocated range constrained by the first address
|
|
|
|
// and the first address with last byte set to 0xff. Proper prefixes should
|
|
|
|
// be used instead.
|
|
|
|
const prefLen = netutil.IPv6BitLen - 8
|
|
|
|
|
2024-09-30 18:17:20 +01:00
|
|
|
i := slices.IndexFunc(ifaces, func(iface *dhcpInterfaceV6) (contains bool) {
|
2024-03-12 14:45:11 +00:00
|
|
|
return !ip.Less(iface.rangeStart) &&
|
|
|
|
netip.PrefixFrom(iface.rangeStart, prefLen).Contains(ip)
|
|
|
|
})
|
|
|
|
if i < 0 {
|
|
|
|
return nil, false
|
2024-01-30 15:43:51 +00:00
|
|
|
}
|
|
|
|
|
2024-09-30 18:17:20 +01:00
|
|
|
return ifaces[i].common, true
|
2023-12-07 14:23:00 +00:00
|
|
|
}
|
2024-07-03 13:38:37 +01:00
|
|
|
|
|
|
|
// options returns the implicit and explicit options for the interface. The two
|
|
|
|
// lists are disjoint and the implicit options are initialized with default
|
|
|
|
// values.
|
|
|
|
//
|
|
|
|
// TODO(e.burkov): Add implicit options according to RFC.
|
|
|
|
func (c *IPv6Config) options(ctx context.Context, l *slog.Logger) (imp, exp layers.DHCPv6Options) {
|
|
|
|
// Set default values of host configuration parameters listed in RFC 8415.
|
|
|
|
imp = layers.DHCPv6Options{}
|
|
|
|
slices.SortFunc(imp, compareV6OptionCodes)
|
|
|
|
|
|
|
|
// Set values for explicitly configured options.
|
|
|
|
for _, e := range c.Options {
|
|
|
|
i, found := slices.BinarySearchFunc(imp, e, compareV6OptionCodes)
|
|
|
|
if found {
|
|
|
|
imp = slices.Delete(imp, i, i+1)
|
|
|
|
}
|
|
|
|
|
|
|
|
exp = append(exp, e)
|
|
|
|
}
|
|
|
|
|
|
|
|
l.DebugContext(ctx, "options", "implicit", imp, "explicit", exp)
|
|
|
|
|
|
|
|
return imp, exp
|
|
|
|
}
|
|
|
|
|
|
|
|
// compareV6OptionCodes compares option codes of a and b.
|
|
|
|
func compareV6OptionCodes(a, b layers.DHCPv6Option) (res int) {
|
|
|
|
return int(a.Code) - int(b.Code)
|
|
|
|
}
|