v2ray-core/transport/ray/direct.go

246 lines
4.2 KiB
Go
Raw Normal View History

2015-10-14 13:07:13 +00:00
package ray
import (
"context"
2016-04-18 16:44:10 +00:00
"io"
2017-04-16 07:57:28 +00:00
"sync"
2017-01-04 11:34:01 +00:00
"time"
2017-10-23 12:15:16 +00:00
"v2ray.com/core/common"
2016-12-09 10:35:27 +00:00
"v2ray.com/core/common/buf"
2017-05-25 23:11:38 +00:00
"v2ray.com/core/common/platform"
2015-10-14 13:07:13 +00:00
)
2015-10-15 11:15:59 +00:00
// NewRay creates a new Ray for direct traffic transport.
func NewRay(ctx context.Context) Ray {
2015-10-14 13:07:13 +00:00
return &directRay{
Input: NewStream(ctx),
Output: NewStream(ctx),
2015-10-14 13:07:13 +00:00
}
}
type directRay struct {
2016-04-18 16:44:10 +00:00
Input *Stream
Output *Stream
2015-10-14 13:07:13 +00:00
}
2016-11-27 20:39:09 +00:00
func (v *directRay) OutboundInput() InputStream {
return v.Input
2015-10-14 13:07:13 +00:00
}
2016-11-27 20:39:09 +00:00
func (v *directRay) OutboundOutput() OutputStream {
return v.Output
2015-10-14 13:07:13 +00:00
}
2016-11-27 20:39:09 +00:00
func (v *directRay) InboundInput() OutputStream {
return v.Input
2015-10-14 13:07:13 +00:00
}
2016-11-27 20:39:09 +00:00
func (v *directRay) InboundOutput() InputStream {
return v.Output
2015-10-14 13:07:13 +00:00
}
2016-04-18 16:44:10 +00:00
2017-05-18 22:21:48 +00:00
var streamSizeLimit uint64 = 10 * 1024 * 1024
func init() {
const raySizeEnvKey = "v2ray.ray.buffer.size"
2017-05-25 23:11:38 +00:00
size := platform.EnvFlag{
Name: raySizeEnvKey,
AltName: platform.NormalizeEnvName(raySizeEnvKey),
}.GetValueAsInt(10)
streamSizeLimit = uint64(size) * 1024 * 1024
}
2017-10-23 12:15:16 +00:00
// Stream is a sequential container for data in bytes.
2016-04-18 16:44:10 +00:00
type Stream struct {
2017-05-15 18:42:10 +00:00
access sync.RWMutex
data buf.MultiBuffer
size uint64
ctx context.Context
readSignal chan bool
writeSignal chan bool
close bool
err bool
2016-04-18 16:44:10 +00:00
}
2017-10-23 12:15:16 +00:00
// NewStream creates a new Stream.
func NewStream(ctx context.Context) *Stream {
2016-04-18 16:44:10 +00:00
return &Stream{
2017-05-15 18:42:10 +00:00
ctx: ctx,
readSignal: make(chan bool, 1),
writeSignal: make(chan bool, 1),
size: 0,
2016-04-18 16:44:10 +00:00
}
}
2017-04-16 07:57:28 +00:00
func (s *Stream) getData() (buf.MultiBuffer, error) {
s.access.Lock()
defer s.access.Unlock()
if s.data != nil {
mb := s.data
s.data = nil
s.size = 0
2017-04-16 07:57:28 +00:00
return mb, nil
}
if s.close {
return nil, io.EOF
}
if s.err {
2016-12-24 23:42:03 +00:00
return nil, io.ErrClosedPipe
2017-04-16 07:57:28 +00:00
}
return nil, nil
}
2017-10-23 12:15:16 +00:00
// Peek fills in the given buffer with data from head of the Stream.
2017-04-25 08:48:06 +00:00
func (s *Stream) Peek(b *buf.Buffer) {
s.access.RLock()
defer s.access.RUnlock()
2017-10-23 12:15:16 +00:00
common.Must(b.Reset(func(data []byte) (int, error) {
2017-04-25 08:48:06 +00:00
return s.data.Copy(data), nil
2017-10-23 12:15:16 +00:00
}))
}
2017-10-23 12:15:16 +00:00
// Read reads data from the Stream.
2017-04-16 07:57:28 +00:00
func (s *Stream) Read() (buf.MultiBuffer, error) {
for {
mb, err := s.getData()
if err != nil {
return nil, err
}
if mb != nil {
2017-05-15 18:42:10 +00:00
s.notifyRead()
2017-04-16 07:57:28 +00:00
return mb, nil
}
2016-12-24 23:42:03 +00:00
select {
2017-04-16 07:57:28 +00:00
case <-s.ctx.Done():
2017-04-16 19:31:16 +00:00
return nil, io.EOF
2017-05-15 18:42:10 +00:00
case <-s.writeSignal:
2016-12-24 23:42:03 +00:00
}
2016-04-18 16:44:10 +00:00
}
}
2017-10-23 12:15:16 +00:00
// ReadTimeout reads from the Stream with a specified timeout.
2017-04-16 07:57:28 +00:00
func (s *Stream) ReadTimeout(timeout time.Duration) (buf.MultiBuffer, error) {
for {
mb, err := s.getData()
if err != nil {
return nil, err
}
if mb != nil {
2017-05-15 18:42:10 +00:00
s.notifyRead()
2017-04-16 07:57:28 +00:00
return mb, nil
2017-03-27 06:56:16 +00:00
}
2017-01-04 11:34:01 +00:00
select {
2017-04-16 07:57:28 +00:00
case <-s.ctx.Done():
2017-04-16 19:31:16 +00:00
return nil, io.EOF
2017-01-04 11:34:01 +00:00
case <-time.After(timeout):
2017-03-27 09:12:34 +00:00
return nil, buf.ErrReadTimeout
2017-05-15 18:42:10 +00:00
case <-s.writeSignal:
2017-01-04 11:34:01 +00:00
}
}
}
2017-10-23 12:09:14 +00:00
// Size returns the number of bytes hold in the Stream.
func (s *Stream) Size() uint64 {
s.access.RLock()
defer s.access.RUnlock()
return s.size
}
2017-10-23 12:15:16 +00:00
// waitForStreamSize waits until the Stream has room for more data, or any error happens.
2017-10-20 21:23:29 +00:00
func (s *Stream) waitForStreamSize() error {
if streamSizeLimit == 0 {
2017-04-19 09:20:08 +00:00
return nil
2016-12-26 23:44:11 +00:00
}
2017-10-23 12:09:14 +00:00
for s.Size() >= streamSizeLimit {
select {
case <-s.ctx.Done():
return io.ErrClosedPipe
2017-05-15 18:42:10 +00:00
case <-s.readSignal:
if s.err || s.close {
return io.ErrClosedPipe
}
}
}
2017-10-20 21:23:29 +00:00
return nil
}
2017-10-23 12:15:16 +00:00
// Write writes more data into the Stream.
2017-10-20 21:23:29 +00:00
func (s *Stream) Write(data buf.MultiBuffer) error {
if data.IsEmpty() {
return nil
}
if err := s.waitForStreamSize(); err != nil {
data.Release()
return err
}
2017-04-16 07:57:28 +00:00
s.access.Lock()
defer s.access.Unlock()
2017-04-19 09:20:08 +00:00
if s.err || s.close {
2017-04-16 07:57:28 +00:00
data.Release()
2016-12-24 23:42:03 +00:00
return io.ErrClosedPipe
}
2016-04-18 16:44:10 +00:00
2017-04-16 07:57:28 +00:00
if s.data == nil {
s.data = data
} else {
s.data.AppendMulti(data)
}
s.size += uint64(data.Len())
2017-05-15 18:42:10 +00:00
s.notifyWrite()
2016-12-22 16:28:06 +00:00
2017-04-16 07:57:28 +00:00
return nil
2016-04-18 16:44:10 +00:00
}
2017-05-15 18:42:10 +00:00
func (s *Stream) notifyRead() {
2017-04-16 07:57:28 +00:00
select {
2017-05-15 18:42:10 +00:00
case s.readSignal <- true:
default:
}
}
func (s *Stream) notifyWrite() {
select {
case s.writeSignal <- true:
2017-04-16 07:57:28 +00:00
default:
}
}
2016-12-22 16:28:06 +00:00
2017-10-23 12:15:16 +00:00
// Close closes the stream for writing. Read() still works until EOF.
2017-04-16 07:57:28 +00:00
func (s *Stream) Close() {
s.access.Lock()
s.close = true
2017-05-15 18:42:10 +00:00
s.notifyRead()
s.notifyWrite()
2017-04-16 07:57:28 +00:00
s.access.Unlock()
}
2016-12-22 16:28:06 +00:00
2017-10-23 12:15:16 +00:00
// CloseError closes the Stream with error. Read() will return an error afterwards.
2017-04-16 07:57:28 +00:00
func (s *Stream) CloseError() {
s.access.Lock()
s.err = true
if s.data != nil {
s.data.Release()
s.data = nil
2017-05-15 18:42:10 +00:00
s.size = 0
2016-04-18 16:44:10 +00:00
}
2017-05-15 18:42:10 +00:00
s.notifyRead()
s.notifyWrite()
2017-04-16 07:57:28 +00:00
s.access.Unlock()
2016-12-22 16:28:06 +00:00
}