dialer.go 2.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596
  1. package kcp
  2. import (
  3. "context"
  4. "io"
  5. "sync/atomic"
  6. "github.com/xtls/xray-core/common"
  7. "github.com/xtls/xray-core/common/buf"
  8. "github.com/xtls/xray-core/common/dice"
  9. "github.com/xtls/xray-core/common/net"
  10. "github.com/xtls/xray-core/transport/internet"
  11. "github.com/xtls/xray-core/transport/internet/stat"
  12. "github.com/xtls/xray-core/transport/internet/tls"
  13. )
  14. var globalConv = uint32(dice.RollUint16())
  15. func fetchInput(_ context.Context, input io.Reader, reader PacketReader, conn *Connection) {
  16. cache := make(chan *buf.Buffer, 1024)
  17. go func() {
  18. for {
  19. payload := buf.New()
  20. if _, err := payload.ReadFrom(input); err != nil {
  21. payload.Release()
  22. close(cache)
  23. return
  24. }
  25. select {
  26. case cache <- payload:
  27. default:
  28. payload.Release()
  29. }
  30. }
  31. }()
  32. for payload := range cache {
  33. segments := reader.Read(payload.Bytes())
  34. payload.Release()
  35. if len(segments) > 0 {
  36. conn.Input(segments)
  37. }
  38. }
  39. }
  40. // DialKCP dials a new KCP connections to the specific destination.
  41. func DialKCP(ctx context.Context, dest net.Destination, streamSettings *internet.MemoryStreamConfig) (stat.Connection, error) {
  42. dest.Network = net.Network_UDP
  43. newError("dialing mKCP to ", dest).WriteToLog()
  44. rawConn, err := internet.DialSystem(ctx, dest, streamSettings.SocketSettings)
  45. if err != nil {
  46. return nil, newError("failed to dial to dest: ", err).AtWarning().Base(err)
  47. }
  48. kcpSettings := streamSettings.ProtocolSettings.(*Config)
  49. header, err := kcpSettings.GetPackerHeader()
  50. if err != nil {
  51. return nil, newError("failed to create packet header").Base(err)
  52. }
  53. security, err := kcpSettings.GetSecurity()
  54. if err != nil {
  55. return nil, newError("failed to create security").Base(err)
  56. }
  57. reader := &KCPPacketReader{
  58. Header: header,
  59. Security: security,
  60. }
  61. writer := &KCPPacketWriter{
  62. Header: header,
  63. Security: security,
  64. Writer: rawConn,
  65. }
  66. conv := uint16(atomic.AddUint32(&globalConv, 1))
  67. session := NewConnection(ConnMetadata{
  68. LocalAddr: rawConn.LocalAddr(),
  69. RemoteAddr: rawConn.RemoteAddr(),
  70. Conversation: conv,
  71. }, writer, rawConn, kcpSettings)
  72. go fetchInput(ctx, rawConn, reader, session)
  73. var iConn stat.Connection = session
  74. if config := tls.ConfigFromStreamSettings(streamSettings); config != nil {
  75. iConn = tls.Client(iConn, config.GetTLSConfig(tls.WithDestination(dest)))
  76. }
  77. return iConn, nil
  78. }
  79. func init() {
  80. common.Must(internet.RegisterTransportDialer(protocolName, DialKCP))
  81. }