system_dialer.go 5.1 KB

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