blob: 21ae6d007a0d40a88511f79eab1199c4b79e26f3 [file] [log] [blame]
// Copyright 2019 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 "gn/ninja_rust_binary_target_writer.h"
#include <sstream>
#include "base/strings/string_util.h"
#include "gn/deps_iterator.h"
#include "gn/filesystem_utils.h"
#include "gn/general_tool.h"
#include "gn/ninja_target_command_util.h"
#include "gn/ninja_utils.h"
#include "gn/rust_substitution_type.h"
#include "gn/rust_values.h"
#include "gn/substitution_writer.h"
#include "gn/target.h"
namespace {
// Returns the proper escape options for writing compiler and linker flags.
EscapeOptions GetFlagOptions() {
EscapeOptions opts;
opts.mode = ESCAPE_NINJA_COMMAND;
return opts;
}
void WriteVar(const char* name,
const std::string& value,
EscapeOptions opts,
std::ostream& out) {
out << name << " = ";
EscapeStringToStream(out, value, opts);
out << std::endl;
}
void WriteCrateVars(const Target* target,
const Tool* tool,
EscapeOptions opts,
std::ostream& out) {
WriteVar(kRustSubstitutionCrateName.ninja_name,
target->rust_values().crate_name(), opts, out);
std::string crate_type;
switch (target->rust_values().crate_type()) {
// Auto-select the crate type for executables, static libraries, and rlibs.
case RustValues::CRATE_AUTO: {
switch (target->output_type()) {
case Target::EXECUTABLE:
crate_type = "bin";
break;
case Target::STATIC_LIBRARY:
crate_type = "staticlib";
break;
case Target::RUST_LIBRARY:
crate_type = "rlib";
break;
case Target::RUST_PROC_MACRO:
crate_type = "proc-macro";
break;
default:
NOTREACHED();
}
break;
}
case RustValues::CRATE_BIN:
crate_type = "bin";
break;
case RustValues::CRATE_CDYLIB:
crate_type = "cdylib";
break;
case RustValues::CRATE_DYLIB:
crate_type = "dylib";
break;
case RustValues::CRATE_PROC_MACRO:
crate_type = "proc-macro";
break;
case RustValues::CRATE_RLIB:
crate_type = "rlib";
break;
case RustValues::CRATE_STATICLIB:
crate_type = "staticlib";
break;
default:
NOTREACHED();
}
WriteVar(kRustSubstitutionCrateType.ninja_name, crate_type, opts, out);
WriteVar(SubstitutionOutputExtension.ninja_name,
SubstitutionWriter::GetLinkerSubstitution(
target, tool, &SubstitutionOutputExtension),
opts, out);
WriteVar(SubstitutionOutputDir.ninja_name,
SubstitutionWriter::GetLinkerSubstitution(target, tool,
&SubstitutionOutputDir),
opts, out);
}
} // namespace
NinjaRustBinaryTargetWriter::NinjaRustBinaryTargetWriter(const Target* target,
std::ostream& out)
: NinjaBinaryTargetWriter(target, out),
tool_(target->toolchain()->GetToolForTargetFinalOutputAsRust(target)) {}
NinjaRustBinaryTargetWriter::~NinjaRustBinaryTargetWriter() = default;
// TODO(juliehockett): add inherited library support? and IsLinkable support?
// for c-cross-compat
void NinjaRustBinaryTargetWriter::Run() {
DCHECK(target_->output_type() != Target::SOURCE_SET);
size_t num_stamp_uses = target_->sources().size();
std::vector<OutputFile> input_deps =
WriteInputsStampAndGetDep(num_stamp_uses);
WriteCompilerVars();
// Classify our dependencies.
ClassifiedDeps classified_deps = GetClassifiedDeps();
// The input dependencies will be an order-only dependency. This will cause
// Ninja to make sure the inputs are up to date before compiling this source,
// but changes in the inputs deps won't cause the file to be recompiled. See
// the comment on NinjaCBinaryTargetWriter::Run for more detailed explanation.
std::vector<OutputFile> order_only_deps = WriteInputDepsStampAndGetDep(
std::vector<const Target*>(), num_stamp_uses);
std::copy(input_deps.begin(), input_deps.end(),
std::back_inserter(order_only_deps));
// Build lists which will go into different bits of the rustc command line.
// Public rust_library deps go in a --extern rlibs, public non-rust deps go in
// -Ldependency. Also assemble a list of extra (i.e. implicit) deps
// for ninja dependency tracking.
UniqueVector<OutputFile> implicit_deps;
AppendSourcesAndInputsToImplicitDeps(&implicit_deps);
implicit_deps.Append(classified_deps.extra_object_files.begin(),
classified_deps.extra_object_files.end());
std::vector<OutputFile> rustdeps;
std::vector<OutputFile> nonrustdeps;
nonrustdeps.insert(nonrustdeps.end(),
classified_deps.extra_object_files.begin(),
classified_deps.extra_object_files.end());
for (const auto* framework_dep : classified_deps.framework_deps) {
order_only_deps.push_back(framework_dep->dependency_output_file());
}
for (const auto* non_linkable_dep : classified_deps.non_linkable_deps) {
if (non_linkable_dep->source_types_used().RustSourceUsed() &&
non_linkable_dep->output_type() != Target::SOURCE_SET) {
rustdeps.push_back(non_linkable_dep->dependency_output_file());
}
order_only_deps.push_back(non_linkable_dep->dependency_output_file());
}
for (const auto* linkable_dep : classified_deps.linkable_deps) {
// Rust cdylibs are treated as non-Rust dependencies for linking purposes.
if (linkable_dep->source_types_used().RustSourceUsed() &&
linkable_dep->rust_values().crate_type() != RustValues::CRATE_CDYLIB) {
rustdeps.push_back(linkable_dep->link_output_file());
} else {
nonrustdeps.push_back(linkable_dep->link_output_file());
}
implicit_deps.push_back(linkable_dep->dependency_output_file());
}
// Rust libraries specified by paths.
for (ConfigValuesIterator iter(target_); !iter.done(); iter.Next()) {
const ConfigValues& cur = iter.cur();
for (const auto& e : cur.externs()) {
if (e.second.is_source_file()) {
implicit_deps.push_back(
OutputFile(settings_->build_settings(), e.second.source_file()));
}
}
}
// Collect the full transitive set of rust libraries that this target depends
// on, and the public flag represents if the target has direct access to the
// dependency through a chain of public_deps.
std::vector<ExternCrate> transitive_crates;
for (const auto& inherited : resolved().GetRustInheritedLibraries(target_)) {
const Target* dep = inherited.target();
bool has_direct_access = inherited.is_public();
// We will tell rustc to look for crate metadata for any rust crate
// dependencies except cdylibs, as they have no metadata present.
if (dep->source_types_used().RustSourceUsed() &&
RustValues::IsRustLibrary(dep)) {
transitive_crates.push_back({dep, has_direct_access});
// If the current crate can directly acccess the `dep` crate, then the
// current crate needs an implicit dependency on `dep` so it will be
// rebuilt if `dep` changes.
if (has_direct_access) {
implicit_deps.push_back(dep->dependency_output_file());
}
}
}
std::vector<OutputFile> tool_outputs;
SubstitutionWriter::ApplyListToLinkerAsOutputFile(
target_, tool_, tool_->outputs(), &tool_outputs);
WriteCompilerBuildLine({target_->rust_values().crate_root()},
implicit_deps.vector(), order_only_deps, tool_->name(),
tool_outputs);
std::vector<const Target*> extern_deps(
classified_deps.linkable_deps.vector());
std::copy(classified_deps.non_linkable_deps.begin(),
classified_deps.non_linkable_deps.end(),
std::back_inserter(extern_deps));
WriteExternsAndDeps(target_->IsFinal(), extern_deps, transitive_crates,
rustdeps, nonrustdeps);
WriteSourcesAndInputs();
WritePool(out_);
}
void NinjaRustBinaryTargetWriter::WriteCompilerVars() {
const SubstitutionBits& subst = target_->toolchain()->substitution_bits();
EscapeOptions opts = GetFlagOptions();
WriteCrateVars(target_, tool_, opts, out_);
WriteRustCompilerVars(subst, /*indent=*/false, /*always_write=*/true);
WriteSharedVars(subst);
}
void NinjaRustBinaryTargetWriter::AppendSourcesAndInputsToImplicitDeps(
UniqueVector<OutputFile>* deps) const {
// Only the crate_root file needs to be given to rustc as input.
// Any other 'sources' are just implicit deps.
// Most Rust targets won't bother specifying the "sources =" line
// because it is handled sufficiently by crate_root and the generation
// of depfiles by rustc. But for those which do...
for (const auto& source : target_->sources()) {
deps->push_back(OutputFile(settings_->build_settings(), source));
}
for (const auto& data : target_->config_values().inputs()) {
deps->push_back(OutputFile(settings_->build_settings(), data));
}
}
void NinjaRustBinaryTargetWriter::WriteSourcesAndInputs() {
out_ << " sources =";
for (const auto& source : target_->sources()) {
out_ << " ";
path_output_.WriteFile(out_,
OutputFile(settings_->build_settings(), source));
}
for (const auto& data : target_->config_values().inputs()) {
out_ << " ";
path_output_.WriteFile(out_, OutputFile(settings_->build_settings(), data));
}
out_ << std::endl;
}
void NinjaRustBinaryTargetWriter::WriteExternsAndDeps(
bool target_is_final,
const std::vector<const Target*>& deps,
const std::vector<ExternCrate>& transitive_rust_deps,
const std::vector<OutputFile>& rustdeps,
const std::vector<OutputFile>& nonrustdeps) {
// Writes a external LibFile which comes from user-specified externs, and may
// be either a string or a SourceFile.
auto write_extern_lib_file = [this](std::string_view crate_name,
LibFile lib_file) {
out_ << " --extern ";
out_ << crate_name;
out_ << "=";
if (lib_file.is_source_file()) {
path_output_.WriteFile(out_, lib_file.source_file());
} else {
EscapeOptions escape_opts_command;
escape_opts_command.mode = ESCAPE_NINJA_COMMAND;
EscapeStringToStream(out_, lib_file.value(), escape_opts_command);
}
};
// Writes an external OutputFile which comes from a dependency of the current
// target.
auto write_extern_target = [this](const Target& dep) {
std::string_view crate_name;
const auto& aliased_deps = target_->rust_values().aliased_deps();
if (auto it = aliased_deps.find(dep.label()); it != aliased_deps.end()) {
crate_name = it->second;
} else {
crate_name = dep.rust_values().crate_name();
}
out_ << " --extern ";
out_ << crate_name;
out_ << "=";
path_output_.WriteFile(out_, dep.dependency_output_file());
};
// Write accessible crates with `--extern` to add them to the extern prelude.
out_ << " externs =";
// Tracking to avoid emitted the same lib twice. We track it instead of
// pre-emptively constructing a UniqueVector since we would have to also store
// the crate name, and in the future the public-ness.
std::unordered_set<OutputFile> emitted_rust_libs;
// TODO: We defer private dependencies to -Ldependency until --extern priv is
// stabilized.
UniqueVector<SourceDir> private_extern_dirs;
// Walk the transitive closure of all rust dependencies.
//
// For dependencies that are meant to be accessible we pass them to --extern
// in order to add them to the crate's extern prelude.
//
// For all transitive dependencies, we add them to `private_extern_dirs` in
// order to generate a -Ldependency switch that points to them. This ensures
// that rustc can find them if they are used by other dependencies. For
// example:
//
// A -> C --public--> D
// -> B --private-> D
//
// Here A has direct access to D, but B and C also make use of D, and they
// will only search the paths specified to -Ldependency, thus D needs to
// appear as both a --extern (for A) and -Ldependency (for B and C).
for (const auto& crate : transitive_rust_deps) {
const OutputFile& rust_lib = crate.target->dependency_output_file();
if (emitted_rust_libs.count(rust_lib) == 0) {
if (crate.has_direct_access) {
write_extern_target(*crate.target);
}
emitted_rust_libs.insert(rust_lib);
}
private_extern_dirs.push_back(
rust_lib.AsSourceFile(settings_->build_settings()).GetDir());
}
// Add explicitly specified externs from the GN target.
for (ConfigValuesIterator iter(target_); !iter.done(); iter.Next()) {
const ConfigValues& cur = iter.cur();
for (const auto& [crate_name, lib_file] : cur.externs()) {
write_extern_lib_file(crate_name, lib_file);
}
}
out_ << std::endl;
out_ << " rustdeps =";
for (const SourceDir& dir : private_extern_dirs) {
// TODO: switch to using `--extern priv:name` after stabilization.
out_ << " -Ldependency=";
path_output_.WriteDir(out_, dir, PathOutput::DIR_NO_LAST_SLASH);
}
UniqueVector<SourceDir> nonrustdep_dirs;
// Non-Rust native dependencies. A dependency from Rust implies the ability
// to specify it in #[link], and GN will ensure that rustc can find it by
// adding it to the native library search paths.
for (const auto& nonrustdep : nonrustdeps) {
nonrustdep_dirs.push_back(
nonrustdep.AsSourceFile(settings_->build_settings()).GetDir());
}
for (const auto& nonrustdep_dir : nonrustdep_dirs) {
out_ << " -Lnative=";
path_output_.WriteDir(out_, nonrustdep_dir, PathOutput::DIR_NO_LAST_SLASH);
}
// If rustc will invoke a linker, then pass linker arguments to include those
// non-Rust native dependencies in the linking step.
if (target_is_final) {
// Before outputting any libraries to link, ensure the linker is in a mode
// that allows dynamic linking, as rustc may have previously put it into
// static-only mode.
if (nonrustdeps.size() > 0) {
out_ << " " << tool_->dynamic_link_switch();
}
for (const auto& nonrustdep : nonrustdeps) {
out_ << " -Clink-arg=";
path_output_.WriteFile(out_, nonrustdep);
}
}
// Library search paths are required to find system libraries named in #[link]
// directives, which will not be specified in non-Rust native dependencies.
WriteLibrarySearchPath(out_, tool_);
// If rustc will invoke a linker, all libraries need the passed through to the
// linker.
if (target_is_final) {
WriteLibs(out_, tool_);
}
out_ << std::endl;
out_ << " ldflags =";
// If rustc will invoke a linker, linker flags need to be forwarded through to
// the linker.
if (target_is_final) {
WriteCustomLinkerFlags(out_, tool_);
}
out_ << std::endl;
}