genetlink_linux.go 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180
  1. package netlink
  2. import (
  3. "errors"
  4. "fmt"
  5. "syscall"
  6. "github.com/vishvananda/netlink/nl"
  7. "golang.org/x/sys/unix"
  8. )
  9. type GenlOp struct {
  10. ID uint32
  11. Flags uint32
  12. }
  13. type GenlMulticastGroup struct {
  14. ID uint32
  15. Name string
  16. }
  17. type GenlFamily struct {
  18. ID uint16
  19. HdrSize uint32
  20. Name string
  21. Version uint32
  22. MaxAttr uint32
  23. Ops []GenlOp
  24. Groups []GenlMulticastGroup
  25. }
  26. func parseOps(b []byte) ([]GenlOp, error) {
  27. attrs, err := nl.ParseRouteAttr(b)
  28. if err != nil {
  29. return nil, err
  30. }
  31. ops := make([]GenlOp, 0, len(attrs))
  32. for _, a := range attrs {
  33. nattrs, err := nl.ParseRouteAttr(a.Value)
  34. if err != nil {
  35. return nil, err
  36. }
  37. var op GenlOp
  38. for _, na := range nattrs {
  39. switch na.Attr.Type {
  40. case nl.GENL_CTRL_ATTR_OP_ID:
  41. op.ID = native.Uint32(na.Value)
  42. case nl.GENL_CTRL_ATTR_OP_FLAGS:
  43. op.Flags = native.Uint32(na.Value)
  44. }
  45. }
  46. ops = append(ops, op)
  47. }
  48. return ops, nil
  49. }
  50. func parseMulticastGroups(b []byte) ([]GenlMulticastGroup, error) {
  51. attrs, err := nl.ParseRouteAttr(b)
  52. if err != nil {
  53. return nil, err
  54. }
  55. groups := make([]GenlMulticastGroup, 0, len(attrs))
  56. for _, a := range attrs {
  57. nattrs, err := nl.ParseRouteAttr(a.Value)
  58. if err != nil {
  59. return nil, err
  60. }
  61. var g GenlMulticastGroup
  62. for _, na := range nattrs {
  63. switch na.Attr.Type {
  64. case nl.GENL_CTRL_ATTR_MCAST_GRP_NAME:
  65. g.Name = nl.BytesToString(na.Value)
  66. case nl.GENL_CTRL_ATTR_MCAST_GRP_ID:
  67. g.ID = native.Uint32(na.Value)
  68. }
  69. }
  70. groups = append(groups, g)
  71. }
  72. return groups, nil
  73. }
  74. func (f *GenlFamily) parseAttributes(attrs []syscall.NetlinkRouteAttr) error {
  75. for _, a := range attrs {
  76. switch a.Attr.Type {
  77. case nl.GENL_CTRL_ATTR_FAMILY_NAME:
  78. f.Name = nl.BytesToString(a.Value)
  79. case nl.GENL_CTRL_ATTR_FAMILY_ID:
  80. f.ID = native.Uint16(a.Value)
  81. case nl.GENL_CTRL_ATTR_VERSION:
  82. f.Version = native.Uint32(a.Value)
  83. case nl.GENL_CTRL_ATTR_HDRSIZE:
  84. f.HdrSize = native.Uint32(a.Value)
  85. case nl.GENL_CTRL_ATTR_MAXATTR:
  86. f.MaxAttr = native.Uint32(a.Value)
  87. case nl.GENL_CTRL_ATTR_OPS:
  88. ops, err := parseOps(a.Value)
  89. if err != nil {
  90. return err
  91. }
  92. f.Ops = ops
  93. case nl.GENL_CTRL_ATTR_MCAST_GROUPS:
  94. groups, err := parseMulticastGroups(a.Value)
  95. if err != nil {
  96. return err
  97. }
  98. f.Groups = groups
  99. }
  100. }
  101. return nil
  102. }
  103. func parseFamilies(msgs [][]byte) ([]*GenlFamily, error) {
  104. families := make([]*GenlFamily, 0, len(msgs))
  105. for _, m := range msgs {
  106. attrs, err := nl.ParseRouteAttr(m[nl.SizeofGenlmsg:])
  107. if err != nil {
  108. return nil, err
  109. }
  110. family := &GenlFamily{}
  111. if err := family.parseAttributes(attrs); err != nil {
  112. return nil, err
  113. }
  114. families = append(families, family)
  115. }
  116. return families, nil
  117. }
  118. // If the returned error is [ErrDumpInterrupted], results may be inconsistent
  119. // or incomplete.
  120. func (h *Handle) GenlFamilyList() ([]*GenlFamily, error) {
  121. msg := &nl.Genlmsg{
  122. Command: nl.GENL_CTRL_CMD_GETFAMILY,
  123. Version: nl.GENL_CTRL_VERSION,
  124. }
  125. req := h.newNetlinkRequest(nl.GENL_ID_CTRL, unix.NLM_F_DUMP)
  126. req.AddData(msg)
  127. msgs, executeErr := req.Execute(unix.NETLINK_GENERIC, 0)
  128. if executeErr != nil && !errors.Is(executeErr, ErrDumpInterrupted) {
  129. return nil, executeErr
  130. }
  131. families, err := parseFamilies(msgs)
  132. if err != nil {
  133. return nil, err
  134. }
  135. return families, executeErr
  136. }
  137. // If the returned error is [ErrDumpInterrupted], results may be inconsistent
  138. // or incomplete.
  139. func GenlFamilyList() ([]*GenlFamily, error) {
  140. return pkgHandle.GenlFamilyList()
  141. }
  142. func (h *Handle) GenlFamilyGet(name string) (*GenlFamily, error) {
  143. msg := &nl.Genlmsg{
  144. Command: nl.GENL_CTRL_CMD_GETFAMILY,
  145. Version: nl.GENL_CTRL_VERSION,
  146. }
  147. req := h.newNetlinkRequest(nl.GENL_ID_CTRL, 0)
  148. req.AddData(msg)
  149. req.AddData(nl.NewRtAttr(nl.GENL_CTRL_ATTR_FAMILY_NAME, nl.ZeroTerminated(name)))
  150. msgs, err := req.Execute(unix.NETLINK_GENERIC, 0)
  151. if err != nil {
  152. return nil, err
  153. }
  154. families, err := parseFamilies(msgs)
  155. if err != nil {
  156. return nil, err
  157. }
  158. if len(families) != 1 {
  159. return nil, fmt.Errorf("invalid response for GENL_CTRL_CMD_GETFAMILY")
  160. }
  161. return families[0], nil
  162. }
  163. func GenlFamilyGet(name string) (*GenlFamily, error) {
  164. return pkgHandle.GenlFamilyGet(name)
  165. }