system_dialer.go 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184
  1. package internet
  2. import (
  3. "context"
  4. "syscall"
  5. "time"
  6. "github.com/v2fly/v2ray-core/v4/common/net"
  7. "github.com/v2fly/v2ray-core/v4/common/session"
  8. )
  9. var (
  10. effectiveSystemDialer SystemDialer = &DefaultSystemDialer{}
  11. )
  12. type SystemDialer interface {
  13. Dial(ctx context.Context, source net.Address, destination net.Destination, sockopt *SocketConfig) (net.Conn, error)
  14. }
  15. type DefaultSystemDialer struct {
  16. controllers []controller
  17. }
  18. func resolveSrcAddr(network net.Network, src net.Address) net.Addr {
  19. if src == nil || src == net.AnyIP {
  20. return nil
  21. }
  22. if network == net.Network_TCP {
  23. return &net.TCPAddr{
  24. IP: src.IP(),
  25. Port: 0,
  26. }
  27. }
  28. return &net.UDPAddr{
  29. IP: src.IP(),
  30. Port: 0,
  31. }
  32. }
  33. func hasBindAddr(sockopt *SocketConfig) bool {
  34. return sockopt != nil && len(sockopt.BindAddress) > 0 && sockopt.BindPort > 0
  35. }
  36. func (d *DefaultSystemDialer) Dial(ctx context.Context, src net.Address, dest net.Destination, sockopt *SocketConfig) (net.Conn, error) {
  37. if dest.Network == net.Network_UDP && !hasBindAddr(sockopt) {
  38. srcAddr := resolveSrcAddr(net.Network_UDP, src)
  39. if srcAddr == nil {
  40. srcAddr = &net.UDPAddr{
  41. IP: []byte{0, 0, 0, 0},
  42. Port: 0,
  43. }
  44. }
  45. packetConn, err := ListenSystemPacket(ctx, srcAddr, sockopt)
  46. if err != nil {
  47. return nil, err
  48. }
  49. destAddr, err := net.ResolveUDPAddr("udp", dest.NetAddr())
  50. if err != nil {
  51. return nil, err
  52. }
  53. return &packetConnWrapper{
  54. conn: packetConn,
  55. dest: destAddr,
  56. }, nil
  57. }
  58. dialer := &net.Dialer{
  59. Timeout: time.Second * 16,
  60. LocalAddr: resolveSrcAddr(dest.Network, src),
  61. }
  62. if sockopt != nil || len(d.controllers) > 0 {
  63. dialer.Control = func(network, address string, c syscall.RawConn) error {
  64. return c.Control(func(fd uintptr) {
  65. if sockopt != nil {
  66. if err := applyOutboundSocketOptions(network, address, fd, sockopt); err != nil {
  67. newError("failed to apply socket options").Base(err).WriteToLog(session.ExportIDToError(ctx))
  68. }
  69. if dest.Network == net.Network_UDP && hasBindAddr(sockopt) {
  70. if err := bindAddr(fd, sockopt.BindAddress, sockopt.BindPort); err != nil {
  71. newError("failed to bind source address to ", sockopt.BindAddress).Base(err).WriteToLog(session.ExportIDToError(ctx))
  72. }
  73. }
  74. }
  75. for _, ctl := range d.controllers {
  76. if err := ctl(network, address, fd); err != nil {
  77. newError("failed to apply external controller").Base(err).WriteToLog(session.ExportIDToError(ctx))
  78. }
  79. }
  80. })
  81. }
  82. }
  83. return dialer.DialContext(ctx, dest.Network.SystemString(), dest.NetAddr())
  84. }
  85. type packetConnWrapper struct {
  86. conn net.PacketConn
  87. dest net.Addr
  88. }
  89. func (c *packetConnWrapper) Close() error {
  90. return c.conn.Close()
  91. }
  92. func (c *packetConnWrapper) LocalAddr() net.Addr {
  93. return c.conn.LocalAddr()
  94. }
  95. func (c *packetConnWrapper) RemoteAddr() net.Addr {
  96. return c.dest
  97. }
  98. func (c *packetConnWrapper) Write(p []byte) (int, error) {
  99. return c.conn.WriteTo(p, c.dest)
  100. }
  101. func (c *packetConnWrapper) Read(p []byte) (int, error) {
  102. n, _, err := c.conn.ReadFrom(p)
  103. return n, err
  104. }
  105. func (c *packetConnWrapper) SetDeadline(t time.Time) error {
  106. return c.conn.SetDeadline(t)
  107. }
  108. func (c *packetConnWrapper) SetReadDeadline(t time.Time) error {
  109. return c.conn.SetReadDeadline(t)
  110. }
  111. func (c *packetConnWrapper) SetWriteDeadline(t time.Time) error {
  112. return c.conn.SetWriteDeadline(t)
  113. }
  114. type SystemDialerAdapter interface {
  115. Dial(network string, address string) (net.Conn, error)
  116. }
  117. type SimpleSystemDialer struct {
  118. adapter SystemDialerAdapter
  119. }
  120. func WithAdapter(dialer SystemDialerAdapter) SystemDialer {
  121. return &SimpleSystemDialer{
  122. adapter: dialer,
  123. }
  124. }
  125. func (v *SimpleSystemDialer) Dial(ctx context.Context, src net.Address, dest net.Destination, sockopt *SocketConfig) (net.Conn, error) {
  126. return v.adapter.Dial(dest.Network.SystemString(), dest.NetAddr())
  127. }
  128. // UseAlternativeSystemDialer replaces the current system dialer with a given one.
  129. // Caller must ensure there is no race condition.
  130. //
  131. // v2ray:api:stable
  132. func UseAlternativeSystemDialer(dialer SystemDialer) {
  133. if dialer == nil {
  134. dialer = &DefaultSystemDialer{}
  135. }
  136. effectiveSystemDialer = dialer
  137. }
  138. // RegisterDialerController adds a controller to the effective system dialer.
  139. // The controller can be used to operate on file descriptors before they are put into use.
  140. // It only works when effective dialer is the default dialer.
  141. //
  142. // v2ray:api:beta
  143. func RegisterDialerController(ctl func(network, address string, fd uintptr) error) error {
  144. if ctl == nil {
  145. return newError("nil listener controller")
  146. }
  147. dialer, ok := effectiveSystemDialer.(*DefaultSystemDialer)
  148. if !ok {
  149. return newError("RegisterListenerController not supported in custom dialer")
  150. }
  151. dialer.controllers = append(dialer.controllers, ctl)
  152. return nil
  153. }