mirror of https://github.com/v2ray/v2ray-core
				
				
				
			udp for shadowsocks
							parent
							
								
									1f9bd5f692
								
							
						
					
					
						commit
						dde3f60e30
					
				|  | @ -1,6 +1,7 @@ | ||||||
| package alloc | package alloc | ||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
|  | 	"io" | ||||||
| 	"sync" | 	"sync" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
|  | @ -72,6 +73,19 @@ func (b *Buffer) Write(data []byte) (int, error) { | ||||||
| 	return len(data), nil | 	return len(data), nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | func (b *Buffer) Read(data []byte) (int, error) { | ||||||
|  | 	if b.Len() == 0 { | ||||||
|  | 		return 0, io.EOF | ||||||
|  | 	} | ||||||
|  | 	nBytes := copy(data, b.Value) | ||||||
|  | 	if nBytes == b.Len() { | ||||||
|  | 		b.Value = b.Value[:0] | ||||||
|  | 	} else { | ||||||
|  | 		b.Value = b.Value[nBytes:] | ||||||
|  | 	} | ||||||
|  | 	return nBytes, nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
| type bufferPool struct { | type bufferPool struct { | ||||||
| 	chain     chan []byte | 	chain     chan []byte | ||||||
| 	allocator *sync.Pool | 	allocator *sync.Pool | ||||||
|  |  | ||||||
|  | @ -20,7 +20,8 @@ type Shadowsocks struct { | ||||||
| 	config    *Config | 	config    *Config | ||||||
| 	port      v2net.Port | 	port      v2net.Port | ||||||
| 	accepting bool | 	accepting bool | ||||||
| 	tcpListener *hub.TCPHub | 	tcpHub    *hub.TCPHub | ||||||
|  | 	udpHub    *hub.UDPHub | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (this *Shadowsocks) Port() v2net.Port { | func (this *Shadowsocks) Port() v2net.Port { | ||||||
|  | @ -29,8 +30,8 @@ func (this *Shadowsocks) Port() v2net.Port { | ||||||
| 
 | 
 | ||||||
| func (this *Shadowsocks) Close() { | func (this *Shadowsocks) Close() { | ||||||
| 	this.accepting = false | 	this.accepting = false | ||||||
| 	this.tcpListener.Close() | 	this.tcpHub.Close() | ||||||
| 	this.tcpListener = nil | 	this.tcpHub = nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (this *Shadowsocks) Listen(port v2net.Port) error { | func (this *Shadowsocks) Listen(port v2net.Port) error { | ||||||
|  | @ -41,17 +42,82 @@ func (this *Shadowsocks) Listen(port v2net.Port) error { | ||||||
| 			return proxy.ErrorAlreadyListening | 			return proxy.ErrorAlreadyListening | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  | 	this.accepting = true | ||||||
| 
 | 
 | ||||||
| 	tcpListener, err := hub.ListenTCP(port, this.handleConnection) | 	tcpHub, err := hub.ListenTCP(port, this.handleConnection) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		log.Error("Shadowsocks: Failed to listen on port ", port, ": ", err) | 		log.Error("Shadowsocks: Failed to listen TCP on port ", port, ": ", err) | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| 	this.tcpListener = tcpListener | 	this.tcpHub = tcpHub | ||||||
| 	this.accepting = true | 
 | ||||||
|  | 	if this.config.UDP { | ||||||
|  | 		udpHub, err := hub.ListenUDP(port, this.handlerUDPPayload) | ||||||
|  | 		if err != nil { | ||||||
|  | 			log.Error("Shadowsocks: Failed to listen UDP on port ", port, ": ", err) | ||||||
|  | 		} | ||||||
|  | 		this.udpHub = udpHub | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | func (this *Shadowsocks) handlerUDPPayload(payload *alloc.Buffer, dest v2net.Destination) { | ||||||
|  | 	defer payload.Release() | ||||||
|  | 
 | ||||||
|  | 	iv := payload.Value[:this.config.Cipher.IVSize()] | ||||||
|  | 	key := this.config.Key | ||||||
|  | 	payload.SliceFrom(this.config.Cipher.IVSize()) | ||||||
|  | 
 | ||||||
|  | 	reader, err := this.config.Cipher.NewDecodingStream(key, iv, payload) | ||||||
|  | 	if err != nil { | ||||||
|  | 		log.Error("Shadowsocks: Failed to create decoding stream: ", err) | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	request, err := ReadRequest(reader) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	buffer, _ := v2net.ReadFrom(reader, nil) | ||||||
|  | 
 | ||||||
|  | 	packet := v2net.NewPacket(v2net.TCPDestination(request.Address, request.Port), buffer, false) | ||||||
|  | 	ray := this.space.PacketDispatcher().DispatchToOutbound(packet) | ||||||
|  | 	close(ray.InboundInput()) | ||||||
|  | 
 | ||||||
|  | 	for respChunk := range ray.InboundOutput() { | ||||||
|  | 
 | ||||||
|  | 		response := alloc.NewBuffer().Slice(0, this.config.Cipher.IVSize()) | ||||||
|  | 		rand.Read(response.Value) | ||||||
|  | 
 | ||||||
|  | 		writer, err := this.config.Cipher.NewEncodingStream(key, response.Value, response) | ||||||
|  | 		if err != nil { | ||||||
|  | 			log.Error("Shadowsocks: Failed to create encoding stream: ", err) | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		switch { | ||||||
|  | 		case request.Address.IsIPv4(): | ||||||
|  | 			writer.Write([]byte{AddrTypeIPv4}) | ||||||
|  | 			writer.Write(request.Address.IP()) | ||||||
|  | 		case request.Address.IsIPv6(): | ||||||
|  | 			writer.Write([]byte{AddrTypeIPv6}) | ||||||
|  | 			writer.Write(request.Address.IP()) | ||||||
|  | 		case request.Address.IsDomain(): | ||||||
|  | 			writer.Write([]byte{AddrTypeDomain, byte(len(request.Address.Domain()))}) | ||||||
|  | 			writer.Write([]byte(request.Address.Domain())) | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		writer.Write(request.Port.Bytes()) | ||||||
|  | 		writer.Write(respChunk.Value) | ||||||
|  | 		respChunk.Release() | ||||||
|  | 
 | ||||||
|  | 		this.udpHub.WriteTo(response.Value, dest) | ||||||
|  | 		response.Release() | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
| func (this *Shadowsocks) handleConnection(conn *hub.TCPConn) { | func (this *Shadowsocks) handleConnection(conn *hub.TCPConn) { | ||||||
| 	defer conn.Close() | 	defer conn.Close() | ||||||
| 
 | 
 | ||||||
|  | @ -81,23 +147,18 @@ func (this *Shadowsocks) handleConnection(conn *hub.TCPConn) { | ||||||
| 	packet := v2net.NewPacket(v2net.TCPDestination(request.Address, request.Port), nil, true) | 	packet := v2net.NewPacket(v2net.TCPDestination(request.Address, request.Port), nil, true) | ||||||
| 	ray := this.space.PacketDispatcher().DispatchToOutbound(packet) | 	ray := this.space.PacketDispatcher().DispatchToOutbound(packet) | ||||||
| 
 | 
 | ||||||
| 	respIv := make([]byte, this.config.Cipher.IVSize()) | 	var writeFinish sync.Mutex | ||||||
| 	rand.Read(respIv) | 	writeFinish.Lock() | ||||||
|  | 	go func() { | ||||||
|  | 		firstChunk := alloc.NewBuffer().Slice(0, this.config.Cipher.IVSize()) | ||||||
|  | 		defer firstChunk.Release() | ||||||
| 
 | 
 | ||||||
| 	writer, err := this.config.Cipher.NewEncodingStream(key, respIv, conn) | 		writer, err := this.config.Cipher.NewEncodingStream(key, firstChunk.Value, conn) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			log.Error("Shadowsocks: Failed to create encoding stream: ", err) | 			log.Error("Shadowsocks: Failed to create encoding stream: ", err) | ||||||
| 			return | 			return | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 	var writeFinish sync.Mutex |  | ||||||
| 	writeFinish.Lock() |  | ||||||
| 	go func() { |  | ||||||
| 		firstChunk := alloc.NewBuffer().Clear() |  | ||||||
| 		defer firstChunk.Release() |  | ||||||
| 
 |  | ||||||
| 		firstChunk.Append(respIv) |  | ||||||
| 
 |  | ||||||
| 		if payload, ok := <-ray.InboundOutput(); ok { | 		if payload, ok := <-ray.InboundOutput(); ok { | ||||||
| 			firstChunk.Append(payload.Value) | 			firstChunk.Append(payload.Value) | ||||||
| 			payload.Release() | 			payload.Release() | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue
	
	 v2ray
						v2ray