blob: 5ad17fa9132f71266a4bd1eabad0687ef2a84b21 [file] [log] [blame]
Colin Cross9272ade2016-08-17 15:24:12 -07001// Copyright 2016 Google Inc. All rights reserved.
2//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7// http://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14
15package android
16
17import (
Colin Cross99c6dfa2017-11-07 13:34:26 -080018 "fmt"
Colin Cross9272ade2016-08-17 15:24:12 -070019 "sync"
Colin Cross9272ade2016-08-17 15:24:12 -070020)
21
22type OncePer struct {
Colin Crosse5cdaf92019-02-11 15:06:16 -080023 values sync.Map
24}
25
26type onceValueWaiter chan bool
27
28func (once *OncePer) maybeWaitFor(key OnceKey, value interface{}) interface{} {
29 if wait, isWaiter := value.(onceValueWaiter); isWaiter {
30 // The entry in the map is a placeholder waiter because something else is constructing the value
31 // wait until the waiter is signalled, then load the real value.
32 <-wait
33 value, _ = once.values.Load(key)
34 if _, isWaiter := value.(onceValueWaiter); isWaiter {
35 panic(fmt.Errorf("Once() waiter completed but key is still not valid"))
36 }
37 }
38
39 return value
Colin Cross9272ade2016-08-17 15:24:12 -070040}
41
Colin Cross9272ade2016-08-17 15:24:12 -070042// Once computes a value the first time it is called with a given key per OncePer, and returns the
43// value without recomputing when called with the same key. key must be hashable.
Colin Crosse48ff5b2019-02-04 13:03:13 -080044func (once *OncePer) Once(key OnceKey, value func() interface{}) interface{} {
Colin Cross99c6dfa2017-11-07 13:34:26 -080045 // Fast path: check if the key is already in the map
46 if v, ok := once.values.Load(key); ok {
Colin Crosse5cdaf92019-02-11 15:06:16 -080047 return once.maybeWaitFor(key, v)
Colin Cross9272ade2016-08-17 15:24:12 -070048 }
49
Colin Crosse5cdaf92019-02-11 15:06:16 -080050 // Slow path: create a OnceValueWrapper and attempt to insert it
51 waiter := make(onceValueWaiter)
52 if v, loaded := once.values.LoadOrStore(key, waiter); loaded {
53 // Got a value, something else inserted its own waiter or a constructed value
54 return once.maybeWaitFor(key, v)
Colin Cross9272ade2016-08-17 15:24:12 -070055 }
56
Colin Crosse5cdaf92019-02-11 15:06:16 -080057 // The waiter is inserted, call the value constructor, store it, and signal the waiter
Colin Cross9272ade2016-08-17 15:24:12 -070058 v := value()
Colin Cross99c6dfa2017-11-07 13:34:26 -080059 once.values.Store(key, v)
Colin Crosse5cdaf92019-02-11 15:06:16 -080060 close(waiter)
Colin Cross9272ade2016-08-17 15:24:12 -070061
Colin Cross99c6dfa2017-11-07 13:34:26 -080062 return v
63}
64
Colin Cross571cccf2019-02-04 11:22:08 -080065// Get returns the value previously computed with Once for a given key. If Once has not been called for the given
66// key Get will panic.
Colin Crosse48ff5b2019-02-04 13:03:13 -080067func (once *OncePer) Get(key OnceKey) interface{} {
Colin Cross99c6dfa2017-11-07 13:34:26 -080068 v, ok := once.values.Load(key)
69 if !ok {
70 panic(fmt.Errorf("Get() called before Once()"))
71 }
Colin Cross9272ade2016-08-17 15:24:12 -070072
Colin Crossd7cfaee2019-02-15 23:00:48 -080073 return once.maybeWaitFor(key, v)
Colin Cross9272ade2016-08-17 15:24:12 -070074}
75
Colin Cross571cccf2019-02-04 11:22:08 -080076// OnceStringSlice is the same as Once, but returns the value cast to a []string
Colin Crosse48ff5b2019-02-04 13:03:13 -080077func (once *OncePer) OnceStringSlice(key OnceKey, value func() []string) []string {
Colin Cross9272ade2016-08-17 15:24:12 -070078 return once.Once(key, func() interface{} { return value() }).([]string)
79}
80
Colin Cross571cccf2019-02-04 11:22:08 -080081// OnceStringSlice is the same as Once, but returns two values cast to []string
Colin Crosse48ff5b2019-02-04 13:03:13 -080082func (once *OncePer) Once2StringSlice(key OnceKey, value func() ([]string, []string)) ([]string, []string) {
Colin Cross9272ade2016-08-17 15:24:12 -070083 type twoStringSlice [2][]string
84 s := once.Once(key, func() interface{} {
85 var s twoStringSlice
86 s[0], s[1] = value()
87 return s
88 }).(twoStringSlice)
89 return s[0], s[1]
90}
Colin Cross571cccf2019-02-04 11:22:08 -080091
92// OnceKey is an opaque type to be used as the key in calls to Once.
93type OnceKey struct {
94 key interface{}
95}
96
97// NewOnceKey returns an opaque OnceKey object for the provided key. Two calls to NewOnceKey with the same key string
98// DO NOT produce the same OnceKey object.
99func NewOnceKey(key string) OnceKey {
100 return OnceKey{&key}
101}
102
103// NewCustomOnceKey returns an opaque OnceKey object for the provided key. The key can be any type that is valid as the
104// key in a map, i.e. comparable. Two calls to NewCustomOnceKey with key values that compare equal will return OnceKey
105// objects that access the same value stored with Once.
106func NewCustomOnceKey(key interface{}) OnceKey {
107 return OnceKey{key}
108}