1 // Copyright (c) 2016 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 "gn/qt_creator_writer.h"
6
7 #include <optional>
8 #include <set>
9 #include <sstream>
10 #include <string>
11
12 #include "base/files/file_path.h"
13 #include "base/files/file_util.h"
14 #include "base/strings/utf_string_conversions.h"
15
16 #include "gn/builder.h"
17 #include "gn/config_values_extractors.h"
18 #include "gn/deps_iterator.h"
19 #include "gn/filesystem_utils.h"
20 #include "gn/label.h"
21 #include "gn/loader.h"
22 #include "gn/string_output_buffer.h"
23
24 namespace {
25 base::FilePath::CharType kProjectDirName[] =
26 FILE_PATH_LITERAL("qtcreator_project");
27 base::FilePath::CharType kProjectName[] = FILE_PATH_LITERAL("all");
28 base::FilePath::CharType kMainProjectFileSuffix[] =
29 FILE_PATH_LITERAL(".creator");
30 base::FilePath::CharType kSourcesFileSuffix[] = FILE_PATH_LITERAL(".files");
31 base::FilePath::CharType kIncludesFileSuffix[] = FILE_PATH_LITERAL(".includes");
32 base::FilePath::CharType kDefinesFileSuffix[] = FILE_PATH_LITERAL(".config");
33 } // namespace
34
35 // static
RunAndWriteFile(const BuildSettings * build_settings,const Builder & builder,Err * err,const std::string & root_target)36 bool QtCreatorWriter::RunAndWriteFile(const BuildSettings* build_settings,
37 const Builder& builder,
38 Err* err,
39 const std::string& root_target) {
40 base::FilePath project_dir =
41 build_settings->GetFullPath(build_settings->build_dir())
42 .Append(kProjectDirName);
43 if (!base::DirectoryExists(project_dir)) {
44 base::File::Error error;
45 if (!base::CreateDirectoryAndGetError(project_dir, &error)) {
46 *err =
47 Err(Location(), "Could not create the QtCreator project directory '" +
48 FilePathToUTF8(project_dir) +
49 "': " + base::File::ErrorToString(error));
50 return false;
51 }
52 }
53
54 base::FilePath project_prefix = project_dir.Append(kProjectName);
55 QtCreatorWriter gen(build_settings, builder, project_prefix, root_target);
56 gen.Run();
57 if (gen.err_.has_error()) {
58 *err = gen.err_;
59 return false;
60 }
61 return true;
62 }
63
QtCreatorWriter(const BuildSettings * build_settings,const Builder & builder,const base::FilePath & project_prefix,const std::string & root_target_name)64 QtCreatorWriter::QtCreatorWriter(const BuildSettings* build_settings,
65 const Builder& builder,
66 const base::FilePath& project_prefix,
67 const std::string& root_target_name)
68 : build_settings_(build_settings),
69 builder_(builder),
70 project_prefix_(project_prefix),
71 root_target_name_(root_target_name) {}
72
73 QtCreatorWriter::~QtCreatorWriter() = default;
74
CollectDeps(const Target * target)75 void QtCreatorWriter::CollectDeps(const Target* target) {
76 for (const auto& dep : target->GetDeps(Target::DEPS_ALL)) {
77 const Target* dep_target = dep.ptr;
78 if (!targets_.add(dep_target))
79 continue;
80 CollectDeps(dep_target);
81 }
82 }
83
DiscoverTargets()84 bool QtCreatorWriter::DiscoverTargets() {
85 auto all_targets = builder_.GetAllResolvedTargets();
86
87 if (root_target_name_.empty()) {
88 targets_.clear();
89 targets_.insert(all_targets.begin(), all_targets.end());
90 return true;
91 }
92
93 const Target* root_target = nullptr;
94 for (const Target* target : all_targets) {
95 if (target->label().name() == root_target_name_) {
96 root_target = target;
97 break;
98 }
99 }
100
101 if (!root_target) {
102 err_ = Err(Location(), "Target '" + root_target_name_ + "' not found.");
103 return false;
104 }
105
106 targets_.insert(root_target);
107 CollectDeps(root_target);
108 return true;
109 }
110
AddToSources(const Target::FileList & files)111 void QtCreatorWriter::AddToSources(const Target::FileList& files) {
112 for (const SourceFile& file : files) {
113 const std::string& file_path =
114 FilePathToUTF8(build_settings_->GetFullPath(file));
115 sources_.insert(file_path);
116 }
117 }
118
119 namespace QtCreatorWriterUtils {
120
121 enum class CVersion {
122 C99,
123 C11,
124 };
125
126 enum class CxxVersion {
127 CXX98,
128 CXX03,
129 CXX11,
130 CXX14,
131 CXX17,
132 };
133
ToMacro(CVersion version)134 std::string ToMacro(CVersion version) {
135 const std::string s = "__STDC_VERSION__";
136
137 switch (version) {
138 case CVersion::C99:
139 return s + " 199901L";
140 case CVersion::C11:
141 return s + " 201112L";
142 }
143
144 return std::string();
145 }
146
ToMacro(CxxVersion version)147 std::string ToMacro(CxxVersion version) {
148 const std::string name = "__cplusplus";
149
150 switch (version) {
151 case CxxVersion::CXX98:
152 case CxxVersion::CXX03:
153 return name + " 199711L";
154 case CxxVersion::CXX11:
155 return name + " 201103L";
156 case CxxVersion::CXX14:
157 return name + " 201402L";
158 case CxxVersion::CXX17:
159 return name + " 201703L";
160 }
161
162 return std::string();
163 }
164
165 const std::map<std::string, CVersion> kFlagToCVersion{
166 {"-std=gnu99", CVersion::C99},
167 {"-std=c99", CVersion::C99},
168 {"-std=gnu11", CVersion::C11},
169 {"-std=c11", CVersion::C11}};
170
171 const std::map<std::string, CxxVersion> kFlagToCxxVersion{
172 {"-std=gnu++11", CxxVersion::CXX11}, {"-std=c++11", CxxVersion::CXX11},
173 {"-std=gnu++98", CxxVersion::CXX98}, {"-std=c++98", CxxVersion::CXX98},
174 {"-std=gnu++03", CxxVersion::CXX03}, {"-std=c++03", CxxVersion::CXX03},
175 {"-std=gnu++14", CxxVersion::CXX14}, {"-std=c++14", CxxVersion::CXX14},
176 {"-std=c++1y", CxxVersion::CXX14}, {"-std=gnu++17", CxxVersion::CXX17},
177 {"-std=c++17", CxxVersion::CXX17}, {"-std=c++1z", CxxVersion::CXX17},
178 };
179
180 template <typename Enum>
181 struct CompVersion {
operator ()QtCreatorWriterUtils::CompVersion182 bool operator()(Enum a, Enum b) {
183 return static_cast<int>(a) < static_cast<int>(b);
184 }
185 };
186
187 struct CompilerOptions {
188 std::optional<CVersion> c_version_;
189 std::optional<CxxVersion> cxx_version_;
190
SetCVersionQtCreatorWriterUtils::CompilerOptions191 void SetCVersion(CVersion ver) { SetVersionImpl(c_version_, ver); }
192
SetCxxVersionQtCreatorWriterUtils::CompilerOptions193 void SetCxxVersion(CxxVersion ver) { SetVersionImpl(cxx_version_, ver); }
194
195 private:
196 template <typename Version>
SetVersionImplQtCreatorWriterUtils::CompilerOptions197 void SetVersionImpl(std::optional<Version>& cur_ver, Version ver) {
198 if (cur_ver)
199 cur_ver = std::max(*cur_ver, ver, CompVersion<Version>{});
200 else
201 cur_ver = ver;
202 }
203 };
204
ParseCompilerOption(const std::string & flag,CompilerOptions * options)205 void ParseCompilerOption(const std::string& flag, CompilerOptions* options) {
206 auto c_ver = kFlagToCVersion.find(flag);
207 if (c_ver != kFlagToCVersion.end())
208 options->SetCVersion(c_ver->second);
209
210 auto cxx_ver = kFlagToCxxVersion.find(flag);
211 if (cxx_ver != kFlagToCxxVersion.end())
212 options->SetCxxVersion(cxx_ver->second);
213 }
214
ParseCompilerOptions(const std::vector<std::string> & cflags,CompilerOptions * options)215 void ParseCompilerOptions(const std::vector<std::string>& cflags,
216 CompilerOptions* options) {
217 for (const std::string& flag : cflags)
218 ParseCompilerOption(flag, options);
219 }
220
221 } // namespace QtCreatorWriterUtils
222
HandleTarget(const Target * target)223 void QtCreatorWriter::HandleTarget(const Target* target) {
224 using namespace QtCreatorWriterUtils;
225
226 SourceFile build_file = builder_.loader()->BuildFileForLabel(target->label());
227 sources_.insert(FilePathToUTF8(build_settings_->GetFullPath(build_file)));
228 AddToSources(target->settings()->import_manager().GetImportedFiles());
229
230 AddToSources(target->sources());
231 AddToSources(target->public_headers());
232
233 for (ConfigValuesIterator it(target); !it.done(); it.Next()) {
234 for (const auto& input : it.cur().inputs())
235 sources_.insert(FilePathToUTF8(build_settings_->GetFullPath(input)));
236
237 SourceFile precompiled_source = it.cur().precompiled_source();
238 if (!precompiled_source.is_null()) {
239 sources_.insert(
240 FilePathToUTF8(build_settings_->GetFullPath(precompiled_source)));
241 }
242
243 for (const SourceDir& include_dir : it.cur().include_dirs()) {
244 includes_.insert(
245 FilePathToUTF8(build_settings_->GetFullPath(include_dir)));
246 }
247
248 static constexpr const char* define_str = "#define ";
249 for (std::string define : it.cur().defines()) {
250 size_t equal_pos = define.find('=');
251 if (equal_pos != std::string::npos)
252 define[equal_pos] = ' ';
253 define.insert(0, define_str);
254 defines_.insert(define);
255 }
256
257 CompilerOptions options;
258 ParseCompilerOptions(it.cur().cflags(), &options);
259 ParseCompilerOptions(it.cur().cflags_c(), &options);
260 ParseCompilerOptions(it.cur().cflags_cc(), &options);
261
262 auto add_define_version = [this](auto& ver) {
263 if (ver)
264 defines_.insert(define_str + ToMacro(*ver));
265 };
266 add_define_version(options.c_version_);
267 add_define_version(options.cxx_version_);
268 }
269 }
270
GenerateFile(const base::FilePath::CharType * suffix,const std::set<std::string> & items)271 void QtCreatorWriter::GenerateFile(const base::FilePath::CharType* suffix,
272 const std::set<std::string>& items) {
273 const base::FilePath file_path = project_prefix_.AddExtension(suffix);
274 StringOutputBuffer storage;
275 std::ostream output(&storage);
276 for (const std::string& item : items)
277 output << item << std::endl;
278 storage.WriteToFileIfChanged(file_path, &err_);
279 }
280
Run()281 void QtCreatorWriter::Run() {
282 if (!DiscoverTargets())
283 return;
284
285 for (const Target* target : targets_) {
286 if (target->toolchain()->label() !=
287 builder_.loader()->GetDefaultToolchain())
288 continue;
289 HandleTarget(target);
290 }
291
292 std::set<std::string> empty_list;
293
294 GenerateFile(kMainProjectFileSuffix, empty_list);
295 GenerateFile(kSourcesFileSuffix, sources_);
296 GenerateFile(kIncludesFileSuffix, includes_);
297 GenerateFile(kDefinesFileSuffix, defines_);
298 }
299