Commit 39382793 authored by Jack Lindamood's avatar Jack Lindamood Committed by Brad Fitzpatrick

context: reduce memory usage of context tree

Modifies context package to use map[]struct{} rather than map[]bool,
since the map is intended as a set object.  Also adds Benchmarks to
the context package switching between different types of root nodes
and a tree with different depths.

Included below are bytes deltas between the old and new code, using
these benchmarks.

benchmark                                                       old bytes     new bytes     delta
BenchmarkContextCancelTree/depth=1/Root=Background-8            176           176           +0.00%
BenchmarkContextCancelTree/depth=1/Root=OpenCanceler-8          560           544           -2.86%
BenchmarkContextCancelTree/depth=1/Root=ClosedCanceler-8        352           352           +0.00%
BenchmarkContextCancelTree/depth=10/Root=Background-8           3632          3488          -3.96%
BenchmarkContextCancelTree/depth=10/Root=OpenCanceler-8         4016          3856          -3.98%
BenchmarkContextCancelTree/depth=10/Root=ClosedCanceler-8       1936          1936          +0.00%
BenchmarkContextCancelTree/depth=100/Root=Background-8          38192         36608         -4.15%
BenchmarkContextCancelTree/depth=100/Root=OpenCanceler-8        38576         36976         -4.15%
BenchmarkContextCancelTree/depth=100/Root=ClosedCanceler-8      17776         17776         +0.00%
BenchmarkContextCancelTree/depth=1000/Root=Background-8         383792        367808        -4.16%
BenchmarkContextCancelTree/depth=1000/Root=OpenCanceler-8       384176        368176        -4.16%
BenchmarkContextCancelTree/depth=1000/Root=ClosedCanceler-8     176176        176176        +0.00%

Change-Id: I699ad704d9f7b461214e1651d24941927315b525
Reviewed-on: https://go-review.googlesource.com/25270Reviewed-by: default avatarBrad Fitzpatrick <bradfitz@golang.org>
Run-TryBot: Brad Fitzpatrick <bradfitz@golang.org>
parent 0ab6bb42
...@@ -252,9 +252,9 @@ func propagateCancel(parent Context, child canceler) { ...@@ -252,9 +252,9 @@ func propagateCancel(parent Context, child canceler) {
child.cancel(false, p.err) child.cancel(false, p.err)
} else { } else {
if p.children == nil { if p.children == nil {
p.children = make(map[canceler]bool) p.children = make(map[canceler]struct{})
} }
p.children[child] = true p.children[child] = struct{}{}
} }
p.mu.Unlock() p.mu.Unlock()
} else { } else {
...@@ -314,8 +314,8 @@ type cancelCtx struct { ...@@ -314,8 +314,8 @@ type cancelCtx struct {
done chan struct{} // closed by the first cancel call. done chan struct{} // closed by the first cancel call.
mu sync.Mutex mu sync.Mutex
children map[canceler]bool // set to nil by the first cancel call children map[canceler]struct{} // set to nil by the first cancel call
err error // set to non-nil by the first cancel call err error // set to non-nil by the first cancel call
} }
func (c *cancelCtx) Done() <-chan struct{} { func (c *cancelCtx) Done() <-chan struct{} {
......
...@@ -92,6 +92,11 @@ func TestWithCancel(t *testing.T) { ...@@ -92,6 +92,11 @@ func TestWithCancel(t *testing.T) {
} }
} }
func contains(m map[canceler]struct{}, key canceler) bool {
_, ret := m[key]
return ret
}
func TestParentFinishesChild(t *testing.T) { func TestParentFinishesChild(t *testing.T) {
// Context tree: // Context tree:
// parent -> cancelChild // parent -> cancelChild
...@@ -120,7 +125,7 @@ func TestParentFinishesChild(t *testing.T) { ...@@ -120,7 +125,7 @@ func TestParentFinishesChild(t *testing.T) {
cc := cancelChild.(*cancelCtx) cc := cancelChild.(*cancelCtx)
tc := timerChild.(*timerCtx) tc := timerChild.(*timerCtx)
pc.mu.Lock() pc.mu.Lock()
if len(pc.children) != 2 || !pc.children[cc] || !pc.children[tc] { if len(pc.children) != 2 || !contains(pc.children, cc) || !contains(pc.children, tc) {
t.Errorf("bad linkage: pc.children = %v, want %v and %v", t.Errorf("bad linkage: pc.children = %v, want %v and %v",
pc.children, cc, tc) pc.children, cc, tc)
} }
...@@ -191,7 +196,7 @@ func TestChildFinishesFirst(t *testing.T) { ...@@ -191,7 +196,7 @@ func TestChildFinishesFirst(t *testing.T) {
if pcok { if pcok {
pc.mu.Lock() pc.mu.Lock()
if len(pc.children) != 1 || !pc.children[cc] { if len(pc.children) != 1 || !contains(pc.children, cc) {
t.Errorf("bad linkage: pc.children = %v, cc = %v", pc.children, cc) t.Errorf("bad linkage: pc.children = %v, cc = %v", pc.children, cc)
} }
pc.mu.Unlock() pc.mu.Unlock()
...@@ -627,3 +632,36 @@ func TestDeadlineExceededSupportsTimeout(t *testing.T) { ...@@ -627,3 +632,36 @@ func TestDeadlineExceededSupportsTimeout(t *testing.T) {
t.Fatal("wrong value for timeout") t.Fatal("wrong value for timeout")
} }
} }
func BenchmarkContextCancelTree(b *testing.B) {
depths := []int{1, 10, 100, 1000}
for _, d := range depths {
b.Run(fmt.Sprintf("depth=%d", d), func(b *testing.B) {
b.Run("Root=Background", func(b *testing.B) {
for i := 0; i < b.N; i++ {
buildContextTree(Background(), d)
}
})
b.Run("Root=OpenCanceler", func(b *testing.B) {
for i := 0; i < b.N; i++ {
ctx, cancel := WithCancel(Background())
buildContextTree(ctx, d)
cancel()
}
})
b.Run("Root=ClosedCanceler", func(b *testing.B) {
for i := 0; i < b.N; i++ {
ctx, cancel := WithCancel(Background())
cancel()
buildContextTree(ctx, d)
}
})
})
}
}
func buildContextTree(root Context, depth int) {
for d := 0; d < depth; d++ {
root, _ = WithCancel(root)
}
}
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