dialer.go 2.4 KB

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