| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245 | 
							- package ray
 
- import (
 
- 	"context"
 
- 	"io"
 
- 	"sync"
 
- 	"time"
 
- 	"v2ray.com/core/common"
 
- 	"v2ray.com/core/common/buf"
 
- 	"v2ray.com/core/common/platform"
 
- )
 
- // NewRay creates a new Ray for direct traffic transport.
 
- func NewRay(ctx context.Context) Ray {
 
- 	return &directRay{
 
- 		Input:  NewStream(ctx),
 
- 		Output: NewStream(ctx),
 
- 	}
 
- }
 
- type directRay struct {
 
- 	Input  *Stream
 
- 	Output *Stream
 
- }
 
- func (v *directRay) OutboundInput() InputStream {
 
- 	return v.Input
 
- }
 
- func (v *directRay) OutboundOutput() OutputStream {
 
- 	return v.Output
 
- }
 
- func (v *directRay) InboundInput() OutputStream {
 
- 	return v.Input
 
- }
 
- func (v *directRay) InboundOutput() InputStream {
 
- 	return v.Output
 
- }
 
- var streamSizeLimit uint64 = 10 * 1024 * 1024
 
- func init() {
 
- 	const raySizeEnvKey = "v2ray.ray.buffer.size"
 
- 	size := platform.EnvFlag{
 
- 		Name:    raySizeEnvKey,
 
- 		AltName: platform.NormalizeEnvName(raySizeEnvKey),
 
- 	}.GetValueAsInt(10)
 
- 	streamSizeLimit = uint64(size) * 1024 * 1024
 
- }
 
- // Stream is a sequential container for data in bytes.
 
- type Stream struct {
 
- 	access      sync.RWMutex
 
- 	data        buf.MultiBuffer
 
- 	size        uint64
 
- 	ctx         context.Context
 
- 	readSignal  chan bool
 
- 	writeSignal chan bool
 
- 	close       bool
 
- 	err         bool
 
- }
 
- // NewStream creates a new Stream.
 
- func NewStream(ctx context.Context) *Stream {
 
- 	return &Stream{
 
- 		ctx:         ctx,
 
- 		readSignal:  make(chan bool, 1),
 
- 		writeSignal: make(chan bool, 1),
 
- 		size:        0,
 
- 	}
 
- }
 
- 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
 
- 		return mb, nil
 
- 	}
 
- 	if s.close {
 
- 		return nil, io.EOF
 
- 	}
 
- 	if s.err {
 
- 		return nil, io.ErrClosedPipe
 
- 	}
 
- 	return nil, nil
 
- }
 
- // Peek fills in the given buffer with data from head of the Stream.
 
- func (s *Stream) Peek(b *buf.Buffer) {
 
- 	s.access.RLock()
 
- 	defer s.access.RUnlock()
 
- 	common.Must(b.Reset(func(data []byte) (int, error) {
 
- 		return s.data.Copy(data), nil
 
- 	}))
 
- }
 
- // Read reads data from the Stream.
 
- func (s *Stream) ReadMultiBuffer() (buf.MultiBuffer, error) {
 
- 	for {
 
- 		mb, err := s.getData()
 
- 		if err != nil {
 
- 			return nil, err
 
- 		}
 
- 		if mb != nil {
 
- 			s.notifyRead()
 
- 			return mb, nil
 
- 		}
 
- 		select {
 
- 		case <-s.ctx.Done():
 
- 			return nil, io.EOF
 
- 		case <-s.writeSignal:
 
- 		}
 
- 	}
 
- }
 
- // ReadTimeout reads from the Stream with a specified timeout.
 
- func (s *Stream) ReadTimeout(timeout time.Duration) (buf.MultiBuffer, error) {
 
- 	for {
 
- 		mb, err := s.getData()
 
- 		if err != nil {
 
- 			return nil, err
 
- 		}
 
- 		if mb != nil {
 
- 			s.notifyRead()
 
- 			return mb, nil
 
- 		}
 
- 		select {
 
- 		case <-s.ctx.Done():
 
- 			return nil, io.EOF
 
- 		case <-time.After(timeout):
 
- 			return nil, buf.ErrReadTimeout
 
- 		case <-s.writeSignal:
 
- 		}
 
- 	}
 
- }
 
- // 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
 
- }
 
- // waitForStreamSize waits until the Stream has room for more data, or any error happens.
 
- func (s *Stream) waitForStreamSize() error {
 
- 	if streamSizeLimit == 0 {
 
- 		return nil
 
- 	}
 
- 	for s.Size() >= streamSizeLimit {
 
- 		select {
 
- 		case <-s.ctx.Done():
 
- 			return io.ErrClosedPipe
 
- 		case <-s.readSignal:
 
- 			if s.err || s.close {
 
- 				return io.ErrClosedPipe
 
- 			}
 
- 		}
 
- 	}
 
- 	return nil
 
- }
 
- // Write writes more data into the Stream.
 
- func (s *Stream) WriteMultiBuffer(data buf.MultiBuffer) error {
 
- 	if data.IsEmpty() {
 
- 		return nil
 
- 	}
 
- 	if err := s.waitForStreamSize(); err != nil {
 
- 		data.Release()
 
- 		return err
 
- 	}
 
- 	s.access.Lock()
 
- 	defer s.access.Unlock()
 
- 	if s.err || s.close {
 
- 		data.Release()
 
- 		return io.ErrClosedPipe
 
- 	}
 
- 	if s.data == nil {
 
- 		s.data = buf.NewMultiBufferCap(128)
 
- 	}
 
- 	s.data.AppendMulti(data)
 
- 	s.size += uint64(data.Len())
 
- 	s.notifyWrite()
 
- 	return nil
 
- }
 
- func (s *Stream) notifyRead() {
 
- 	select {
 
- 	case s.readSignal <- true:
 
- 	default:
 
- 	}
 
- }
 
- func (s *Stream) notifyWrite() {
 
- 	select {
 
- 	case s.writeSignal <- true:
 
- 	default:
 
- 	}
 
- }
 
- // Close closes the stream for writing. Read() still works until EOF.
 
- func (s *Stream) Close() {
 
- 	s.access.Lock()
 
- 	s.close = true
 
- 	s.notifyRead()
 
- 	s.notifyWrite()
 
- 	s.access.Unlock()
 
- }
 
- // CloseError closes the Stream with error. Read() will return an error afterwards.
 
- func (s *Stream) CloseError() {
 
- 	s.access.Lock()
 
- 	s.err = true
 
- 	if s.data != nil {
 
- 		s.data.Release()
 
- 		s.data = nil
 
- 		s.size = 0
 
- 	}
 
- 	s.notifyRead()
 
- 	s.notifyWrite()
 
- 	s.access.Unlock()
 
- }
 
 
  |