• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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