You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
142 lines
2.7 KiB
Go
142 lines
2.7 KiB
Go
4 years ago
|
// SPDX-License-Identifier: Unlicense OR MIT
|
||
|
|
||
|
package gpu
|
||
|
|
||
|
import (
|
||
|
"fmt"
|
||
|
|
||
|
"gioui.org/f32"
|
||
|
"gioui.org/internal/ops"
|
||
|
)
|
||
|
|
||
|
type resourceCache struct {
|
||
|
res map[interface{}]resource
|
||
|
newRes map[interface{}]resource
|
||
|
}
|
||
|
|
||
|
// opCache is like a resourceCache but using concrete types and a
|
||
|
// freelist instead of two maps to avoid runtime.mapaccess2 calls
|
||
|
// since benchmarking showed them as a bottleneck.
|
||
|
type opCache struct {
|
||
|
// store the index + 1 in cache this key is stored in
|
||
|
index map[ops.Key]int
|
||
|
// list of indexes in cache that are free and can be used
|
||
|
freelist []int
|
||
|
cache []opCacheValue
|
||
|
}
|
||
|
|
||
|
type opCacheValue struct {
|
||
|
data pathData
|
||
|
bounds f32.Rectangle
|
||
|
// the fields below are handled by opCache
|
||
|
key ops.Key
|
||
|
keep bool
|
||
|
}
|
||
|
|
||
|
func newResourceCache() *resourceCache {
|
||
|
return &resourceCache{
|
||
|
res: make(map[interface{}]resource),
|
||
|
newRes: make(map[interface{}]resource),
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func (r *resourceCache) get(key interface{}) (resource, bool) {
|
||
|
v, exists := r.res[key]
|
||
|
if exists {
|
||
|
r.newRes[key] = v
|
||
|
}
|
||
|
return v, exists
|
||
|
}
|
||
|
|
||
|
func (r *resourceCache) put(key interface{}, val resource) {
|
||
|
if _, exists := r.newRes[key]; exists {
|
||
|
panic(fmt.Errorf("key exists, %p", key))
|
||
|
}
|
||
|
r.res[key] = val
|
||
|
r.newRes[key] = val
|
||
|
}
|
||
|
|
||
|
func (r *resourceCache) frame() {
|
||
|
for k, v := range r.res {
|
||
|
if _, exists := r.newRes[k]; !exists {
|
||
|
delete(r.res, k)
|
||
|
v.release()
|
||
|
}
|
||
|
}
|
||
|
for k, v := range r.newRes {
|
||
|
delete(r.newRes, k)
|
||
|
r.res[k] = v
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func (r *resourceCache) release() {
|
||
|
for _, v := range r.newRes {
|
||
|
v.release()
|
||
|
}
|
||
|
r.newRes = nil
|
||
|
r.res = nil
|
||
|
}
|
||
|
|
||
|
func newOpCache() *opCache {
|
||
|
return &opCache{
|
||
|
index: make(map[ops.Key]int),
|
||
|
freelist: make([]int, 0),
|
||
|
cache: make([]opCacheValue, 0),
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func (r *opCache) get(key ops.Key) (o opCacheValue, exist bool) {
|
||
|
v := r.index[key]
|
||
|
if v == 0 {
|
||
|
return
|
||
|
}
|
||
|
r.cache[v-1].keep = true
|
||
|
return r.cache[v-1], true
|
||
|
}
|
||
|
|
||
|
func (r *opCache) put(key ops.Key, val opCacheValue) {
|
||
|
v := r.index[key]
|
||
|
val.keep = true
|
||
|
val.key = key
|
||
|
if v == 0 {
|
||
|
// not in cache
|
||
|
i := len(r.cache)
|
||
|
if len(r.freelist) > 0 {
|
||
|
i = r.freelist[len(r.freelist)-1]
|
||
|
r.freelist = r.freelist[:len(r.freelist)-1]
|
||
|
r.cache[i] = val
|
||
|
} else {
|
||
|
r.cache = append(r.cache, val)
|
||
|
}
|
||
|
r.index[key] = i + 1
|
||
|
} else {
|
||
|
r.cache[v-1] = val
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func (r *opCache) frame() {
|
||
|
r.freelist = r.freelist[:0]
|
||
|
for i, v := range r.cache {
|
||
|
r.cache[i].keep = false
|
||
|
if v.keep {
|
||
|
continue
|
||
|
}
|
||
|
if v.data.data != nil {
|
||
|
v.data.release()
|
||
|
r.cache[i].data.data = nil
|
||
|
}
|
||
|
delete(r.index, v.key)
|
||
|
r.freelist = append(r.freelist, i)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func (r *opCache) release() {
|
||
|
for i := range r.cache {
|
||
|
r.cache[i].keep = false
|
||
|
}
|
||
|
r.frame()
|
||
|
r.index = nil
|
||
|
r.freelist = nil
|
||
|
r.cache = nil
|
||
|
}
|