interface_android.go 7.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324
  1. package anet
  2. import (
  3. "bytes"
  4. "errors"
  5. "net"
  6. "os"
  7. "syscall"
  8. "unsafe"
  9. )
  10. const (
  11. android11 = 11
  12. )
  13. var (
  14. androidVersion uint
  15. errNoSuchInterface = errors.New("no such network interface")
  16. errInvalidInterface = errors.New("invalid network interface")
  17. )
  18. type ifReq [40]byte
  19. // Interfaces returns a list of the system's network interfaces.
  20. func Interfaces() ([]net.Interface, error) {
  21. if androidVersion > android11 {
  22. return net.Interfaces()
  23. }
  24. ift, err := interfaceTable(0)
  25. if err != nil {
  26. return nil, &net.OpError{Op: "route", Net: "ip+net", Source: nil, Addr: nil, Err: err}
  27. }
  28. // TODO: zoneCache implementation
  29. return ift, nil
  30. }
  31. // InterfaceAddrs returns a list of the system's unicast interface
  32. // addresses.
  33. //
  34. // The returned list does not identify the associated interface; use
  35. // Interfaces and Interface.Addrs for more detail.
  36. func InterfaceAddrs() ([]net.Addr, error) {
  37. if androidVersion > android11 {
  38. return net.InterfaceAddrs()
  39. }
  40. ifat, err := interfaceAddrTable(nil)
  41. if err != nil {
  42. err = &net.OpError{Op: "route", Net: "ip+net", Source: nil, Addr: nil, Err: err}
  43. }
  44. return ifat, err
  45. }
  46. // InterfaceAddrsByInterface returns a list of the system's unicast
  47. // interface addresses by specific interface.
  48. func InterfaceAddrsByInterface(ifi *net.Interface) ([]net.Addr, error) {
  49. if ifi == nil {
  50. return nil, &net.OpError{Op: "route", Net: "ip+net", Source: nil, Addr: nil, Err: errInvalidInterface}
  51. }
  52. if androidVersion > android11 {
  53. return ifi.Addrs()
  54. }
  55. ifat, err := interfaceAddrTable(ifi)
  56. if err != nil {
  57. err = &net.OpError{Op: "route", Net: "ip+net", Source: nil, Addr: nil, Err: err}
  58. }
  59. return ifat, err
  60. }
  61. // SetAndroidVersion set the Android environment in which the program runs.
  62. // The Android system version number can be obtained through
  63. // `android.os.Build.VERSION.RELEASE` of the Android framework.
  64. func SetAndroidVersion(version uint) {
  65. androidVersion = version
  66. }
  67. // If the ifindex is zero, interfaceTable returns mappings of all
  68. // network interfaces. Otherwise it returns a mapping of a specific
  69. // interface.
  70. func interfaceTable(ifindex int) ([]net.Interface, error) {
  71. tab, err := NetlinkRIB(syscall.RTM_GETADDR, syscall.AF_UNSPEC)
  72. if err != nil {
  73. return nil, os.NewSyscallError("netlinkrib", err)
  74. }
  75. msgs, err := syscall.ParseNetlinkMessage(tab)
  76. if err != nil {
  77. return nil, os.NewSyscallError("parsenetlinkmessage", err)
  78. }
  79. var ift []net.Interface
  80. im := make(map[uint32]struct{})
  81. loop:
  82. for _, m := range msgs {
  83. switch m.Header.Type {
  84. case syscall.NLMSG_DONE:
  85. break loop
  86. case syscall.RTM_NEWADDR:
  87. ifam := (*syscall.IfAddrmsg)(unsafe.Pointer(&m.Data[0]))
  88. if _, ok := im[ifam.Index]; ok {
  89. continue
  90. } else {
  91. im[ifam.Index] = struct{}{}
  92. }
  93. if ifindex == 0 || ifindex == int(ifam.Index) {
  94. ifi := newLink(ifam)
  95. if ifi != nil {
  96. ift = append(ift, *ifi)
  97. }
  98. if ifindex == int(ifam.Index) {
  99. break loop
  100. }
  101. }
  102. }
  103. }
  104. return ift, nil
  105. }
  106. func newLink(ifam *syscall.IfAddrmsg) *net.Interface {
  107. ift := &net.Interface{Index: int(ifam.Index)}
  108. name, err := indexToName(ifam.Index)
  109. if err != nil {
  110. return nil
  111. }
  112. ift.Name = name
  113. mtu, err := nameToMTU(name)
  114. if err != nil {
  115. return nil
  116. }
  117. ift.MTU = mtu
  118. flags, err := nameToFlags(name)
  119. if err != nil {
  120. return nil
  121. }
  122. ift.Flags = flags
  123. return ift
  124. }
  125. func linkFlags(rawFlags uint32) net.Flags {
  126. var f net.Flags
  127. if rawFlags&syscall.IFF_UP != 0 {
  128. f |= net.FlagUp
  129. }
  130. if rawFlags&syscall.IFF_RUNNING != 0 {
  131. f |= net.FlagRunning
  132. }
  133. if rawFlags&syscall.IFF_BROADCAST != 0 {
  134. f |= net.FlagBroadcast
  135. }
  136. if rawFlags&syscall.IFF_LOOPBACK != 0 {
  137. f |= net.FlagLoopback
  138. }
  139. if rawFlags&syscall.IFF_POINTOPOINT != 0 {
  140. f |= net.FlagPointToPoint
  141. }
  142. if rawFlags&syscall.IFF_MULTICAST != 0 {
  143. f |= net.FlagMulticast
  144. }
  145. return f
  146. }
  147. // If the ifi is nil, interfaceAddrTable returns addresses for all
  148. // network interfaces. Otherwise it returns addresses for a specific
  149. // interface.
  150. func interfaceAddrTable(ifi *net.Interface) ([]net.Addr, error) {
  151. tab, err := NetlinkRIB(syscall.RTM_GETADDR, syscall.AF_UNSPEC)
  152. if err != nil {
  153. return nil, os.NewSyscallError("netlinkrib", err)
  154. }
  155. msgs, err := syscall.ParseNetlinkMessage(tab)
  156. if err != nil {
  157. return nil, os.NewSyscallError("parsenetlinkmessage", err)
  158. }
  159. var ift []net.Interface
  160. if ifi == nil {
  161. var err error
  162. ift, err = interfaceTable(0)
  163. if err != nil {
  164. return nil, err
  165. }
  166. }
  167. ifat, err := addrTable(ift, ifi, msgs)
  168. if err != nil {
  169. return nil, err
  170. }
  171. return ifat, nil
  172. }
  173. func addrTable(ift []net.Interface, ifi *net.Interface, msgs []syscall.NetlinkMessage) ([]net.Addr, error) {
  174. var ifat []net.Addr
  175. loop:
  176. for _, m := range msgs {
  177. switch m.Header.Type {
  178. case syscall.NLMSG_DONE:
  179. break loop
  180. case syscall.RTM_NEWADDR:
  181. ifam := (*syscall.IfAddrmsg)(unsafe.Pointer(&m.Data[0]))
  182. if len(ift) != 0 || ifi.Index == int(ifam.Index) {
  183. attrs, err := syscall.ParseNetlinkRouteAttr(&m)
  184. if err != nil {
  185. return nil, os.NewSyscallError("parsenetlinkrouteattr", err)
  186. }
  187. ifa := newAddr(ifam, attrs)
  188. if ifa != nil {
  189. ifat = append(ifat, ifa)
  190. }
  191. }
  192. }
  193. }
  194. return ifat, nil
  195. }
  196. func newAddr(ifam *syscall.IfAddrmsg, attrs []syscall.NetlinkRouteAttr) net.Addr {
  197. var ipPointToPoint bool
  198. // Seems like we need to make sure whether the IP interface
  199. // stack consists of IP point-to-point numbered or unnumbered
  200. // addressing.
  201. for _, a := range attrs {
  202. if a.Attr.Type == syscall.IFA_LOCAL {
  203. ipPointToPoint = true
  204. break
  205. }
  206. }
  207. for _, a := range attrs {
  208. if ipPointToPoint && a.Attr.Type == syscall.IFA_ADDRESS {
  209. continue
  210. }
  211. switch ifam.Family {
  212. case syscall.AF_INET:
  213. return &net.IPNet{IP: net.IPv4(a.Value[0], a.Value[1], a.Value[2], a.Value[3]), Mask: net.CIDRMask(int(ifam.Prefixlen), 8*net.IPv4len)}
  214. case syscall.AF_INET6:
  215. ifa := &net.IPNet{IP: make(net.IP, net.IPv6len), Mask: net.CIDRMask(int(ifam.Prefixlen), 8*net.IPv6len)}
  216. copy(ifa.IP, a.Value[:])
  217. return ifa
  218. }
  219. }
  220. return nil
  221. }
  222. func interfaceByIndex(ift []net.Interface, index int) (*net.Interface, error) {
  223. for _, ifi := range ift {
  224. if index == ifi.Index {
  225. return &ifi, nil
  226. }
  227. }
  228. return nil, errNoSuchInterface
  229. }
  230. func ioctl(fd int, req uint, arg unsafe.Pointer) error {
  231. _, _, e1 := syscall.Syscall(syscall.SYS_IOCTL, uintptr(fd), uintptr(req), uintptr(arg))
  232. if e1 != 0 {
  233. return e1
  234. }
  235. return nil
  236. }
  237. func indexToName(index uint32) (string, error) {
  238. fd, err := syscall.Socket(syscall.AF_INET, syscall.SOCK_DGRAM|syscall.SOCK_CLOEXEC, 0)
  239. if err != nil {
  240. return "", err
  241. }
  242. defer syscall.Close(fd)
  243. var ifr ifReq
  244. *(*uint32)(unsafe.Pointer(&ifr[syscall.IFNAMSIZ])) = index
  245. err = ioctl(fd, syscall.SIOCGIFNAME, unsafe.Pointer(&ifr[0]))
  246. if err != nil {
  247. return "", err
  248. }
  249. return string(bytes.Trim(ifr[:syscall.IFNAMSIZ], "\x00")), nil
  250. }
  251. func nameToMTU(name string) (int, error) {
  252. // Leave room for terminating NULL byte.
  253. if len(name) >= syscall.IFNAMSIZ {
  254. return -1, syscall.EINVAL
  255. }
  256. fd, err := syscall.Socket(syscall.AF_INET, syscall.SOCK_DGRAM|syscall.SOCK_CLOEXEC, 0)
  257. if err != nil {
  258. return -1, err
  259. }
  260. defer syscall.Close(fd)
  261. var ifr ifReq
  262. copy(ifr[:], name)
  263. err = ioctl(fd, syscall.SIOCGIFMTU, unsafe.Pointer(&ifr[0]))
  264. if err != nil {
  265. return -1, err
  266. }
  267. return int(*(*int32)(unsafe.Pointer(&ifr[syscall.IFNAMSIZ]))), nil
  268. }
  269. func nameToFlags(name string) (net.Flags, error) {
  270. // Leave room for terminating NULL byte.
  271. if len(name) >= syscall.IFNAMSIZ {
  272. return 0, syscall.EINVAL
  273. }
  274. fd, err := syscall.Socket(syscall.AF_INET, syscall.SOCK_DGRAM|syscall.SOCK_CLOEXEC, 0)
  275. if err != nil {
  276. return 0, err
  277. }
  278. defer syscall.Close(fd)
  279. var ifr ifReq
  280. copy(ifr[:], name)
  281. err = ioctl(fd, syscall.SIOCGIFFLAGS, unsafe.Pointer(&ifr[0]))
  282. if err != nil {
  283. return 0, err
  284. }
  285. return linkFlags(*(*uint32)(unsafe.Pointer(&ifr[syscall.IFNAMSIZ]))), nil
  286. }