Skip to content

Commit 5d1b48d

Browse files
committed
console_linux: Fix race: lock Cond before Signal.
Possible race: Thread1: Thread2: line 178: Read() -> EAGAIN ... <reschedule> line 110: EpollWait() ... line 124: signalRead() ... ... line 191: Wait() ... line 110: EpollWait() Thread2 (epoll) sends signalRead() (via sync.Cond) but Thread1 is not waiting yet. Then it starts to wait, but no more signals will arrive (because Thread2 is sleeping in EpollWait() on edge-triggered epoll. To prevent this race one should held Lock when sending signalRead(). The same situation goes with Write loop. Bug was reported to docker: docker/for-linux#353 Signed-off-by: Alexander Gerasiov <[email protected]>
1 parent 9a57d21 commit 5d1b48d

File tree

1 file changed

+4
-0
lines changed

1 file changed

+4
-0
lines changed

console_linux.go

+4
Original file line numberDiff line numberDiff line change
@@ -262,10 +262,14 @@ func (ec *EpollConsole) Shutdown(close func(int) error) error {
262262

263263
// signalRead signals that the console is readable.
264264
func (ec *EpollConsole) signalRead() {
265+
ec.readc.L.Lock()
265266
ec.readc.Signal()
267+
ec.readc.L.Unlock()
266268
}
267269

268270
// signalWrite signals that the console is writable.
269271
func (ec *EpollConsole) signalWrite() {
272+
ec.writec.L.Lock()
270273
ec.writec.Signal()
274+
ec.writec.L.Unlock()
271275
}

0 commit comments

Comments
 (0)