1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
|
// file : build/target.cxx -*- C++ -*-
// copyright : Copyright (c) 2014-2015 Code Synthesis Tools CC
// license : MIT; see accompanying LICENSE file
#include <build/target>
#include <build/context>
#include <build/diagnostics>
using namespace std;
namespace build
{
// target_type
//
const target_type*
resolve_target_type (const std::string name);
// target
//
ostream&
operator<< (ostream& os, const target& t)
{
os << t.type ().name << '{';
if (!t.directory.empty ())
{
string s (diagnostic_string (t.directory));
if (!s.empty ())
{
os << s;
if (!t.name.empty ())
os << path::traits::directory_separator;
}
}
os << t.name;
if (t.ext != nullptr)
os << '.' << *t.ext;
os << '}';
return os;
}
// target_set
//
auto target_set::
insert (const target_type& tt,
path dir,
std::string name,
const std::string* ext,
tracer& trace) -> pair<target&, bool>
{
//@@ OPT: would be nice to somehow first check if this target is
// already in the set before allocating a new instance.
// Find or insert.
//
auto r (
emplace (
unique_ptr<target> (tt.factory (move (dir), move (name), ext))));
target& t (**r.first);
// Update the extension if the existing target has it unspecified.
//
if (t.ext != ext)
{
level4 ([&]{
diag_record r (trace);
r << "assuming target " << t << " is the same as the one with ";
if (ext == nullptr)
r << "unspecified extension";
else if (ext->empty ())
r << "no extension";
else
r << "extension " << *ext;
});
if (ext != nullptr)
t.ext = ext;
}
return pair<target&, bool> (t, r.second);
}
target_set targets;
target* default_target = nullptr;
target_type_map target_types;
// path_target
//
timestamp path_target::
load_mtime () const
{
assert (!path_.empty ());
return path_mtime (path_);
}
const target_type target::static_type {
typeid (target), "target", nullptr, nullptr};
const target_type mtime_target::static_type {
typeid (mtime_target), "mtime_target", &target::static_type, nullptr};
const target_type path_target::static_type {
typeid (path_target), "path_target", &mtime_target::static_type, nullptr};
const target_type file::static_type {
typeid (file), "file", &path_target::static_type, &target_factory<file>};
const target_type dir::static_type {
typeid (dir), "dir", &target::static_type, &target_factory<dir>};
}
|