tun_darwin.go 1.4 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859
  1. package nebula
  2. import (
  3. "fmt"
  4. "net"
  5. "os/exec"
  6. "strconv"
  7. "github.com/songgao/water"
  8. )
  9. type Tun struct {
  10. Device string
  11. Cidr *net.IPNet
  12. MTU int
  13. *water.Interface
  14. }
  15. func newTun(deviceName string, cidr *net.IPNet, defaultMTU int, routes []route, txQueueLen int) (ifce *Tun, err error) {
  16. if len(routes) > 0 {
  17. return nil, fmt.Errorf("Route MTU not supported in Darwin")
  18. }
  19. // NOTE: You cannot set the deviceName under Darwin, so you must check tun.Device after calling .Activate()
  20. return &Tun{
  21. Cidr: cidr,
  22. MTU: defaultMTU,
  23. }, nil
  24. }
  25. func (c *Tun) Activate() error {
  26. var err error
  27. c.Interface, err = water.New(water.Config{
  28. DeviceType: water.TUN,
  29. })
  30. if err != nil {
  31. return fmt.Errorf("Activate failed: %v", err)
  32. }
  33. c.Device = c.Interface.Name()
  34. // TODO use syscalls instead of exec.Command
  35. if err = exec.Command("ifconfig", c.Device, c.Cidr.String(), c.Cidr.IP.String()).Run(); err != nil {
  36. return fmt.Errorf("failed to run 'ifconfig': %s", err)
  37. }
  38. if err = exec.Command("route", "-n", "add", "-net", c.Cidr.String(), "-interface", c.Device).Run(); err != nil {
  39. return fmt.Errorf("failed to run 'route add': %s", err)
  40. }
  41. if err = exec.Command("ifconfig", c.Device, "mtu", strconv.Itoa(c.MTU)).Run(); err != nil {
  42. return fmt.Errorf("failed to run 'ifconfig': %s", err)
  43. }
  44. return nil
  45. }
  46. func (c *Tun) WriteRaw(b []byte) error {
  47. _, err := c.Write(b)
  48. return err
  49. }