sockopt_posix.go 2.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142
  1. // Copyright 2009 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 windows
  5. package net
  6. import (
  7. "os"
  8. "syscall"
  9. )
  10. // Boolean to int.
  11. func boolint(b bool) int {
  12. if b {
  13. return 1
  14. }
  15. return 0
  16. }
  17. func ipv4AddrToInterface(ip IP) (*Interface, error) {
  18. ift, err := Interfaces()
  19. if err != nil {
  20. return nil, err
  21. }
  22. for _, ifi := range ift {
  23. ifat, err := ifi.Addrs()
  24. if err != nil {
  25. return nil, err
  26. }
  27. for _, ifa := range ifat {
  28. switch v := ifa.(type) {
  29. case *IPAddr:
  30. if ip.Equal(v.IP) {
  31. return &ifi, nil
  32. }
  33. case *IPNet:
  34. if ip.Equal(v.IP) {
  35. return &ifi, nil
  36. }
  37. }
  38. }
  39. }
  40. if ip.Equal(IPv4zero) {
  41. return nil, nil
  42. }
  43. return nil, errNoSuchInterface
  44. }
  45. func interfaceToIPv4Addr(ifi *Interface) (IP, error) {
  46. if ifi == nil {
  47. return IPv4zero, nil
  48. }
  49. ifat, err := ifi.Addrs()
  50. if err != nil {
  51. return nil, err
  52. }
  53. for _, ifa := range ifat {
  54. switch v := ifa.(type) {
  55. case *IPAddr:
  56. if v.IP.To4() != nil {
  57. return v.IP, nil
  58. }
  59. case *IPNet:
  60. if v.IP.To4() != nil {
  61. return v.IP, nil
  62. }
  63. }
  64. }
  65. return nil, errNoSuchInterface
  66. }
  67. func setIPv4MreqToInterface(mreq *syscall.IPMreq, ifi *Interface) error {
  68. if ifi == nil {
  69. return nil
  70. }
  71. ifat, err := ifi.Addrs()
  72. if err != nil {
  73. return err
  74. }
  75. for _, ifa := range ifat {
  76. switch v := ifa.(type) {
  77. case *IPAddr:
  78. if a := v.IP.To4(); a != nil {
  79. copy(mreq.Interface[:], a)
  80. goto done
  81. }
  82. case *IPNet:
  83. if a := v.IP.To4(); a != nil {
  84. copy(mreq.Interface[:], a)
  85. goto done
  86. }
  87. }
  88. }
  89. done:
  90. if bytesEqual(mreq.Multiaddr[:], IPv4zero.To4()) {
  91. return errNoSuchMulticastInterface
  92. }
  93. return nil
  94. }
  95. func setReadBuffer(fd *netFD, bytes int) error {
  96. if err := fd.incref(); err != nil {
  97. return err
  98. }
  99. defer fd.decref()
  100. return os.NewSyscallError("setsockopt", syscall.SetsockoptInt(fd.sysfd, syscall.SOL_SOCKET, syscall.SO_RCVBUF, bytes))
  101. }
  102. func setWriteBuffer(fd *netFD, bytes int) error {
  103. if err := fd.incref(); err != nil {
  104. return err
  105. }
  106. defer fd.decref()
  107. return os.NewSyscallError("setsockopt", syscall.SetsockoptInt(fd.sysfd, syscall.SOL_SOCKET, syscall.SO_SNDBUF, bytes))
  108. }
  109. func setKeepAlive(fd *netFD, keepalive bool) error {
  110. if err := fd.incref(); err != nil {
  111. return err
  112. }
  113. defer fd.decref()
  114. return os.NewSyscallError("setsockopt", syscall.SetsockoptInt(fd.sysfd, syscall.SOL_SOCKET, syscall.SO_KEEPALIVE, boolint(keepalive)))
  115. }
  116. func setLinger(fd *netFD, sec int) error {
  117. var l syscall.Linger
  118. if sec >= 0 {
  119. l.Onoff = 1
  120. l.Linger = int32(sec)
  121. } else {
  122. l.Onoff = 0
  123. l.Linger = 0
  124. }
  125. if err := fd.incref(); err != nil {
  126. return err
  127. }
  128. defer fd.decref()
  129. return os.NewSyscallError("setsockopt", syscall.SetsockoptLinger(fd.sysfd, syscall.SOL_SOCKET, syscall.SO_LINGER, &l))
  130. }