18
18
// Package buffer provides an implementation of an unbounded buffer.
19
19
package buffer
20
20
21
- import "sync"
21
+ import (
22
+ "sync"
23
+ )
22
24
23
25
// Unbounded is an implementation of an unbounded buffer which does not use
24
26
// extra goroutines. This is typically used for passing updates from one entity
@@ -34,9 +36,10 @@ import "sync"
34
36
// defining a new type specific implementation of this buffer is preferred. See
35
37
// internal/transport/transport.go for an example of this.
36
38
type Unbounded struct {
37
- c chan interface {}
38
- mu sync.Mutex
39
- backlog []interface {}
39
+ c chan interface {}
40
+ mu sync.Mutex
41
+ backlog []interface {}
42
+ disposed bool
40
43
}
41
44
42
45
// NewUnbounded returns a new instance of Unbounded.
@@ -47,35 +50,46 @@ func NewUnbounded() *Unbounded {
47
50
// Put adds t to the unbounded buffer.
48
51
func (b * Unbounded ) Put (t interface {}) (ok bool ) {
49
52
b .mu .Lock ()
50
- defer func () {
51
- ok = recover () == nil
53
+
54
+ if b . disposed {
52
55
b .mu .Unlock ()
53
- }()
56
+ return
57
+ }
58
+
59
+ ok = true
60
+
54
61
if len (b .backlog ) == 0 {
55
62
select {
56
63
case b .c <- t :
64
+ b .mu .Unlock ()
57
65
return
58
66
default :
59
67
}
60
68
}
61
69
b .backlog = append (b .backlog , t )
70
+ b .mu .Unlock ()
62
71
return
63
72
}
64
73
65
74
// Load sends the earliest buffered data, if any, onto the read channel
66
75
// returned by Get(). Users are expected to call this every time they read a
67
76
// value from the read channel.
68
- func (b * Unbounded ) Load () {
77
+ func (b * Unbounded ) Load () ( n int ) {
69
78
b .mu .Lock ()
70
79
if len (b .backlog ) > 0 {
71
80
select {
72
81
case b .c <- b .backlog [0 ]:
73
82
b .backlog [0 ] = nil
74
83
b .backlog = b .backlog [1 :]
84
+ n = 1
75
85
default :
76
86
}
87
+ } else if b .disposed {
88
+ b .close ()
89
+ n = - 1
77
90
}
78
91
b .mu .Unlock ()
92
+ return
79
93
}
80
94
81
95
// Get returns a read channel on which values added to the buffer, via Put(),
@@ -87,6 +101,20 @@ func (b *Unbounded) Get() <-chan interface{} {
87
101
return b .c
88
102
}
89
103
104
+ // Dispose mark current Unbounded as disposed.
90
105
func (b * Unbounded ) Dispose () {
106
+ b .mu .Lock ()
107
+ b .disposed = true
108
+ if len (b .backlog ) == 0 {
109
+ b .close ()
110
+ }
111
+ b .mu .Unlock ()
112
+ }
113
+
114
+ func (b * Unbounded ) close () (ok bool ) {
115
+ defer func () {
116
+ ok = recover () == nil
117
+ }()
91
118
close (b .c )
119
+ return
92
120
}
0 commit comments