tun_linux.go 2.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120
  1. //go:build linux && !android
  2. package tun
  3. import (
  4. "github.com/vishvananda/netlink"
  5. "golang.org/x/sys/unix"
  6. "gvisor.dev/gvisor/pkg/tcpip/link/fdbased"
  7. "gvisor.dev/gvisor/pkg/tcpip/stack"
  8. )
  9. // LinuxTun is an object that handles tun network interface on linux
  10. // current version is heavily stripped to do nothing more,
  11. // then create a network interface, to be provided as file descriptor to gVisor ip stack
  12. type LinuxTun struct {
  13. tunFd int
  14. tunLink netlink.Link
  15. options TunOptions
  16. }
  17. // LinuxTun implements Tun
  18. var _ Tun = (*LinuxTun)(nil)
  19. // LinuxTun implements GVisorTun
  20. var _ GVisorTun = (*LinuxTun)(nil)
  21. // NewTun builds new tun interface handler (linux specific)
  22. func NewTun(options TunOptions) (Tun, error) {
  23. tunFd, err := open(options.Name)
  24. if err != nil {
  25. return nil, err
  26. }
  27. tunLink, err := setup(options.Name, int(options.MTU))
  28. if err != nil {
  29. _ = unix.Close(tunFd)
  30. return nil, err
  31. }
  32. linuxTun := &LinuxTun{
  33. tunFd: tunFd,
  34. tunLink: tunLink,
  35. options: options,
  36. }
  37. return linuxTun, nil
  38. }
  39. // open the file that implements tun interface in the OS
  40. func open(name string) (int, error) {
  41. fd, err := unix.Open("/dev/net/tun", unix.O_RDWR, 0)
  42. if err != nil {
  43. return -1, err
  44. }
  45. ifr, err := unix.NewIfreq(name)
  46. if err != nil {
  47. _ = unix.Close(fd)
  48. return 0, err
  49. }
  50. flags := unix.IFF_TUN | unix.IFF_NO_PI
  51. ifr.SetUint16(uint16(flags))
  52. err = unix.IoctlIfreq(fd, unix.TUNSETIFF, ifr)
  53. if err != nil {
  54. _ = unix.Close(fd)
  55. return 0, err
  56. }
  57. err = unix.SetNonblock(fd, true)
  58. if err != nil {
  59. _ = unix.Close(fd)
  60. return 0, err
  61. }
  62. return fd, nil
  63. }
  64. // setup the interface through netlink socket
  65. func setup(name string, MTU int) (netlink.Link, error) {
  66. tunLink, err := netlink.LinkByName(name)
  67. if err != nil {
  68. return nil, err
  69. }
  70. err = netlink.LinkSetMTU(tunLink, MTU)
  71. if err != nil {
  72. _ = netlink.LinkSetDown(tunLink)
  73. return nil, err
  74. }
  75. return tunLink, nil
  76. }
  77. // Start is called by handler to bring tun interface to life
  78. func (t *LinuxTun) Start() error {
  79. err := netlink.LinkSetUp(t.tunLink)
  80. if err != nil {
  81. return err
  82. }
  83. return nil
  84. }
  85. // Close is called to shut down the tun interface
  86. func (t *LinuxTun) Close() error {
  87. _ = netlink.LinkSetDown(t.tunLink)
  88. _ = unix.Close(t.tunFd)
  89. return nil
  90. }
  91. // newEndpoint builds new gVisor stack.LinkEndpoint from the tun interface file descriptor
  92. func (t *LinuxTun) newEndpoint() (stack.LinkEndpoint, error) {
  93. return fdbased.New(&fdbased.Options{
  94. FDs: []int{t.tunFd},
  95. MTU: t.options.MTU,
  96. RXChecksumOffload: true,
  97. })
  98. }