tfo.go 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160
  1. //go:build go1.20
  2. package dialer
  3. import (
  4. "context"
  5. "io"
  6. "net"
  7. "os"
  8. "sync"
  9. "time"
  10. "github.com/sagernet/sing/common"
  11. "github.com/sagernet/sing/common/bufio"
  12. E "github.com/sagernet/sing/common/exceptions"
  13. M "github.com/sagernet/sing/common/metadata"
  14. N "github.com/sagernet/sing/common/network"
  15. "github.com/metacubex/tfo-go"
  16. )
  17. type slowOpenConn struct {
  18. dialer *tfo.Dialer
  19. ctx context.Context
  20. network string
  21. destination M.Socksaddr
  22. conn net.Conn
  23. create chan struct{}
  24. access sync.Mutex
  25. err error
  26. }
  27. func DialSlowContext(dialer *tcpDialer, ctx context.Context, network string, destination M.Socksaddr) (net.Conn, error) {
  28. if dialer.DisableTFO || N.NetworkName(network) != N.NetworkTCP {
  29. switch N.NetworkName(network) {
  30. case N.NetworkTCP, N.NetworkUDP:
  31. return dialer.Dialer.DialContext(ctx, network, destination.String())
  32. default:
  33. return dialer.Dialer.DialContext(ctx, network, destination.AddrString())
  34. }
  35. }
  36. return &slowOpenConn{
  37. dialer: dialer,
  38. ctx: ctx,
  39. network: network,
  40. destination: destination,
  41. create: make(chan struct{}),
  42. }, nil
  43. }
  44. func (c *slowOpenConn) Read(b []byte) (n int, err error) {
  45. if c.conn == nil {
  46. select {
  47. case <-c.create:
  48. if c.err != nil {
  49. return 0, c.err
  50. }
  51. case <-c.ctx.Done():
  52. return 0, c.ctx.Err()
  53. }
  54. }
  55. return c.conn.Read(b)
  56. }
  57. func (c *slowOpenConn) Write(b []byte) (n int, err error) {
  58. if c.conn != nil {
  59. return c.conn.Write(b)
  60. }
  61. c.access.Lock()
  62. defer c.access.Unlock()
  63. select {
  64. case <-c.create:
  65. if c.err != nil {
  66. return 0, c.err
  67. }
  68. return c.conn.Write(b)
  69. default:
  70. }
  71. c.conn, err = c.dialer.DialContext(c.ctx, c.network, c.destination.String(), b)
  72. if err != nil {
  73. c.conn = nil
  74. c.err = E.Cause(err, "dial tcp fast open")
  75. }
  76. n = len(b)
  77. close(c.create)
  78. return
  79. }
  80. func (c *slowOpenConn) Close() error {
  81. return common.Close(c.conn)
  82. }
  83. func (c *slowOpenConn) LocalAddr() net.Addr {
  84. if c.conn == nil {
  85. return M.Socksaddr{}
  86. }
  87. return c.conn.LocalAddr()
  88. }
  89. func (c *slowOpenConn) RemoteAddr() net.Addr {
  90. if c.conn == nil {
  91. return M.Socksaddr{}
  92. }
  93. return c.conn.RemoteAddr()
  94. }
  95. func (c *slowOpenConn) SetDeadline(t time.Time) error {
  96. if c.conn == nil {
  97. return os.ErrInvalid
  98. }
  99. return c.conn.SetDeadline(t)
  100. }
  101. func (c *slowOpenConn) SetReadDeadline(t time.Time) error {
  102. if c.conn == nil {
  103. return os.ErrInvalid
  104. }
  105. return c.conn.SetReadDeadline(t)
  106. }
  107. func (c *slowOpenConn) SetWriteDeadline(t time.Time) error {
  108. if c.conn == nil {
  109. return os.ErrInvalid
  110. }
  111. return c.conn.SetWriteDeadline(t)
  112. }
  113. func (c *slowOpenConn) Upstream() any {
  114. return c.conn
  115. }
  116. func (c *slowOpenConn) ReaderReplaceable() bool {
  117. return c.conn != nil
  118. }
  119. func (c *slowOpenConn) WriterReplaceable() bool {
  120. return c.conn != nil
  121. }
  122. func (c *slowOpenConn) LazyHeadroom() bool {
  123. return c.conn == nil
  124. }
  125. func (c *slowOpenConn) NeedHandshake() bool {
  126. return c.conn == nil
  127. }
  128. func (c *slowOpenConn) WriteTo(w io.Writer) (n int64, err error) {
  129. if c.conn == nil {
  130. select {
  131. case <-c.create:
  132. if c.err != nil {
  133. return 0, c.err
  134. }
  135. case <-c.ctx.Done():
  136. return 0, c.ctx.Err()
  137. }
  138. }
  139. return bufio.Copy(w, c.conn)
  140. }