@@ -14,8 +14,16 @@ import (
14
14
)
15
15
16
16
type streamsState struct {
17
- queue queue [* Stream ] // new, peer-created streams
18
- streams map [streamID ]* Stream
17
+ queue queue [* Stream ] // new, peer-created streams
18
+
19
+ // All peer-created streams.
20
+ //
21
+ // Implicitly created streams are included as an empty entry in the map.
22
+ // (For example, if we receive a frame for stream 4, we implicitly create stream 0 and
23
+ // insert an empty entry for it to the map.)
24
+ //
25
+ // The map value is maybeStream rather than *Stream as a reminder that values can be nil.
26
+ streams map [streamID ]maybeStream
19
27
20
28
// Limits on the number of streams, indexed by streamType.
21
29
localLimit [streamTypeCount ]localStreamLimits
@@ -37,8 +45,13 @@ type streamsState struct {
37
45
queueData streamRing // streams with only flow-controlled frames
38
46
}
39
47
48
+ // maybeStream is a possibly nil *Stream. See streamsState.streams.
49
+ type maybeStream struct {
50
+ s * Stream
51
+ }
52
+
40
53
func (c * Conn ) streamsInit () {
41
- c .streams .streams = make (map [streamID ]* Stream )
54
+ c .streams .streams = make (map [streamID ]maybeStream )
42
55
c .streams .queue = newQueue [* Stream ]()
43
56
c .streams .localLimit [bidiStream ].init ()
44
57
c .streams .localLimit [uniStream ].init ()
@@ -52,8 +65,8 @@ func (c *Conn) streamsCleanup() {
52
65
c .streams .localLimit [bidiStream ].connHasClosed ()
53
66
c .streams .localLimit [uniStream ].connHasClosed ()
54
67
for _ , s := range c .streams .streams {
55
- if s != nil {
56
- s .connHasClosed ()
68
+ if s . s != nil {
69
+ s .s . connHasClosed ()
57
70
}
58
71
}
59
72
}
@@ -97,7 +110,7 @@ func (c *Conn) newLocalStream(ctx context.Context, styp streamType) (*Stream, er
97
110
98
111
// Modify c.streams on the conn's loop.
99
112
if err := c .runOnLoop (ctx , func (now time.Time , c * Conn ) {
100
- c .streams .streams [s .id ] = s
113
+ c .streams .streams [s .id ] = maybeStream { s }
101
114
}); err != nil {
102
115
return nil , err
103
116
}
@@ -119,7 +132,7 @@ const (
119
132
// streamForID returns the stream with the given id.
120
133
// If the stream does not exist, it returns nil.
121
134
func (c * Conn ) streamForID (id streamID ) * Stream {
122
- return c .streams .streams [id ]
135
+ return c .streams .streams [id ]. s
123
136
}
124
137
125
138
// streamForFrame returns the stream with the given id.
@@ -144,9 +157,9 @@ func (c *Conn) streamForFrame(now time.Time, id streamID, ftype streamFrameType)
144
157
}
145
158
}
146
159
147
- s , isOpen := c .streams .streams [id ]
148
- if s != nil {
149
- return s
160
+ ms , isOpen := c .streams .streams [id ]
161
+ if ms . s != nil {
162
+ return ms . s
150
163
}
151
164
152
165
num := id .num ()
@@ -183,10 +196,10 @@ func (c *Conn) streamForFrame(now time.Time, id streamID, ftype streamFrameType)
183
196
// with the same initiator and type and a lower number.
184
197
// Add a nil entry to the streams map for each implicitly created stream.
185
198
for n := newStreamID (id .initiator (), id .streamType (), prevOpened ); n < id ; n += 4 {
186
- c .streams .streams [n ] = nil
199
+ c .streams .streams [n ] = maybeStream {}
187
200
}
188
201
189
- s = newStream (c , id )
202
+ s : = newStream (c , id )
190
203
s .inmaxbuf = c .config .maxStreamReadBufferSize ()
191
204
s .inwin = c .config .maxStreamReadBufferSize ()
192
205
if id .streamType () == bidiStream {
@@ -196,7 +209,7 @@ func (c *Conn) streamForFrame(now time.Time, id streamID, ftype streamFrameType)
196
209
s .inUnlock ()
197
210
s .outUnlock ()
198
211
199
- c .streams .streams [id ] = s
212
+ c .streams .streams [id ] = maybeStream { s }
200
213
c .streams .queue .put (s )
201
214
return s
202
215
}
@@ -400,7 +413,11 @@ func (c *Conn) appendStreamFramesPTO(w *packetWriter, pnum packetNumber) bool {
400
413
c .streams .sendMu .Lock ()
401
414
defer c .streams .sendMu .Unlock ()
402
415
const pto = true
403
- for _ , s := range c .streams .streams {
416
+ for _ , ms := range c .streams .streams {
417
+ s := ms .s
418
+ if s == nil {
419
+ continue
420
+ }
404
421
const pto = true
405
422
s .ingate .lock ()
406
423
inOK := s .appendInFramesLocked (w , pnum , pto )
0 commit comments