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
|
// Copyright (c) 2013 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "tools/gn/ninja_toolchain_writer.h"
#include <fstream>
#include "base/file_util.h"
#include "base/strings/stringize_macros.h"
#include "tools/gn/build_settings.h"
#include "tools/gn/item_node.h"
#include "tools/gn/settings.h"
#include "tools/gn/target.h"
#include "tools/gn/toolchain.h"
#include "tools/gn/toolchain_manager.h"
#include "tools/gn/trace.h"
NinjaToolchainWriter::NinjaToolchainWriter(
const Settings* settings,
const std::vector<const Target*>& targets,
const std::set<std::string>& skip_files,
std::ostream& out)
: settings_(settings),
targets_(targets),
skip_files_(skip_files),
out_(out),
path_output_(settings_->build_settings()->build_dir(),
ESCAPE_NINJA, true),
helper_(settings->build_settings()) {
}
NinjaToolchainWriter::~NinjaToolchainWriter() {
}
void NinjaToolchainWriter::Run() {
WriteRules();
WriteSubninjas();
}
// static
bool NinjaToolchainWriter::RunAndWriteFile(
const Settings* settings,
const std::vector<const Target*>& targets,
const std::set<std::string>& skip_files) {
NinjaHelper helper(settings->build_settings());
base::FilePath ninja_file(settings->build_settings()->GetFullPath(
helper.GetNinjaFileForToolchain(settings).GetSourceFile(
settings->build_settings())));
ScopedTrace trace(TraceItem::TRACE_FILE_WRITE, FilePathToUTF8(ninja_file));
file_util::CreateDirectory(ninja_file.DirName());
std::ofstream file;
file.open(FilePathToUTF8(ninja_file).c_str(),
std::ios_base::out | std::ios_base::binary);
if (file.fail())
return false;
NinjaToolchainWriter gen(settings, targets, skip_files, file);
gen.Run();
return true;
}
void NinjaToolchainWriter::WriteRules() {
const Toolchain* tc = settings_->build_settings()->toolchain_manager()
.GetToolchainDefinitionUnlocked(settings_->toolchain_label());
CHECK(tc);
std::string indent(" ");
NinjaHelper helper(settings_->build_settings());
std::string rule_prefix = helper.GetRulePrefix(settings_);
for (int i = Toolchain::TYPE_NONE + 1; i < Toolchain::TYPE_NUMTYPES; i++) {
Toolchain::ToolType tool_type = static_cast<Toolchain::ToolType>(i);
const Toolchain::Tool& tool = tc->GetTool(tool_type);
if (tool.command.empty())
continue;
out_ << "rule " << rule_prefix << Toolchain::ToolTypeToName(tool_type)
<< std::endl;
#define WRITE_ARG(name) \
if (!tool.name.empty()) \
out_ << indent << " " STRINGIZE(name) " = " << tool.name << std::endl;
WRITE_ARG(command);
WRITE_ARG(depfile);
WRITE_ARG(deps);
WRITE_ARG(description);
WRITE_ARG(pool);
WRITE_ARG(restat);
WRITE_ARG(rspfile);
WRITE_ARG(rspfile_content);
#undef WRITE_ARG
}
out_ << std::endl;
}
void NinjaToolchainWriter::WriteSubninjas() {
// Write subninja commands for each generated target.
for (size_t i = 0; i < targets_.size(); i++) {
if (!targets_[i]->item_node()->should_generate() ||
(targets_[i]->settings()->build_settings()->using_external_generator()
&& targets_[i]->external()))
continue;
OutputFile ninja_file = helper_.GetNinjaFileForTarget(targets_[i]);
if (skip_files_.find(ninja_file.value()) != skip_files_.end())
continue;
out_ << "subninja ";
path_output_.WriteFile(out_, ninja_file);
out_ << std::endl;
}
out_ << std::endl;
}
|