internal/restic: Don't allocate in Tree.Insert

name         old time/op    new time/op    delta
BuildTree-8    34.6µs ± 4%     7.0µs ± 3%  -79.68%  (p=0.000 n=18+19)

name         old alloc/op   new alloc/op   delta
BuildTree-8    34.0kB ± 0%     0.9kB ± 0%  -97.37%  (p=0.000 n=20+20)

name         old allocs/op  new allocs/op  delta
BuildTree-8       108 ± 0%         1 ± 0%  -99.07%  (p=0.000 n=20+20)
This commit is contained in:
greatroar 2021-09-26 17:18:42 +02:00
parent 78dac2fd48
commit c892c0bab9
6 changed files with 35 additions and 11 deletions

View file

@ -107,7 +107,7 @@ func runRecover(gopts GlobalOptions) error {
return nil
}
tree := restic.NewTree()
tree := restic.NewTree(len(roots))
for id := range roots {
var subtreeID = id
node := restic.Node{

View file

@ -550,12 +550,13 @@ func (arch *Archiver) statDir(dir string) (os.FileInfo, error) {
func (arch *Archiver) SaveTree(ctx context.Context, snPath string, atree *Tree, previous *restic.Tree) (*restic.Tree, error) {
debug.Log("%v (%v nodes), parent %v", snPath, len(atree.Nodes), previous)
tree := restic.NewTree()
nodeNames := atree.NodeNames()
tree := restic.NewTree(len(nodeNames))
futureNodes := make(map[string]FutureNode)
// iterate over the nodes of atree in lexicographic (=deterministic) order
for _, name := range atree.NodeNames() {
for _, name := range nodeNames {
subatree := atree.Nodes[name]
// test if context has been cancelled

View file

@ -103,7 +103,8 @@ type saveTreeResponse struct {
func (s *TreeSaver) save(ctx context.Context, snPath string, node *restic.Node, nodes []FutureNode) (*restic.Node, ItemStats, error) {
var stats ItemStats
tree := restic.NewTree()
tree := restic.NewTree(len(nodes))
for _, fn := range nodes {
fn.wait(ctx)

View file

@ -14,10 +14,10 @@ type Tree struct {
Nodes []*Node `json:"nodes"`
}
// NewTree creates a new tree object.
func NewTree() *Tree {
// NewTree creates a new tree object with the given initial capacity.
func NewTree(capacity int) *Tree {
return &Tree{
Nodes: []*Node{},
Nodes: make([]*Node, 0, capacity),
}
}
@ -51,8 +51,8 @@ func (t *Tree) Insert(node *Node) error {
return errors.Errorf("node %q already present", node.Name)
}
// https://code.google.com/p/go-wiki/wiki/SliceTricks
t.Nodes = append(t.Nodes, &Node{})
// https://github.com/golang/go/wiki/SliceTricks
t.Nodes = append(t.Nodes, nil)
copy(t.Nodes[pos+1:], t.Nodes[pos:])
t.Nodes[pos] = node

View file

@ -6,6 +6,7 @@ import (
"io/ioutil"
"os"
"path/filepath"
"strconv"
"testing"
"github.com/restic/restic/internal/repository"
@ -98,7 +99,7 @@ func TestLoadTree(t *testing.T) {
defer cleanup()
// save tree
tree := restic.NewTree()
tree := restic.NewTree(0)
id, err := repo.SaveTree(context.TODO(), tree)
rtest.OK(t, err)
@ -113,3 +114,24 @@ func TestLoadTree(t *testing.T) {
"trees are not equal: want %v, got %v",
tree, tree2)
}
func BenchmarkBuildTree(b *testing.B) {
const size = 100 // Directories of this size are not uncommon.
nodes := make([]restic.Node, size)
for i := range nodes {
// Archiver.SaveTree inputs in sorted order, so do that here too.
nodes[i].Name = strconv.Itoa(i)
}
b.ResetTimer()
b.ReportAllocs()
for i := 0; i < b.N; i++ {
t := restic.NewTree(size)
for i := range nodes {
_ = t.Insert(&nodes[i])
}
}
}

View file

@ -23,7 +23,7 @@ func BuildTreeMap(tree TestTree) (m TreeMap, root restic.ID) {
}
func buildTreeMap(tree TestTree, m TreeMap) restic.ID {
res := restic.NewTree()
res := restic.NewTree(0)
for name, item := range tree {
switch elem := item.(type) {