| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221 |
- package ray
- import (
- "context"
- "io"
- "os"
- "strconv"
- "sync"
- "time"
- "v2ray.com/core/common/buf"
- )
- // 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"
- sizeStr := os.Getenv(raySizeEnvKey)
- if len(sizeStr) > 0 {
- customSize, err := strconv.ParseUint(sizeStr, 10, 32)
- if err == nil {
- streamSizeLimit = customSize * 1024 * 1024
- }
- }
- }
- type Stream struct {
- access sync.RWMutex
- data buf.MultiBuffer
- size uint64
- ctx context.Context
- readSignal chan bool
- writeSignal chan bool
- close bool
- err bool
- }
- 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
- }
- func (s *Stream) Peek(b *buf.Buffer) {
- s.access.RLock()
- defer s.access.RUnlock()
- b.Reset(func(data []byte) (int, error) {
- return s.data.Copy(data), nil
- })
- }
- func (s *Stream) Read() (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:
- }
- }
- }
- 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:
- }
- }
- }
- func (s *Stream) Write(data buf.MultiBuffer) error {
- if data.IsEmpty() {
- return nil
- }
- for streamSizeLimit > 0 && s.size >= streamSizeLimit {
- select {
- case <-s.ctx.Done():
- return io.ErrClosedPipe
- case <-s.readSignal:
- s.access.RLock()
- if s.err || s.close {
- data.Release()
- s.access.RUnlock()
- return io.ErrClosedPipe
- }
- s.access.RUnlock()
- }
- }
- s.access.Lock()
- defer s.access.Unlock()
- if s.err || s.close {
- data.Release()
- return io.ErrClosedPipe
- }
- if s.data == nil {
- s.data = data
- } else {
- 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:
- }
- }
- func (s *Stream) Close() {
- s.access.Lock()
- s.close = true
- s.notifyRead()
- s.notifyWrite()
- s.access.Unlock()
- }
- 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()
- }
|