@@ -82,9 +82,32 @@ type impl struct {
82
82
receivers []Receiver
83
83
}
84
84
85
+ // interfaceWrapper is concrete type that wraps an interface. Necessary because
86
+ // atomic.Value needs the same type and can not Store(nil). This indirection
87
+ // allows us to store nil.
88
+ type interfaceWrapper [T any ] struct {
89
+ t T
90
+ }
91
+ type atomicInterface [T any ] struct {
92
+ iface atomic.Value
93
+ }
94
+
95
+ func (a * atomicInterface [T ]) Load () T {
96
+ var v T
97
+ x := a .iface .Load ()
98
+ if x != nil {
99
+ return x .(interfaceWrapper [T ]).t
100
+ }
101
+ return v
102
+ }
103
+
104
+ func (a * atomicInterface [T ]) Store (v T ) {
105
+ a .iface .Store (interfaceWrapper [T ]{v })
106
+ }
107
+
85
108
type streamMessageSender struct {
86
109
to peer.ID
87
- stream network.Stream
110
+ stream atomicInterface [ network.Stream ]
88
111
bsnet * impl
89
112
opts * MessageSenderOpts
90
113
}
@@ -95,7 +118,7 @@ type HasContext interface {
95
118
96
119
// Open a stream to the remote peer
97
120
func (s * streamMessageSender ) Connect (ctx context.Context ) (network.Stream , error ) {
98
- stream := s .stream
121
+ stream := s .stream . Load ()
99
122
if stream != nil {
100
123
return stream , nil
101
124
}
@@ -111,36 +134,41 @@ func (s *streamMessageSender) Connect(ctx context.Context) (network.Stream, erro
111
134
if err != nil {
112
135
return nil , err
113
136
}
137
+ if withCtx , ok := stream .Conn ().(HasContext ); ok {
138
+ context .AfterFunc (withCtx .Context (), func () {
139
+ s .stream .Store (nil )
140
+ })
141
+ }
114
142
115
- s .stream = stream
143
+ s .stream . Store ( stream )
116
144
return stream , nil
117
145
}
118
146
119
147
// Reset the stream
120
148
func (s * streamMessageSender ) Reset () error {
121
- stream := s .stream
149
+ stream := s .stream . Load ()
122
150
if stream != nil {
123
151
err := stream .Reset ()
124
- s .stream = nil
152
+ s .stream . Store ( nil )
125
153
return err
126
154
}
127
155
return nil
128
156
}
129
157
130
158
// Close the stream
131
159
func (s * streamMessageSender ) Close () error {
132
- stream := s .stream
160
+ stream := s .stream . Load ()
133
161
if stream != nil {
134
162
err := stream .Close ()
135
- s .stream = nil
163
+ s .stream . Store ( nil )
136
164
return err
137
165
}
138
166
return nil
139
167
}
140
168
141
169
// Indicates whether the peer supports HAVE / DONT_HAVE messages
142
170
func (s * streamMessageSender ) SupportsHave () bool {
143
- stream := s .stream
171
+ stream := s .stream . Load ()
144
172
if stream == nil {
145
173
return false
146
174
}
0 commit comments