Commit b22dc638 authored by Rob Pike's avatar Rob Pike

sync/once: panicking functions still marked as complete

This is a corner case, and one that was even tested, but this
CL changes the behavior to say that f is "complete" even if it panics.
But don't think of it that way, think of it as sync.Once runs
the function only the first time it is called, rather than
repeatedly until a run of the function completes.

Fixes #8118.

LGTM=dvyukov
R=golang-codereviews, dvyukov
CC=golang-codereviews
https://golang.org/cl/137350043
parent 15274e5c
......@@ -15,7 +15,7 @@ type Once struct {
}
// Do calls the function f if and only if Do is being called for the
// first time for this instance of Once. In other words, given
// first time for this instance of Once. In other words, given
// var once Once
// if once.Do(f) is called multiple times, only the first call will invoke f,
// even if f has a different value in each invocation. A new instance of
......@@ -29,6 +29,9 @@ type Once struct {
// Because no call to Do returns until the one call to f returns, if f causes
// Do to be called, it will deadlock.
//
// If f panics, Do considers it to have returned; future calls of Do return
// without calling f.
//
func (o *Once) Do(f func()) {
if atomic.LoadUint32(&o.done) == 1 {
return
......@@ -37,7 +40,7 @@ func (o *Once) Do(f func()) {
o.m.Lock()
defer o.m.Unlock()
if o.done == 0 {
defer atomic.StoreUint32(&o.done, 1)
f()
atomic.StoreUint32(&o.done, 1)
}
}
......@@ -44,8 +44,12 @@ func TestOncePanic(t *testing.T) {
for i := 0; i < 2; i++ {
func() {
defer func() {
if recover() == nil {
t.Fatalf("Once.Do() has not panic'ed")
r := recover()
if r == nil && i == 0 {
t.Fatalf("Once.Do() has not panic'ed on first iteration")
}
if r != nil && i == 1 {
t.Fatalf("Once.Do() has panic'ed on second iteration")
}
}()
once.Do(func() {
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment