123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215 |
- // +build linux
- package netlink
- import (
- "encoding/binary"
- "errors"
- "github.com/vishvananda/netlink/nl"
- "golang.org/x/sys/unix"
- )
- const (
- FOU_GENL_NAME = "fou"
- )
- const (
- FOU_CMD_UNSPEC uint8 = iota
- FOU_CMD_ADD
- FOU_CMD_DEL
- FOU_CMD_GET
- FOU_CMD_MAX = FOU_CMD_GET
- )
- const (
- FOU_ATTR_UNSPEC = iota
- FOU_ATTR_PORT
- FOU_ATTR_AF
- FOU_ATTR_IPPROTO
- FOU_ATTR_TYPE
- FOU_ATTR_REMCSUM_NOPARTIAL
- FOU_ATTR_MAX = FOU_ATTR_REMCSUM_NOPARTIAL
- )
- const (
- FOU_ENCAP_UNSPEC = iota
- FOU_ENCAP_DIRECT
- FOU_ENCAP_GUE
- FOU_ENCAP_MAX = FOU_ENCAP_GUE
- )
- var fouFamilyId int
- func FouFamilyId() (int, error) {
- if fouFamilyId != 0 {
- return fouFamilyId, nil
- }
- fam, err := GenlFamilyGet(FOU_GENL_NAME)
- if err != nil {
- return -1, err
- }
- fouFamilyId = int(fam.ID)
- return fouFamilyId, nil
- }
- func FouAdd(f Fou) error {
- return pkgHandle.FouAdd(f)
- }
- func (h *Handle) FouAdd(f Fou) error {
- fam_id, err := FouFamilyId()
- if err != nil {
- return err
- }
- // setting ip protocol conflicts with encapsulation type GUE
- if f.EncapType == FOU_ENCAP_GUE && f.Protocol != 0 {
- return errors.New("GUE encapsulation doesn't specify an IP protocol")
- }
- req := h.newNetlinkRequest(fam_id, unix.NLM_F_ACK)
- // int to byte for port
- bp := make([]byte, 2)
- binary.BigEndian.PutUint16(bp[0:2], uint16(f.Port))
- attrs := []*nl.RtAttr{
- nl.NewRtAttr(FOU_ATTR_PORT, bp),
- nl.NewRtAttr(FOU_ATTR_TYPE, []byte{uint8(f.EncapType)}),
- nl.NewRtAttr(FOU_ATTR_AF, []byte{uint8(f.Family)}),
- nl.NewRtAttr(FOU_ATTR_IPPROTO, []byte{uint8(f.Protocol)}),
- }
- raw := []byte{FOU_CMD_ADD, 1, 0, 0}
- for _, a := range attrs {
- raw = append(raw, a.Serialize()...)
- }
- req.AddRawData(raw)
- _, err = req.Execute(unix.NETLINK_GENERIC, 0)
- if err != nil {
- return err
- }
- return nil
- }
- func FouDel(f Fou) error {
- return pkgHandle.FouDel(f)
- }
- func (h *Handle) FouDel(f Fou) error {
- fam_id, err := FouFamilyId()
- if err != nil {
- return err
- }
- req := h.newNetlinkRequest(fam_id, unix.NLM_F_ACK)
- // int to byte for port
- bp := make([]byte, 2)
- binary.BigEndian.PutUint16(bp[0:2], uint16(f.Port))
- attrs := []*nl.RtAttr{
- nl.NewRtAttr(FOU_ATTR_PORT, bp),
- nl.NewRtAttr(FOU_ATTR_AF, []byte{uint8(f.Family)}),
- }
- raw := []byte{FOU_CMD_DEL, 1, 0, 0}
- for _, a := range attrs {
- raw = append(raw, a.Serialize()...)
- }
- req.AddRawData(raw)
- _, err = req.Execute(unix.NETLINK_GENERIC, 0)
- if err != nil {
- return err
- }
- return nil
- }
- func FouList(fam int) ([]Fou, error) {
- return pkgHandle.FouList(fam)
- }
- func (h *Handle) FouList(fam int) ([]Fou, error) {
- fam_id, err := FouFamilyId()
- if err != nil {
- return nil, err
- }
- req := h.newNetlinkRequest(fam_id, unix.NLM_F_DUMP)
- attrs := []*nl.RtAttr{
- nl.NewRtAttr(FOU_ATTR_AF, []byte{uint8(fam)}),
- }
- raw := []byte{FOU_CMD_GET, 1, 0, 0}
- for _, a := range attrs {
- raw = append(raw, a.Serialize()...)
- }
- req.AddRawData(raw)
- msgs, err := req.Execute(unix.NETLINK_GENERIC, 0)
- if err != nil {
- return nil, err
- }
- fous := make([]Fou, 0, len(msgs))
- for _, m := range msgs {
- f, err := deserializeFouMsg(m)
- if err != nil {
- return fous, err
- }
- fous = append(fous, f)
- }
- return fous, nil
- }
- func deserializeFouMsg(msg []byte) (Fou, error) {
- // we'll skip to byte 4 to first attribute
- msg = msg[3:]
- var shift int
- fou := Fou{}
- for {
- // attribute header is at least 16 bits
- if len(msg) < 4 {
- return fou, ErrAttrHeaderTruncated
- }
- lgt := int(binary.BigEndian.Uint16(msg[0:2]))
- if len(msg) < lgt+4 {
- return fou, ErrAttrBodyTruncated
- }
- attr := binary.BigEndian.Uint16(msg[2:4])
- shift = lgt + 3
- switch attr {
- case FOU_ATTR_AF:
- fou.Family = int(msg[5])
- case FOU_ATTR_PORT:
- fou.Port = int(binary.BigEndian.Uint16(msg[5:7]))
- // port is 2 bytes
- shift = lgt + 2
- case FOU_ATTR_IPPROTO:
- fou.Protocol = int(msg[5])
- case FOU_ATTR_TYPE:
- fou.EncapType = int(msg[5])
- }
- msg = msg[shift:]
- if len(msg) < 4 {
- break
- }
- }
- return fou, nil
- }
|