fakeudp_linux.go 2.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172
  1. // +build linux
  2. package dokodemo
  3. import (
  4. "fmt"
  5. "net"
  6. "os"
  7. "syscall"
  8. )
  9. func FakeUDP(addr *net.UDPAddr, mark int) (net.PacketConn, error) {
  10. localSocketAddress, af, err := udpAddrToSocketAddr(addr)
  11. if err != nil {
  12. return nil, &net.OpError{Op: "fake", Err: fmt.Errorf("build local socket address: %s", err)}
  13. }
  14. fileDescriptor, err := syscall.Socket(af, syscall.SOCK_DGRAM, 0)
  15. if err != nil {
  16. return nil, &net.OpError{Op: "fake", Err: fmt.Errorf("socket open: %s", err)}
  17. }
  18. if mark != 0 {
  19. if err = syscall.SetsockoptInt(fileDescriptor, syscall.SOL_SOCKET, syscall.SO_MARK, mark); err != nil {
  20. syscall.Close(fileDescriptor)
  21. return nil, &net.OpError{Op: "fake", Err: fmt.Errorf("set socket option: SO_MARK: %s", err)}
  22. }
  23. }
  24. if err = syscall.SetsockoptInt(fileDescriptor, syscall.SOL_SOCKET, syscall.SO_REUSEADDR, 1); err != nil {
  25. syscall.Close(fileDescriptor)
  26. return nil, &net.OpError{Op: "fake", Err: fmt.Errorf("set socket option: SO_REUSEADDR: %s", err)}
  27. }
  28. if err = syscall.SetsockoptInt(fileDescriptor, syscall.SOL_IP, syscall.IP_TRANSPARENT, 1); err != nil {
  29. syscall.Close(fileDescriptor)
  30. return nil, &net.OpError{Op: "fake", Err: fmt.Errorf("set socket option: IP_TRANSPARENT: %s", err)}
  31. }
  32. if err = syscall.Bind(fileDescriptor, localSocketAddress); err != nil {
  33. syscall.Close(fileDescriptor)
  34. return nil, &net.OpError{Op: "fake", Err: fmt.Errorf("socket bind: %s", err)}
  35. }
  36. fdFile := os.NewFile(uintptr(fileDescriptor), fmt.Sprintf("net-udp-fake-%s", addr.String()))
  37. defer fdFile.Close()
  38. packetConn, err := net.FilePacketConn(fdFile)
  39. if err != nil {
  40. syscall.Close(fileDescriptor)
  41. return nil, &net.OpError{Op: "fake", Err: fmt.Errorf("convert file descriptor to connection: %s", err)}
  42. }
  43. return packetConn, nil
  44. }
  45. func udpAddrToSocketAddr(addr *net.UDPAddr) (syscall.Sockaddr, int, error) {
  46. switch {
  47. case addr.IP.To4() != nil:
  48. ip := [4]byte{}
  49. copy(ip[:], addr.IP.To4())
  50. return &syscall.SockaddrInet4{Addr: ip, Port: addr.Port}, syscall.AF_INET, nil
  51. default:
  52. ip := [16]byte{}
  53. copy(ip[:], addr.IP.To16())
  54. return &syscall.SockaddrInet6{Addr: ip, Port: addr.Port, ZoneId: 0}, syscall.AF_INET6, nil
  55. }
  56. }