Speed up adding children to the various Node lists (depends, ignore, sources, implicit).
authorstevenknight <stevenknight@fdb21ef1-2011-0410-befe-b5e4ea1792b1>
Sun, 20 Jul 2003 16:07:00 +0000 (16:07 +0000)
committerstevenknight <stevenknight@fdb21ef1-2011-0410-befe-b5e4ea1792b1>
Sun, 20 Jul 2003 16:07:00 +0000 (16:07 +0000)
git-svn-id: http://scons.tigris.org/svn/scons/trunk@738 fdb21ef1-2011-0410-befe-b5e4ea1792b1

bin/timebuild [new file with mode: 0644]
src/CHANGES.txt
src/engine/SCons/Node/FS.py
src/engine/SCons/Node/NodeTests.py
src/engine/SCons/Node/__init__.py

diff --git a/bin/timebuild b/bin/timebuild
new file mode 100644 (file)
index 0000000..d5af983
--- /dev/null
@@ -0,0 +1,65 @@
+#!/bin/sh
+#
+# Profile running SCons to build itself from the current package.
+#
+# This runs "aegis -build" to build a current scons-src-*.tar.gz
+# package, unpacks it in the supplied directory name, and then
+# starts a profiled run of an SCons build, followed by another.
+# This results in two profiles:
+#
+#       NAME/NAME-0.prof
+#               profile of a build-everything run
+#
+#       NAME/NAME-1.prof
+#               profile of an all-up-to-date run
+#
+# This also copies the build scons-src-*.tar.gz file to the NAME
+# subdirectory, and tars up everything under src/ as NAME/src.tar.gz,
+# so that repeated runs with different in-progress changes can serve
+# as their own crude version control, so you don't lose that exact
+# combination of features which performed best.
+
+if test X$1 = X; then
+    echo "Must supply name!" >&2
+    exit 1
+fi
+
+VERSION=0.90
+
+DIR=$1
+
+SRC="scons-src-$VERSION"
+SRC_TAR_GZ="${SRC}.tar.gz"
+B_D_SRC_TAR_GZ="build/dist/${SRC_TAR_GZ}"
+
+echo "Building ${B_D_SRC_TAR_GZ}: " `date`
+aegis -build ${B_D_SRC_TAR_GZ}
+
+echo "mkdir ${DIR}: " `date`
+mkdir ${DIR}
+
+echo "cp ${B_D_SRC_TAR_GZ} ${DIR}: " `date`
+cp ${B_D_SRC_TAR_GZ} ${DIR}
+
+echo "tar cf ${DIR}/src.tar.gz: " `date`
+tar cf ${DIR}/src.tar.gz src
+
+cd ${DIR}
+
+echo "tar zxf ${SRC_TAR_GZ}: " `date`
+tar zxf ${SRC_TAR_GZ}
+
+cd ${SRC}
+
+SCRIPT="src/script/scons.py"
+ARGS="version=$VERSION"
+
+export SCONS_LIB_DIR=`pwd`/src/engine
+
+echo "Build run starting: " `date`
+python $SCRIPT --profile=../$DIR-0.prof $ARGS > ../$DIR-0.log 2>&1
+
+echo "Up-to-date run starting: " `date`
+python $SCRIPT --profile=../$DIR-1.prof $ARGS > ../$DIR-1.log 2>&1
+
+echo "Finished $DIR at: " `date`
index a0666890b4702523fe8da89480d6d8a19f4eb524..3283760b74d2dc90d4c91701c4cc7096589fe39a 100644 (file)
@@ -28,6 +28,12 @@ RELEASE 0.XX - XXX
 
   - Add an "sconsign" script to print the contents of .sconsign files.
 
+  - Speed up maintaining the various lists of Node children by using
+    dictionaries to avoid "x in list" searches.
+
+  - Cache the computed list of Node children minus those being Ignored
+    so it's only calculated once.
+
   From Gary Oberbrunner:
 
   - Report the target being built in error messages when building
index c2d001e88262638beae6b4d2727e303b715714dc..1f883307dfd02069078efde09094ecccd8a572a5 100644 (file)
@@ -969,7 +969,11 @@ class Dir(Entry):
         else:
             return self.entries['..'].root()
 
-    def all_children(self, scan):
+    def children(self, scan=1):
+        return filter(lambda x, i=self.ignore: x not in i,
+                             self.all_children(scan))
+
+    def all_children(self, scan=1):
         keys = filter(lambda k: k != '.' and k != '..', self.entries.keys())
         kids = map(lambda x, s=self: s.entries[x], keys)
         def c(one, two):
index f8593d7ec331342d6e53ee3a10953c05cd397bbe..103cfae35c9bd0da594442aef0d589816f1a7b71 100644 (file)
@@ -429,20 +429,23 @@ class NodeTestCase(unittest.TestCase):
         n1 = SCons.Node.Node()
         n1.builder_set(Builder())
         node.implicit = []
-        node._add_child(node.implicit, [n1])
+        node.implicit_dict = {}
+        node._add_child(node.implicit, node.implicit_dict, [n1])
 
         node.prepare()  # should not throw an exception
 
         n2 = SCons.Node.Node()
         n2.linked = 1
         node.implicit = []
-        node._add_child(node.implicit, [n2])
+        node.implicit_dict = {}
+        node._add_child(node.implicit, node.implicit_dict, [n2])
 
         node.prepare()  # should not throw an exception
 
         n3 = SCons.Node.Node()
         node.implicit = []
-        node._add_child(node.implicit, [n3])
+        node.implicit_dict = {}
+        node._add_child(node.implicit, node.implicit_dict, [n3])
 
         node.prepare()  # should not throw an exception
 
@@ -451,7 +454,8 @@ class NodeTestCase(unittest.TestCase):
                 return None
         n4 = MyNode()
         node.implicit = []
-        node._add_child(node.implicit, [n4]) 
+        node.implicit_dict = {}
+        node._add_child(node.implicit, node.implicit_dict, [n4]) 
         exc_caught = 0
         try:
             node.prepare()
@@ -650,8 +654,9 @@ class NodeTestCase(unittest.TestCase):
         node.add_source([n1, n2, n3])
         node.add_dependency([n4, n5, n6])
         node.implicit = []
-        node._add_child(node.implicit, [n7, n8, n9])
-        node._add_child(node.implicit, [n10, n11, n12])
+        node.implicit_dict = {}
+        node._add_child(node.implicit, node.implicit_dict, [n7, n8, n9])
+        node._add_child(node.implicit, node.implicit_dict, [n10, n11, n12])
         node.add_ignore([n2, n5, n8, n11])
 
         kids = node.children()
@@ -680,8 +685,9 @@ class NodeTestCase(unittest.TestCase):
         node.add_source([n1, n2, n3])
         node.add_dependency([n4, n5, n6])
         node.implicit = []
-        node._add_child(node.implicit, [n7, n8, n9])
-        node._add_child(node.implicit, [n10, n11, n12])
+        node.implicit_dict = {}
+        node._add_child(node.implicit, node.implicit_dict, [n7, n8, n9])
+        node._add_child(node.implicit, node.implicit_dict, [n10, n11, n12])
         node.add_ignore([n2, n5, n8, n11])
 
         kids = node.all_children()
@@ -717,10 +723,14 @@ class NodeTestCase(unittest.TestCase):
         n1.add_source([n2, n3])
 
         nw = SCons.Node.Walker(n1)
-        assert nw.next().name ==  "n2"
-        assert nw.next().name ==  "n3"
-        assert nw.next().name ==  "n1"
-        assert nw.next() == None
+        n = nw.next()
+        assert n.name ==  "n2", n.name
+        n = nw.next()
+        assert n.name ==  "n3", n.name
+        n = nw.next()
+        assert n.name ==  "n1", n.name
+        n = nw.next()
+        assert n == None, n
 
         n4 = MyNode("n4")
         n5 = MyNode("n5")
index 0fc2365b4067ced934430ab13f3a1b621417f8a7..6947c93578dcc77fb609e931602711b804710bd7 100644 (file)
@@ -95,10 +95,20 @@ class Node:
         # canonical example being a builder to fetch a file from a
         # source code system like CVS or Subversion).
 
+        # Each list of children that we maintain is accompanied by a
+        # dictionary used to look up quickly whether a node is already
+        # present in the list.  Empirical tests showed that it was
+        # fastest to maintain them as side-by-side Node attributes in
+        # this way, instead of wrapping up each list+dictionary pair in
+        # a class.  (Of course, we could always still do that in the
+        # future if we had a good reason to...).
         self.sources = []       # source files used to build node
+        self.sources_dict = {}
         self.depends = []       # explicit dependencies (from Depends)
-        self.implicit = None    # implicit (scanned) dependencies (None means not scanned yet)
+        self.depends_dict = {}
         self.ignore = []        # dependencies to ignore
+        self.ignore_dict = {}
+        self.implicit = None    # implicit (scanned) dependencies (None means not scanned yet)
         self.parents = {}
         self.wkids = None       # Kids yet to walk, when it's an array
         self.source_scanner = None      # implicit scanner from scanner map
@@ -349,6 +359,7 @@ class Node:
         if not self.implicit is None:
             return
         self.implicit = []
+        self.implicit_dict = {}
         if not self.has_builder():
             return
 
@@ -356,7 +367,7 @@ class Node:
             implicit = self.get_stored_implicit()
             if implicit is not None:
                 implicit = map(self.implicit_factory, implicit)
-                self._add_child(self.implicit, implicit)
+                self._add_child(self.implicit, self.implicit_dict, implicit)
                 calc = SCons.Sig.default_calc
                 if implicit_deps_unchanged or calc.current(self, calc.bsig(self)):
                     return
@@ -365,18 +376,21 @@ class Node:
                     # we need to recalculate the implicit deps,
                     # and the bsig:
                     self.implicit = []
+                    self.implicit_dict = {}
                     self.del_bsig()
 
         build_env = self.get_build_env()
 
         for child in self.children(scan=0):
             self._add_child(self.implicit,
+                            self.implicit_dict,
                             child.get_implicit_deps(build_env,
                                                     child.source_scanner,
                                                     self))
 
         # scan this node itself for implicit dependencies
         self._add_child(self.implicit,
+                        self.implicit_dict,
                         self.get_implicit_deps(build_env,
                                                self.target_scanner,
                                                self))
@@ -557,26 +571,33 @@ class Node:
 
     def add_dependency(self, depend):
         """Adds dependencies. The depend argument must be a list."""
-        self._add_child(self.depends, depend)
+        self._add_child(self.depends, self.depends_dict, depend)
 
     def add_ignore(self, depend):
         """Adds dependencies to ignore. The depend argument must be a list."""
-        self._add_child(self.ignore, depend)
+        self._add_child(self.ignore, self.ignore_dict, depend)
 
     def add_source(self, source):
         """Adds sources. The source argument must be a list."""
-        self._add_child(self.sources, source)
+        self._add_child(self.sources, self.sources_dict, source)
 
-    def _add_child(self, collection, child):
-        """Adds 'child' to 'collection'. The 'child' argument must be a list"""
+    def _add_child(self, collection, dict, child):
+        """Adds 'child' to 'collection', first checking 'dict' to see if
+        it's already present. The 'child' argument must be a list"""
         if type(child) is not type([]):
             raise TypeError("child must be a list")
-        child = filter(lambda x, s=collection: x not in s, child)
-        if child:
-            collection.extend(child)
-
+        added = None
         for c in child:
+            if not dict.has_key(c):
+                collection.append(c)
+                dict[c] = 1
+                added = 1
             c.parents[self] = 1
+        if added:
+            try:
+                delattr(self, '_children')
+            except AttributeError:
+                pass
 
     def add_wkid(self, wkid):
         """Add a node to the list of kids waiting to be evaluated"""
@@ -586,8 +607,15 @@ class Node:
     def children(self, scan=1):
         """Return a list of the node's direct children, minus those
         that are ignored by this node."""
-        return filter(lambda x, i=self.ignore: x not in i,
-                      self.all_children(scan))
+        if scan:
+            self.scan()
+        try:
+            return self._children
+        except AttributeError:
+            c = filter(lambda x, i=self.ignore: x not in i,
+                              self.all_children(scan=0))
+            self._children = c
+            return c
 
     def all_children(self, scan=1):
         """Return a list of all the node's direct children."""