tun.go 981 B

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950
  1. package ip
  2. import (
  3. "bytes"
  4. "fmt"
  5. "os"
  6. "syscall"
  7. "unsafe"
  8. )
  9. const (
  10. tunDevice = "/dev/net/tun"
  11. ifnameSize = 16
  12. )
  13. type ifreqFlags struct {
  14. IfrnName [ifnameSize]byte
  15. IfruFlags uint16
  16. }
  17. func ioctl(fd int, request, argp uintptr) error {
  18. _, _, errno := syscall.Syscall(syscall.SYS_IOCTL, uintptr(fd), request, argp)
  19. if errno != 0 {
  20. return fmt.Errorf("ioctl failed with '%s'", errno)
  21. }
  22. return nil
  23. }
  24. func fromZeroTerm(s []byte) string {
  25. return string(bytes.TrimRight(s, "\000"))
  26. }
  27. func OpenTun(name string) (*os.File, string, error) {
  28. tun, err := os.OpenFile(tunDevice, os.O_RDWR, 0)
  29. if err != nil {
  30. return nil, "", err
  31. }
  32. var ifr ifreqFlags
  33. copy(ifr.IfrnName[:len(ifr.IfrnName)-1], []byte(name+"\000"))
  34. ifr.IfruFlags = syscall.IFF_TUN | syscall.IFF_NO_PI
  35. err = ioctl(int(tun.Fd()), syscall.TUNSETIFF, uintptr(unsafe.Pointer(&ifr)))
  36. if err != nil {
  37. return nil, "", err
  38. }
  39. ifname := fromZeroTerm(ifr.IfrnName[:ifnameSize])
  40. return tun, ifname, nil
  41. }