interface_windows.go 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177
  1. // Copyright 2011 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. package net
  5. import (
  6. "internal/syscall/windows"
  7. "os"
  8. "syscall"
  9. "unsafe"
  10. )
  11. // adapterAddresses returns a list of IP adapter and address
  12. // structures. The structure contains an IP adapter and flattened
  13. // multiple IP addresses including unicast, anycast and multicast
  14. // addresses.
  15. func adapterAddresses() ([]*windows.IpAdapterAddresses, error) {
  16. var b []byte
  17. l := uint32(15000) // recommended initial size
  18. for {
  19. b = make([]byte, l)
  20. err := windows.GetAdaptersAddresses(syscall.AF_UNSPEC, windows.GAA_FLAG_INCLUDE_PREFIX, 0, (*windows.IpAdapterAddresses)(unsafe.Pointer(&b[0])), &l)
  21. if err == nil {
  22. if l == 0 {
  23. return nil, nil
  24. }
  25. break
  26. }
  27. if err.(syscall.Errno) != syscall.ERROR_BUFFER_OVERFLOW {
  28. return nil, os.NewSyscallError("getadaptersaddresses", err)
  29. }
  30. if l <= uint32(len(b)) {
  31. return nil, os.NewSyscallError("getadaptersaddresses", err)
  32. }
  33. }
  34. var aas []*windows.IpAdapterAddresses
  35. for aa := (*windows.IpAdapterAddresses)(unsafe.Pointer(&b[0])); aa != nil; aa = aa.Next {
  36. aas = append(aas, aa)
  37. }
  38. return aas, nil
  39. }
  40. // If the ifindex is zero, interfaceTable returns mappings of all
  41. // network interfaces. Otherwise it returns a mapping of a specific
  42. // interface.
  43. func interfaceTable(ifindex int) ([]Interface, error) {
  44. aas, err := adapterAddresses()
  45. if err != nil {
  46. return nil, err
  47. }
  48. var ift []Interface
  49. for _, aa := range aas {
  50. index := aa.IfIndex
  51. if index == 0 { // ipv6IfIndex is a substitute for ifIndex
  52. index = aa.Ipv6IfIndex
  53. }
  54. if ifindex == 0 || ifindex == int(index) {
  55. ifi := Interface{
  56. Index: int(index),
  57. Name: windows.UTF16PtrToString(aa.FriendlyName),
  58. }
  59. if aa.OperStatus == windows.IfOperStatusUp {
  60. ifi.Flags |= FlagUp
  61. }
  62. // For now we need to infer link-layer service
  63. // capabilities from media types.
  64. // TODO: use MIB_IF_ROW2.AccessType now that we no longer support
  65. // Windows XP.
  66. switch aa.IfType {
  67. case windows.IF_TYPE_ETHERNET_CSMACD, windows.IF_TYPE_ISO88025_TOKENRING, windows.IF_TYPE_IEEE80211, windows.IF_TYPE_IEEE1394:
  68. ifi.Flags |= FlagBroadcast | FlagMulticast
  69. case windows.IF_TYPE_PPP, windows.IF_TYPE_TUNNEL:
  70. ifi.Flags |= FlagPointToPoint | FlagMulticast
  71. case windows.IF_TYPE_SOFTWARE_LOOPBACK:
  72. ifi.Flags |= FlagLoopback | FlagMulticast
  73. case windows.IF_TYPE_ATM:
  74. ifi.Flags |= FlagBroadcast | FlagPointToPoint | FlagMulticast // assume all services available; LANE, point-to-point and point-to-multipoint
  75. }
  76. if aa.Mtu == 0xffffffff {
  77. ifi.MTU = -1
  78. } else {
  79. ifi.MTU = int(aa.Mtu)
  80. }
  81. if aa.PhysicalAddressLength > 0 {
  82. ifi.HardwareAddr = make(HardwareAddr, aa.PhysicalAddressLength)
  83. copy(ifi.HardwareAddr, aa.PhysicalAddress[:])
  84. }
  85. ift = append(ift, ifi)
  86. if ifindex == ifi.Index {
  87. break
  88. }
  89. }
  90. }
  91. return ift, nil
  92. }
  93. // If the ifi is nil, interfaceAddrTable returns addresses for all
  94. // network interfaces. Otherwise it returns addresses for a specific
  95. // interface.
  96. func interfaceAddrTable(ifi *Interface) ([]Addr, error) {
  97. aas, err := adapterAddresses()
  98. if err != nil {
  99. return nil, err
  100. }
  101. var ifat []Addr
  102. for _, aa := range aas {
  103. index := aa.IfIndex
  104. if index == 0 { // ipv6IfIndex is a substitute for ifIndex
  105. index = aa.Ipv6IfIndex
  106. }
  107. if ifi == nil || ifi.Index == int(index) {
  108. for puni := aa.FirstUnicastAddress; puni != nil; puni = puni.Next {
  109. sa, err := puni.Address.Sockaddr.Sockaddr()
  110. if err != nil {
  111. return nil, os.NewSyscallError("sockaddr", err)
  112. }
  113. switch sa := sa.(type) {
  114. case *syscall.SockaddrInet4:
  115. ifat = append(ifat, &IPNet{IP: IPv4(sa.Addr[0], sa.Addr[1], sa.Addr[2], sa.Addr[3]), Mask: CIDRMask(int(puni.OnLinkPrefixLength), 8*IPv4len)})
  116. case *syscall.SockaddrInet6:
  117. ifa := &IPNet{IP: make(IP, IPv6len), Mask: CIDRMask(int(puni.OnLinkPrefixLength), 8*IPv6len)}
  118. copy(ifa.IP, sa.Addr[:])
  119. ifat = append(ifat, ifa)
  120. }
  121. }
  122. for pany := aa.FirstAnycastAddress; pany != nil; pany = pany.Next {
  123. sa, err := pany.Address.Sockaddr.Sockaddr()
  124. if err != nil {
  125. return nil, os.NewSyscallError("sockaddr", err)
  126. }
  127. switch sa := sa.(type) {
  128. case *syscall.SockaddrInet4:
  129. ifat = append(ifat, &IPAddr{IP: IPv4(sa.Addr[0], sa.Addr[1], sa.Addr[2], sa.Addr[3])})
  130. case *syscall.SockaddrInet6:
  131. ifa := &IPAddr{IP: make(IP, IPv6len)}
  132. copy(ifa.IP, sa.Addr[:])
  133. ifat = append(ifat, ifa)
  134. }
  135. }
  136. }
  137. }
  138. return ifat, nil
  139. }
  140. // interfaceMulticastAddrTable returns addresses for a specific
  141. // interface.
  142. func interfaceMulticastAddrTable(ifi *Interface) ([]Addr, error) {
  143. aas, err := adapterAddresses()
  144. if err != nil {
  145. return nil, err
  146. }
  147. var ifat []Addr
  148. for _, aa := range aas {
  149. index := aa.IfIndex
  150. if index == 0 { // ipv6IfIndex is a substitute for ifIndex
  151. index = aa.Ipv6IfIndex
  152. }
  153. if ifi == nil || ifi.Index == int(index) {
  154. for pmul := aa.FirstMulticastAddress; pmul != nil; pmul = pmul.Next {
  155. sa, err := pmul.Address.Sockaddr.Sockaddr()
  156. if err != nil {
  157. return nil, os.NewSyscallError("sockaddr", err)
  158. }
  159. switch sa := sa.(type) {
  160. case *syscall.SockaddrInet4:
  161. ifat = append(ifat, &IPAddr{IP: IPv4(sa.Addr[0], sa.Addr[1], sa.Addr[2], sa.Addr[3])})
  162. case *syscall.SockaddrInet6:
  163. ifa := &IPAddr{IP: make(IP, IPv6len)}
  164. copy(ifa.IP, sa.Addr[:])
  165. ifat = append(ifat, ifa)
  166. }
  167. }
  168. }
  169. }
  170. return ifat, nil
  171. }