mirror of
https://github.com/restic/restic.git
synced 2025-01-03 05:35:43 +00:00
bloblru: fix race condition that can compute value multiple times
This commit is contained in:
parent
3828313974
commit
0ad0b7ca7c
1 changed files with 13 additions and 4 deletions
|
@ -114,11 +114,20 @@ func (c *Cache) GetOrCompute(id restic.ID, compute func() ([]byte, error)) ([]by
|
|||
if isDownloading {
|
||||
// wait for result of parallel download
|
||||
<-waitForResult
|
||||
blob, ok := c.Get(id)
|
||||
}
|
||||
|
||||
// try again. This is necessary independent of whether isDownloading is true or not.
|
||||
// The calls to `c.Get()` and checking/adding the entry in `c.inProgress` are not atomic,
|
||||
// thus the item might have been computed in the meantime.
|
||||
// The following scenario would compute() the value multiple times otherwise:
|
||||
// Goroutine A does not find a value in the initial call to `c.Get`, then goroutine B
|
||||
// takes over, caches the computed value and cleans up its channel in c.inProgress.
|
||||
// Then goroutine A continues, does not detect a parallel computation and would try
|
||||
// to call compute() again.
|
||||
blob, ok = c.Get(id)
|
||||
if ok {
|
||||
return blob, nil
|
||||
}
|
||||
}
|
||||
|
||||
// download it
|
||||
blob, err := compute()
|
||||
|
|
Loading…
Reference in a new issue