// Copyright 2018 Google LLC. All Rights Reserved. // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. package nftables import ( "encoding/binary" "net" "github.com/google/nftables/binaryutil" "golang.org/x/sys/unix" ) func extraHeader(family uint8, resID uint16) []byte { return append([]byte{ family, unix.NFNETLINK_V0, }, binaryutil.BigEndian.PutUint16(resID)...) } // General form of address family dependent message, see // https://git.netfilter.org/libnftnl/tree/include/linux/netfilter/nfnetlink.h#29 type NFGenMsg struct { NFGenFamily uint8 Version uint8 ResourceID uint16 } func (genmsg *NFGenMsg) Decode(b []byte) { if len(b) < 16 { return } genmsg.NFGenFamily = b[0] genmsg.Version = b[1] genmsg.ResourceID = binary.BigEndian.Uint16(b[2:]) } // GetFirstAndLastIPFromCIDR returns the first and last IP address from a CIDR. func GetFirstAndLastIPFromCIDR(cidr string) (firstIP, lastIP net.IP, err error) { _, subnet, err := net.ParseCIDR(cidr) if err != nil { return nil, nil, err } firstIP = make(net.IP, len(subnet.IP)) lastIP = make(net.IP, len(subnet.IP)) switch len(subnet.IP) { case net.IPv4len: mask := binary.BigEndian.Uint32(subnet.Mask) ip := binary.BigEndian.Uint32(subnet.IP) binary.BigEndian.PutUint32(firstIP, ip&mask) binary.BigEndian.PutUint32(lastIP, (ip&mask)|(mask^0xffffffff)) case net.IPv6len: mask1 := binary.BigEndian.Uint64(subnet.Mask[:8]) mask2 := binary.BigEndian.Uint64(subnet.Mask[8:]) ip1 := binary.BigEndian.Uint64(subnet.IP[:8]) ip2 := binary.BigEndian.Uint64(subnet.IP[8:]) binary.BigEndian.PutUint64(firstIP[:8], ip1&mask1) binary.BigEndian.PutUint64(firstIP[8:], ip2&mask2) binary.BigEndian.PutUint64(lastIP[:8], (ip1&mask1)|(mask1^0xffffffffffffffff)) binary.BigEndian.PutUint64(lastIP[8:], (ip2&mask2)|(mask2^0xffffffffffffffff)) } return firstIP, lastIP, nil }