/osutil/udev/netlink/conn.go

https://github.com/snapcore/snapd · Go · 175 lines · 122 code · 27 blank · 26 comment · 31 complexity · 5b077b2857f63ffdb01d3c571bda16fa MD5 · raw file

  1. package netlink
  2. import (
  3. "fmt"
  4. "os"
  5. "syscall"
  6. "time"
  7. )
  8. type Mode int
  9. // Mode determines event source: kernel events or udev-processed events.
  10. // See libudev/libudev-monitor.c.
  11. const (
  12. KernelEvent Mode = 1
  13. // Events that are processed by udev - much richer, with more attributes (such as vendor info, serial numbers and more).
  14. UdevEvent Mode = 2
  15. )
  16. // Generic connection
  17. type NetlinkConn struct {
  18. Fd int
  19. Addr syscall.SockaddrNetlink
  20. }
  21. type UEventConn struct {
  22. NetlinkConn
  23. }
  24. // Connect allow to connect to system socket AF_NETLINK with family NETLINK_KOBJECT_UEVENT to
  25. // catch events about block/char device
  26. // see:
  27. // - http://elixir.free-electrons.com/linux/v3.12/source/include/uapi/linux/netlink.h#L23
  28. // - http://elixir.free-electrons.com/linux/v3.12/source/include/uapi/linux/socket.h#L11
  29. func (c *UEventConn) Connect(mode Mode) (err error) {
  30. if c.Fd, err = syscall.Socket(syscall.AF_NETLINK, syscall.SOCK_RAW, syscall.NETLINK_KOBJECT_UEVENT); err != nil {
  31. return
  32. }
  33. c.Addr = syscall.SockaddrNetlink{
  34. Family: syscall.AF_NETLINK,
  35. Groups: uint32(mode),
  36. Pid: uint32(os.Getpid()),
  37. }
  38. if err = syscall.Bind(c.Fd, &c.Addr); err != nil {
  39. syscall.Close(c.Fd)
  40. }
  41. return
  42. }
  43. // Close allow to close file descriptor and socket bound
  44. func (c *UEventConn) Close() error {
  45. return syscall.Close(c.Fd)
  46. }
  47. // ReadMsg allow to read an entire uevent msg
  48. func (c *UEventConn) ReadMsg() (msg []byte, err error) {
  49. var n int
  50. buf := make([]byte, os.Getpagesize())
  51. for {
  52. // Just read how many bytes are available in the socket
  53. if n, _, err = syscall.Recvfrom(c.Fd, buf, syscall.MSG_PEEK); err != nil {
  54. return
  55. }
  56. // If all message could be store inside the buffer : break
  57. if n < len(buf) {
  58. break
  59. }
  60. // Increase size of buffer if not enough
  61. buf = make([]byte, len(buf)+os.Getpagesize())
  62. }
  63. // Now read complete data
  64. n, _, err = syscall.Recvfrom(c.Fd, buf, 0)
  65. if err != nil {
  66. return
  67. }
  68. // Extract only real data from buffer and return that
  69. msg = buf[:n]
  70. return
  71. }
  72. // ReadMsg allow to read an entire uevent msg
  73. func (c *UEventConn) ReadUEvent() (*UEvent, error) {
  74. msg, err := c.ReadMsg()
  75. if err != nil {
  76. return nil, err
  77. }
  78. return ParseUEvent(msg)
  79. }
  80. // Monitor run in background a worker to read netlink msg in loop and notify
  81. // when msg receive inside a queue using channel.
  82. // To be notified with only relevant message, use Matcher.
  83. func (c *UEventConn) Monitor(queue chan UEvent, errors chan error, matcher Matcher) (stop func(stopTimeout time.Duration) (ok bool)) {
  84. if matcher != nil {
  85. if err := matcher.Compile(); err != nil {
  86. errors <- fmt.Errorf("Wrong matcher, err: %v", err)
  87. return func(time.Duration) bool {
  88. return true
  89. }
  90. }
  91. }
  92. quitting := make(chan struct{})
  93. quit := make(chan struct{})
  94. readableOrStop, stop1, err := RawSockStopper(c.Fd)
  95. if err != nil {
  96. errors <- fmt.Errorf("Internal error: %v", err)
  97. return func(time.Duration) bool {
  98. return true
  99. }
  100. }
  101. // c.Fd is set to non-blocking at this point
  102. stop = func(stopTimeout time.Duration) bool {
  103. close(quitting)
  104. stop1()
  105. select {
  106. case <-quit:
  107. return true
  108. case <-time.After(stopTimeout):
  109. }
  110. return false
  111. }
  112. go func() {
  113. EventReading:
  114. for {
  115. _, err := readableOrStop()
  116. if err != nil {
  117. errors <- fmt.Errorf("Internal error: %v", err)
  118. return
  119. }
  120. select {
  121. case <-quitting:
  122. close(quit)
  123. return
  124. default:
  125. uevent, err := c.ReadUEvent()
  126. // underlying file descriptor is
  127. // non-blocking here, be paranoid if
  128. // for some reason we get here after
  129. // readableOrStop but the read would
  130. // block anyway
  131. if errno, ok := err.(syscall.Errno); ok && errno.Temporary() {
  132. continue EventReading
  133. }
  134. if err != nil {
  135. errors <- fmt.Errorf("Unable to parse uevent, err: %v", err)
  136. continue
  137. }
  138. if matcher != nil {
  139. if !matcher.Evaluate(*uevent) {
  140. continue // Drop uevent if not match
  141. }
  142. }
  143. queue <- *uevent
  144. }
  145. }
  146. }()
  147. return stop
  148. }