fakeudp_linux.go 2.3 KB

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