control_unix.go 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154
  1. // Copyright 2013 The Go Authors. All rights reserved.
  2. // Use of this source code is governed by a BSD-style
  3. // license that can be found in the LICENSE file.
  4. // +build darwin dragonfly freebsd linux netbsd openbsd solaris
  5. package ipv6
  6. import (
  7. "os"
  8. "syscall"
  9. "golang.org/x/net/internal/iana"
  10. "golang.org/x/net/internal/socket"
  11. )
  12. func setControlMessage(c *socket.Conn, opt *rawOpt, cf ControlFlags, on bool) error {
  13. opt.Lock()
  14. defer opt.Unlock()
  15. if so, ok := sockOpts[ssoReceiveTrafficClass]; ok && cf&FlagTrafficClass != 0 {
  16. if err := so.SetInt(c, boolint(on)); err != nil {
  17. return err
  18. }
  19. if on {
  20. opt.set(FlagTrafficClass)
  21. } else {
  22. opt.clear(FlagTrafficClass)
  23. }
  24. }
  25. if so, ok := sockOpts[ssoReceiveHopLimit]; ok && cf&FlagHopLimit != 0 {
  26. if err := so.SetInt(c, boolint(on)); err != nil {
  27. return err
  28. }
  29. if on {
  30. opt.set(FlagHopLimit)
  31. } else {
  32. opt.clear(FlagHopLimit)
  33. }
  34. }
  35. if so, ok := sockOpts[ssoReceivePacketInfo]; ok && cf&flagPacketInfo != 0 {
  36. if err := so.SetInt(c, boolint(on)); err != nil {
  37. return err
  38. }
  39. if on {
  40. opt.set(cf & flagPacketInfo)
  41. } else {
  42. opt.clear(cf & flagPacketInfo)
  43. }
  44. }
  45. if so, ok := sockOpts[ssoReceivePathMTU]; ok && cf&FlagPathMTU != 0 {
  46. if err := so.SetInt(c, boolint(on)); err != nil {
  47. return err
  48. }
  49. if on {
  50. opt.set(FlagPathMTU)
  51. } else {
  52. opt.clear(FlagPathMTU)
  53. }
  54. }
  55. return nil
  56. }
  57. func newControlMessage(opt *rawOpt) (oob []byte) {
  58. opt.RLock()
  59. var l int
  60. if opt.isset(FlagTrafficClass) && ctlOpts[ctlTrafficClass].name > 0 {
  61. l += syscall.CmsgSpace(ctlOpts[ctlTrafficClass].length)
  62. }
  63. if opt.isset(FlagHopLimit) && ctlOpts[ctlHopLimit].name > 0 {
  64. l += syscall.CmsgSpace(ctlOpts[ctlHopLimit].length)
  65. }
  66. if opt.isset(flagPacketInfo) && ctlOpts[ctlPacketInfo].name > 0 {
  67. l += syscall.CmsgSpace(ctlOpts[ctlPacketInfo].length)
  68. }
  69. if opt.isset(FlagPathMTU) && ctlOpts[ctlPathMTU].name > 0 {
  70. l += syscall.CmsgSpace(ctlOpts[ctlPathMTU].length)
  71. }
  72. if l > 0 {
  73. oob = make([]byte, l)
  74. }
  75. opt.RUnlock()
  76. return
  77. }
  78. func parseControlMessage(b []byte) (*ControlMessage, error) {
  79. if len(b) == 0 {
  80. return nil, nil
  81. }
  82. cmsgs, err := syscall.ParseSocketControlMessage(b)
  83. if err != nil {
  84. return nil, os.NewSyscallError("parse socket control message", err)
  85. }
  86. cm := &ControlMessage{}
  87. for _, m := range cmsgs {
  88. if m.Header.Level != iana.ProtocolIPv6 {
  89. continue
  90. }
  91. switch int(m.Header.Type) {
  92. case ctlOpts[ctlTrafficClass].name:
  93. ctlOpts[ctlTrafficClass].parse(cm, m.Data[:])
  94. case ctlOpts[ctlHopLimit].name:
  95. ctlOpts[ctlHopLimit].parse(cm, m.Data[:])
  96. case ctlOpts[ctlPacketInfo].name:
  97. ctlOpts[ctlPacketInfo].parse(cm, m.Data[:])
  98. case ctlOpts[ctlPathMTU].name:
  99. ctlOpts[ctlPathMTU].parse(cm, m.Data[:])
  100. }
  101. }
  102. return cm, nil
  103. }
  104. func marshalControlMessage(cm *ControlMessage) (oob []byte) {
  105. if cm == nil {
  106. return
  107. }
  108. var l int
  109. tclass := false
  110. if ctlOpts[ctlTrafficClass].name > 0 && cm.TrafficClass > 0 {
  111. tclass = true
  112. l += syscall.CmsgSpace(ctlOpts[ctlTrafficClass].length)
  113. }
  114. hoplimit := false
  115. if ctlOpts[ctlHopLimit].name > 0 && cm.HopLimit > 0 {
  116. hoplimit = true
  117. l += syscall.CmsgSpace(ctlOpts[ctlHopLimit].length)
  118. }
  119. pktinfo := false
  120. if ctlOpts[ctlPacketInfo].name > 0 && (cm.Src.To16() != nil && cm.Src.To4() == nil || cm.IfIndex > 0) {
  121. pktinfo = true
  122. l += syscall.CmsgSpace(ctlOpts[ctlPacketInfo].length)
  123. }
  124. nexthop := false
  125. if ctlOpts[ctlNextHop].name > 0 && cm.NextHop.To16() != nil && cm.NextHop.To4() == nil {
  126. nexthop = true
  127. l += syscall.CmsgSpace(ctlOpts[ctlNextHop].length)
  128. }
  129. if l > 0 {
  130. oob = make([]byte, l)
  131. b := oob
  132. if tclass {
  133. b = ctlOpts[ctlTrafficClass].marshal(b, cm)
  134. }
  135. if hoplimit {
  136. b = ctlOpts[ctlHopLimit].marshal(b, cm)
  137. }
  138. if pktinfo {
  139. b = ctlOpts[ctlPacketInfo].marshal(b, cm)
  140. }
  141. if nexthop {
  142. b = ctlOpts[ctlNextHop].marshal(b, cm)
  143. }
  144. }
  145. return
  146. }