1
0
mirror of https://github.com/v2fly/v2ray-core.git synced 2024-06-10 09:50:43 +00:00
v2fly/transport/internet/kcp/dialer.go

154 lines
3.4 KiB
Go
Raw Normal View History

2016-06-14 20:54:08 +00:00
package kcp
import (
"context"
2017-01-13 23:27:45 +00:00
"crypto/cipher"
2016-09-30 14:53:40 +00:00
"crypto/tls"
2016-06-14 22:30:11 +00:00
"net"
2016-11-27 07:58:31 +00:00
"sync"
2016-07-12 16:54:16 +00:00
"sync/atomic"
2016-12-08 15:27:41 +00:00
2017-02-10 15:42:24 +00:00
"v2ray.com/core/app/log"
2017-01-03 14:16:48 +00:00
"v2ray.com/core/common"
2016-12-09 10:35:27 +00:00
"v2ray.com/core/common/buf"
2016-08-20 18:55:45 +00:00
"v2ray.com/core/common/dice"
v2net "v2ray.com/core/common/net"
"v2ray.com/core/transport/internet"
2016-09-30 14:53:40 +00:00
v2tls "v2ray.com/core/transport/internet/tls"
2016-06-14 20:54:08 +00:00
)
2016-07-12 16:54:16 +00:00
var (
2017-02-14 21:29:44 +00:00
globalConv = uint32(dice.RandomUint16())
2016-07-12 16:54:16 +00:00
)
2016-11-27 07:58:31 +00:00
type ClientConnection struct {
2016-12-08 15:27:41 +00:00
sync.RWMutex
2016-11-27 07:58:31 +00:00
net.Conn
2016-12-08 15:27:41 +00:00
input func([]Segment)
reader PacketReader
writer PacketWriter
}
func (o *ClientConnection) Overhead() int {
o.RLock()
defer o.RUnlock()
if o.writer == nil {
return 0
}
return o.writer.Overhead()
}
func (o *ClientConnection) Write(b []byte) (int, error) {
o.RLock()
defer o.RUnlock()
if o.writer == nil {
return len(b), nil
}
return o.writer.Write(b)
2016-11-27 07:58:31 +00:00
}
func (o *ClientConnection) Read([]byte) (int, error) {
panic("KCP|ClientConnection: Read should not be called.")
}
func (o *ClientConnection) Close() error {
return o.Conn.Close()
}
2016-12-08 15:27:41 +00:00
func (o *ClientConnection) Reset(inputCallback func([]Segment)) {
2016-11-27 07:58:31 +00:00
o.Lock()
o.input = inputCallback
2016-12-08 15:27:41 +00:00
o.Unlock()
}
func (o *ClientConnection) ResetSecurity(header internet.PacketHeader, security cipher.AEAD) {
o.Lock()
if o.reader == nil {
o.reader = new(KCPPacketReader)
}
o.reader.(*KCPPacketReader).Header = header
o.reader.(*KCPPacketReader).Security = security
if o.writer == nil {
o.writer = new(KCPPacketWriter)
}
o.writer.(*KCPPacketWriter).Header = header
o.writer.(*KCPPacketWriter).Security = security
o.writer.(*KCPPacketWriter).Writer = o.Conn
2016-11-27 07:58:31 +00:00
o.Unlock()
}
func (o *ClientConnection) Run() {
2016-12-09 11:08:25 +00:00
payload := buf.NewSmall()
2016-11-27 07:58:31 +00:00
defer payload.Release()
for {
2016-12-11 20:43:16 +00:00
err := payload.Reset(buf.ReadFrom(o.Conn))
2016-11-27 07:58:31 +00:00
if err != nil {
payload.Release()
return
}
2016-12-08 15:27:41 +00:00
o.RLock()
if o.input != nil {
segments := o.reader.Read(payload.Bytes())
if len(segments) > 0 {
o.input(segments)
}
2016-11-27 07:58:31 +00:00
}
2016-12-08 15:27:41 +00:00
o.RUnlock()
2016-11-27 07:58:31 +00:00
}
}
func DialKCP(ctx context.Context, dest v2net.Destination) (internet.Connection, error) {
2016-09-20 09:53:05 +00:00
dest.Network = v2net.Network_UDP
2017-04-08 23:43:25 +00:00
log.Trace(newError("KCP|Dialer: Dialing KCP to ", dest))
2016-11-27 07:58:31 +00:00
src := internet.DialerSourceFromContext(ctx)
rawConn, err := internet.DialSystem(ctx, src, dest)
if err != nil {
2017-04-08 23:43:25 +00:00
log.Trace(newError("KCP|Dialer: Failed to dial to dest: ", err).AtError())
return nil, err
}
conn := &ClientConnection{
Conn: rawConn,
2016-06-14 20:54:08 +00:00
}
go conn.Run()
2016-06-14 22:30:11 +00:00
kcpSettings := internet.TransportSettingsFromContext(ctx).(*Config)
2016-10-02 21:43:58 +00:00
2016-12-08 15:27:41 +00:00
header, err := kcpSettings.GetPackerHeader()
2016-08-06 19:59:22 +00:00
if err != nil {
2017-04-08 23:43:25 +00:00
return nil, newError("KCP|Dialer: Failed to create packet header.").Base(err)
2016-12-08 15:27:41 +00:00
}
security, err := kcpSettings.GetSecurity()
if err != nil {
2017-04-08 23:43:25 +00:00
return nil, newError("KCP|Dialer: Failed to create security.").Base(err)
2016-08-06 19:59:22 +00:00
}
conn.ResetSecurity(header, security)
2016-07-12 16:54:16 +00:00
conv := uint16(atomic.AddUint32(&globalConv, 1))
session := NewConnection(conv, conn, kcpSettings)
2016-06-17 14:51:41 +00:00
2016-09-30 14:53:40 +00:00
var iConn internet.Connection
iConn = session
if securitySettings := internet.SecuritySettingsFromContext(ctx); securitySettings != nil {
2016-10-16 12:22:21 +00:00
switch securitySettings := securitySettings.(type) {
case *v2tls.Config:
config := securitySettings.GetTLSConfig()
if dest.Address.Family().IsDomain() {
config.ServerName = dest.Address.Domain()
}
2016-12-20 12:03:20 +00:00
tlsConn := tls.Client(iConn, config)
iConn = tlsConn
2016-09-30 14:53:40 +00:00
}
}
return iConn, nil
2016-06-14 20:54:08 +00:00
}
func init() {
common.Must(internet.RegisterTransportDialer(internet.TransportProtocol_MKCP, DialKCP))
2016-06-14 20:54:08 +00:00
}