tun_android.go 2.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899
  1. //go:build !e2e_testing
  2. // +build !e2e_testing
  3. package overlay
  4. import (
  5. "fmt"
  6. "io"
  7. "net/netip"
  8. "os"
  9. "sync/atomic"
  10. "github.com/gaissmai/bart"
  11. "github.com/sirupsen/logrus"
  12. "github.com/slackhq/nebula/config"
  13. "github.com/slackhq/nebula/util"
  14. )
  15. type tun struct {
  16. io.ReadWriteCloser
  17. fd int
  18. vpnNetworks []netip.Prefix
  19. Routes atomic.Pointer[[]Route]
  20. routeTree atomic.Pointer[bart.Table[netip.Addr]]
  21. l *logrus.Logger
  22. }
  23. func newTunFromFd(c *config.C, l *logrus.Logger, deviceFd int, vpnNetworks []netip.Prefix) (*tun, error) {
  24. // XXX Android returns an fd in non-blocking mode which is necessary for shutdown to work properly.
  25. // Be sure not to call file.Fd() as it will set the fd to blocking mode.
  26. file := os.NewFile(uintptr(deviceFd), "/dev/net/tun")
  27. t := &tun{
  28. ReadWriteCloser: file,
  29. fd: deviceFd,
  30. vpnNetworks: vpnNetworks,
  31. l: l,
  32. }
  33. err := t.reload(c, true)
  34. if err != nil {
  35. return nil, err
  36. }
  37. c.RegisterReloadCallback(func(c *config.C) {
  38. err := t.reload(c, false)
  39. if err != nil {
  40. util.LogWithContextIfNeeded("failed to reload tun device", err, t.l)
  41. }
  42. })
  43. return t, nil
  44. }
  45. func newTun(_ *config.C, _ *logrus.Logger, _ []netip.Prefix, _ bool) (*tun, error) {
  46. return nil, fmt.Errorf("newTun not supported in Android")
  47. }
  48. func (t *tun) RouteFor(ip netip.Addr) netip.Addr {
  49. r, _ := t.routeTree.Load().Lookup(ip)
  50. return r
  51. }
  52. func (t tun) Activate() error {
  53. return nil
  54. }
  55. func (t *tun) reload(c *config.C, initial bool) error {
  56. change, routes, err := getAllRoutesFromConfig(c, t.vpnNetworks, initial)
  57. if err != nil {
  58. return err
  59. }
  60. if !initial && !change {
  61. return nil
  62. }
  63. routeTree, err := makeRouteTree(t.l, routes, false)
  64. if err != nil {
  65. return err
  66. }
  67. // Teach nebula how to handle the routes
  68. t.Routes.Store(&routes)
  69. t.routeTree.Store(routeTree)
  70. return nil
  71. }
  72. func (t *tun) Networks() []netip.Prefix {
  73. return t.vpnNetworks
  74. }
  75. func (t *tun) Name() string {
  76. return "android"
  77. }
  78. func (t *tun) NewMultiQueueReader() (io.ReadWriteCloser, error) {
  79. return nil, fmt.Errorf("TODO: multiqueue not implemented for android")
  80. }