impl.go 2.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155
  1. package pipe
  2. import (
  3. "errors"
  4. "io"
  5. "sync"
  6. "time"
  7. "v2ray.com/core/common/buf"
  8. "v2ray.com/core/common/signal"
  9. )
  10. type state byte
  11. const (
  12. open state = iota
  13. closed
  14. errord
  15. )
  16. type pipe struct {
  17. sync.Mutex
  18. data buf.MultiBuffer
  19. readSignal *signal.Notifier
  20. writeSignal *signal.Notifier
  21. limit int32
  22. state state
  23. }
  24. var errBufferFull = errors.New("buffer full")
  25. func (p *pipe) getState(forRead bool) error {
  26. switch p.state {
  27. case open:
  28. if !forRead && p.limit >= 0 && p.data.Len() > p.limit {
  29. return errBufferFull
  30. }
  31. return nil
  32. case closed:
  33. if !forRead {
  34. return io.ErrClosedPipe
  35. }
  36. if !p.data.IsEmpty() {
  37. return nil
  38. }
  39. return io.EOF
  40. case errord:
  41. return io.ErrClosedPipe
  42. default:
  43. panic("impossible case")
  44. }
  45. }
  46. func (p *pipe) readMultiBufferInternal() (buf.MultiBuffer, error) {
  47. p.Lock()
  48. defer p.Unlock()
  49. if err := p.getState(true); err != nil {
  50. return nil, err
  51. }
  52. data := p.data
  53. p.data = nil
  54. return data, nil
  55. }
  56. func (p *pipe) ReadMultiBuffer() (buf.MultiBuffer, error) {
  57. for {
  58. data, err := p.readMultiBufferInternal()
  59. if data != nil || err != nil {
  60. p.writeSignal.Signal()
  61. return data, err
  62. }
  63. <-p.readSignal.Wait()
  64. }
  65. }
  66. func (p *pipe) ReadMultiBufferWithTimeout(d time.Duration) (buf.MultiBuffer, error) {
  67. timer := time.After(d)
  68. for {
  69. data, err := p.readMultiBufferInternal()
  70. if data != nil || err != nil {
  71. p.writeSignal.Signal()
  72. return data, err
  73. }
  74. select {
  75. case <-p.readSignal.Wait():
  76. case <-timer:
  77. return nil, buf.ErrReadTimeout
  78. }
  79. }
  80. }
  81. func (p *pipe) writeMultiBufferInternal(mb buf.MultiBuffer) error {
  82. p.Lock()
  83. defer p.Unlock()
  84. if err := p.getState(false); err != nil {
  85. return err
  86. }
  87. p.data.AppendMulti(mb)
  88. return nil
  89. }
  90. func (p *pipe) WriteMultiBuffer(mb buf.MultiBuffer) error {
  91. if mb.IsEmpty() {
  92. return nil
  93. }
  94. for {
  95. err := p.writeMultiBufferInternal(mb)
  96. if err == nil || err != errBufferFull {
  97. p.readSignal.Signal()
  98. return err
  99. }
  100. <-p.writeSignal.Wait()
  101. }
  102. }
  103. func (p *pipe) Close() error {
  104. p.Lock()
  105. defer p.Unlock()
  106. if p.state == closed || p.state == errord {
  107. return nil
  108. }
  109. p.state = closed
  110. p.readSignal.Signal()
  111. p.writeSignal.Signal()
  112. return nil
  113. }
  114. func (p *pipe) CloseError() {
  115. p.Lock()
  116. defer p.Unlock()
  117. if p.state == closed || p.state == errord {
  118. return
  119. }
  120. p.state = errord
  121. if !p.data.IsEmpty() {
  122. p.data.Release()
  123. p.data = nil
  124. }
  125. p.readSignal.Signal()
  126. p.writeSignal.Signal()
  127. }