aboutsummaryrefslogtreecommitdiff
path: root/build/algorithm.cxx
diff options
context:
space:
mode:
Diffstat (limited to 'build/algorithm.cxx')
-rw-r--r--build/algorithm.cxx93
1 files changed, 93 insertions, 0 deletions
diff --git a/build/algorithm.cxx b/build/algorithm.cxx
new file mode 100644
index 0000000..fac0bf6
--- /dev/null
+++ b/build/algorithm.cxx
@@ -0,0 +1,93 @@
+// file : build/algorithm.cxx -*- C++ -*-
+// copyright : Copyright (c) 2014-2015 Code Synthesis Tools CC
+// license : MIT; see accompanying LICENSE file
+
+#include <build/algorithm>
+
+#include <memory> // unique_ptr
+#include <utility> // move
+#include <cassert>
+#include <iostream>
+
+#include <build/path>
+#include <build/scope>
+#include <build/target>
+#include <build/prerequisite>
+#include <build/rule>
+#include <build/diagnostics>
+
+using namespace std;
+
+namespace build
+{
+ target*
+ search (prerequisite& p)
+ {
+ assert (p.target == nullptr);
+
+ //@@ TODO for now we just default to the directory scope.
+ //
+ path d;
+ if (p.directory.absolute ())
+ d = p.directory; // Already normalized.
+ else
+ {
+ d = p.scope.path () / p.directory;
+ d.normalize ();
+ }
+
+ //@@ TODO would be nice to first check if this target is
+ // already in the set before allocating a new instance.
+
+ // Find or insert.
+ //
+ auto r (
+ targets.emplace (
+ unique_ptr<target> (p.type.factory (p.name, move (d)))));
+
+ //if (r.second)
+ // cout << "new target for prerequsite " << p << " " << d << endl;
+
+ return (p.target = r.first->get ());
+ }
+
+ bool
+ match (target& t)
+ {
+ assert (!t.recipe ());
+
+ for (auto tt (&t.type ());
+ tt != nullptr && !t.recipe ();
+ tt = tt->base)
+ {
+ for (auto rs (rules.equal_range (tt->id));
+ rs.first != rs.second;
+ ++rs.first)
+ {
+ const rule& ru (rs.first->second);
+
+ recipe re;
+
+ {
+ auto g (
+ make_exception_guard (
+ [] (target& t)
+ {
+ cerr << "info: while matching a rule for target " << t << endl;
+ },
+ t));
+
+ re = ru.match (t);
+ }
+
+ if (re)
+ {
+ t.recipe (re);
+ break;
+ }
+ }
+ }
+
+ return bool (t.recipe ());
+ }
+}