From 7c1c3a6a063550f83f0355feb5130220e36cf017 Mon Sep 17 00:00:00 2001 From: Raphael Vigee Date: Sat, 11 May 2024 16:41:01 +0100 Subject: [PATCH] Cache deepDo --- utils/sets/set.go | 7 +++++++ worker2/dep_utils.go | 49 +++++++++++++++++++++++++++++--------------- 2 files changed, 40 insertions(+), 16 deletions(-) diff --git a/utils/sets/set.go b/utils/sets/set.go index 0a7a5e9e..333a3a23 100644 --- a/utils/sets/set.go +++ b/utils/sets/set.go @@ -110,6 +110,9 @@ func (ts *Set[K, T]) Slice() []T { return nil } + ts.mu.RLock() + defer ts.mu.RUnlock() + return ts.a[:] } @@ -118,6 +121,9 @@ func (ts *Set[K, T]) Len() int { return 0 } + ts.mu.RLock() + defer ts.mu.RUnlock() + return len(ts.a) } @@ -143,6 +149,7 @@ func (ts *Set[K, T]) Remove(v T) { k := ts.f(v) delete(ts.m, k) + ts.a = slices.DeleteFunc(ts.a, func(t T) bool { return ts.f(t) == k }) diff --git a/worker2/dep_utils.go b/worker2/dep_utils.go index be7fac5e..d73291eb 100644 --- a/worker2/dep_utils.go +++ b/worker2/dep_utils.go @@ -6,6 +6,16 @@ import ( "sync/atomic" ) +func deepDo(a Dep, f func(Dep)) { + if a.GetNode().IsFrozen() { + // This approach sounds good on paper, but in reality very CPU intensive since it requires + // read & write to the deps set at every change, at every level... + deepDoPrecomputed(a, f) + } else { + deepDoRecursive(a, f) + } +} + func deepDoPrecomputed(a Dep, f func(Dep)) { f(a) for _, dep := range a.GetNode().Dependencies.TransitiveValues() { @@ -17,32 +27,39 @@ var deepDoMapPool = xsync.Pool[map[Dep]struct{}]{New: func() map[Dep]struct{} { return map[Dep]struct{}{} }} -func deepDo(a Dep, f func(Dep)) { - if false { - // This approach sounds good on paper, but in reality very CPU intensive since it requires - // read & write to the deps set at every change, at every level... - deepDoPrecomputed(a, f) - } else { - deepDoRecursive(a, f) - } -} - func deepDoRecursive(a Dep, f func(Dep)) { m := deepDoMapPool.Get() - maps.Clear(m) - defer deepDoMapPool.Put(m) + defer func() { + maps.Clear(m) + deepDoMapPool.Put(m) + }() deepDoRecursiveInner(a, f, m) } -func deepDoRecursiveInner(a Dep, f func(Dep), m map[Dep]struct{}) { +func deepDoInner(a Dep, f func(Dep), m map[Dep]struct{}) bool { if _, ok := m[a]; ok { - return + return false } m[a] = struct{}{} f(a) - for _, dep := range a.GetNode().Dependencies.Values() { - deepDoRecursiveInner(dep, f, m) + + return true +} + +func deepDoRecursiveInner(a Dep, f func(Dep), m map[Dep]struct{}) { + if !deepDoInner(a, f, m) { + return + } + + if a.GetNode().IsFrozen() { + for _, dep := range a.GetNode().Dependencies.TransitiveValues() { + deepDoInner(dep, f, m) + } + } else { + for _, dep := range a.GetNode().Dependencies.Values() { + deepDoRecursiveInner(dep, f, m) + } } }