1 // Copyright (c) 2013 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4
5 #include "tools/gn/ninja_helper.h"
6
7 #include "base/logging.h"
8 #include "tools/gn/filesystem_utils.h"
9 #include "tools/gn/string_utils.h"
10 #include "tools/gn/target.h"
11
12 namespace {
13
14 const char kLibDirWithSlash[] = "lib/";
15 const char kObjectDirNoSlash[] = "obj";
16
17 } // namespace
18
NinjaHelper(const BuildSettings * build_settings)19 NinjaHelper::NinjaHelper(const BuildSettings* build_settings)
20 : build_settings_(build_settings) {
21 build_to_src_no_last_slash_ = build_settings->build_to_source_dir_string();
22 if (!build_to_src_no_last_slash_.empty() &&
23 build_to_src_no_last_slash_[build_to_src_no_last_slash_.size() - 1] ==
24 '/')
25 build_to_src_no_last_slash_.resize(build_to_src_no_last_slash_.size() - 1);
26
27 build_to_src_system_no_last_slash_ = build_to_src_no_last_slash_;
28 ConvertPathToSystem(&build_to_src_system_no_last_slash_);
29 }
30
~NinjaHelper()31 NinjaHelper::~NinjaHelper() {
32 }
33
GetTopleveOutputDir() const34 std::string NinjaHelper::GetTopleveOutputDir() const {
35 return kObjectDirNoSlash;
36 }
37
GetTargetOutputDir(const Target * target) const38 std::string NinjaHelper::GetTargetOutputDir(const Target* target) const {
39 return kObjectDirNoSlash + target->label().dir().SourceAbsoluteWithOneSlash();
40 }
41
GetNinjaFileForTarget(const Target * target) const42 OutputFile NinjaHelper::GetNinjaFileForTarget(const Target* target) const {
43 OutputFile ret(target->settings()->toolchain_output_subdir());
44 ret.value().append(kObjectDirNoSlash);
45 AppendStringPiece(&ret.value(),
46 target->label().dir().SourceAbsoluteWithOneSlash());
47 ret.value().append(target->label().name());
48 ret.value().append(".ninja");
49 return ret;
50 }
51
GetNinjaFileForToolchain(const Settings * settings) const52 OutputFile NinjaHelper::GetNinjaFileForToolchain(
53 const Settings* settings) const {
54 OutputFile ret;
55 ret.value().append(settings->toolchain_output_subdir().value());
56 ret.value().append("toolchain.ninja");
57 return ret;
58 }
59
60 // In Python, GypPathToUniqueOutput does the qualification. The only case where
61 // the Python version doesn't qualify the name is for target outputs, which we
62 // handle in a separate function.
GetOutputFileForSource(const Target * target,const SourceFile & source,SourceFileType type) const63 OutputFile NinjaHelper::GetOutputFileForSource(
64 const Target* target,
65 const SourceFile& source,
66 SourceFileType type) const {
67 // Extract the filename and remove the extension (keep the dot).
68 base::StringPiece filename = FindFilename(&source.value());
69 std::string name(filename.data(), filename.size());
70 size_t extension_offset = FindExtensionOffset(name);
71 CHECK(extension_offset != std::string::npos);
72 name.resize(extension_offset);
73
74 // Append the new extension.
75 switch (type) {
76 case SOURCE_ASM:
77 case SOURCE_C:
78 case SOURCE_CC:
79 case SOURCE_M:
80 case SOURCE_MM:
81 case SOURCE_S:
82 name.append(target->settings()->IsWin() ? "obj" : "o");
83 break;
84
85 case SOURCE_RC:
86 name.append("res");
87 break;
88
89 case SOURCE_H:
90 case SOURCE_UNKNOWN:
91 NOTREACHED();
92 return OutputFile();
93 }
94
95 // Use the scheme <path>/<target>.<name>.<extension> so that all output
96 // names are unique to different targets.
97
98 // This will look like "obj" or "toolchain_name/obj".
99 OutputFile ret(target->settings()->toolchain_output_subdir());
100 ret.value().append(kObjectDirNoSlash);
101
102 // Find the directory, assume it starts with two slashes, and trim to one.
103 base::StringPiece dir = FindDir(&source.value());
104 CHECK(dir.size() >= 2 && dir[0] == '/' && dir[1] == '/')
105 << "Source file isn't in the source repo: " << dir;
106 AppendStringPiece(&ret.value(), dir.substr(1));
107
108 ret.value().append(target->label().name());
109 ret.value().append(".");
110 ret.value().append(name);
111 return ret;
112 }
113
GetTargetOutputFile(const Target * target) const114 OutputFile NinjaHelper::GetTargetOutputFile(const Target* target) const {
115 OutputFile ret;
116
117 // Use the output name if given, fall back to target name if not.
118 const std::string& name = target->output_name().empty() ?
119 target->label().name() : target->output_name();
120
121 // This is prepended to the output file name. Some platforms get "lib"
122 // prepended to library names. but be careful not to make a duplicate (e.g.
123 // some targets like "libxml" already have the "lib" in the name).
124 const char* prefix;
125 if (!target->settings()->IsWin() &&
126 (target->output_type() == Target::SHARED_LIBRARY ||
127 target->output_type() == Target::STATIC_LIBRARY) &&
128 name.compare(0, 3, "lib") != 0)
129 prefix = "lib";
130 else
131 prefix = "";
132
133 const char* extension;
134 if (target->output_type() == Target::GROUP ||
135 target->output_type() == Target::SOURCE_SET ||
136 target->output_type() == Target::COPY_FILES ||
137 target->output_type() == Target::CUSTOM) {
138 extension = "stamp";
139 } else {
140 extension = GetExtensionForOutputType(target->output_type(),
141 target->settings()->target_os());
142 }
143
144 // Everything goes into the toolchain directory (which will be empty for the
145 // default toolchain, and will end in a slash otherwise).
146 ret.value().append(target->settings()->toolchain_output_subdir().value());
147
148 // Binaries and loadable libraries go into the toolchain root.
149 if (target->output_type() == Target::EXECUTABLE ||
150 (target->settings()->IsMac() &&
151 (target->output_type() == Target::SHARED_LIBRARY ||
152 target->output_type() == Target::STATIC_LIBRARY)) ||
153 (target->settings()->IsWin() &&
154 target->output_type() == Target::SHARED_LIBRARY)) {
155 // Generate a name like "<toolchain>/<prefix><name>.<extension>".
156 ret.value().append(prefix);
157 ret.value().append(name);
158 if (extension[0]) {
159 ret.value().push_back('.');
160 ret.value().append(extension);
161 }
162 return ret;
163 }
164
165 // Libraries go into the library subdirectory like
166 // "<toolchain>/lib/<prefix><name>.<extension>".
167 if (target->output_type() == Target::SHARED_LIBRARY) {
168 ret.value().append(kLibDirWithSlash);
169 ret.value().append(prefix);
170 ret.value().append(name);
171 if (extension[0]) {
172 ret.value().push_back('.');
173 ret.value().append(extension);
174 }
175 return ret;
176 }
177
178 // Everything else goes next to the target's .ninja file like
179 // "<toolchain>/obj/<path>/<name>.<extension>".
180 ret.value().append(kObjectDirNoSlash);
181 AppendStringPiece(&ret.value(),
182 target->label().dir().SourceAbsoluteWithOneSlash());
183 ret.value().append(prefix);
184 ret.value().append(name);
185 if (extension[0]) {
186 ret.value().push_back('.');
187 ret.value().append(extension);
188 }
189 return ret;
190 }
191
GetRulePrefix(const Settings * settings) const192 std::string NinjaHelper::GetRulePrefix(const Settings* settings) const {
193 // Don't prefix the default toolchain so it looks prettier, prefix everything
194 // else.
195 if (settings->is_default())
196 return std::string(); // Default toolchain has no prefix.
197 return settings->toolchain_label().name() + "_";
198 }
199
GetRuleForSourceType(const Settings * settings,SourceFileType type) const200 std::string NinjaHelper::GetRuleForSourceType(const Settings* settings,
201 SourceFileType type) const {
202 // This function may be hot since it will be called for every source file
203 // in the tree. We could cache the results to avoid making a string for
204 // every invocation.
205 std::string prefix = GetRulePrefix(settings);
206
207 if (type == SOURCE_C)
208 return prefix + "cc";
209 if (type == SOURCE_CC)
210 return prefix + "cxx";
211
212 // TODO(brettw) asm files.
213
214 if (settings->IsMac()) {
215 if (type == SOURCE_M)
216 return prefix + "objc";
217 if (type == SOURCE_MM)
218 return prefix + "objcxx";
219 }
220
221 if (settings->IsWin()) {
222 if (type == SOURCE_RC)
223 return prefix + "rc";
224 } else {
225 if (type == SOURCE_S)
226 return prefix + "cc"; // Assembly files just get compiled by CC.
227 }
228
229 return std::string();
230 }
231