blob: 60701b937567d9faa0c67943966b5fe217c6cea3 [file] [log] [blame]
// Copyright 2014 The Bazel Authors. All rights reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package com.google.devtools.build.lib.rules.cpp;
import static com.google.devtools.build.lib.packages.Attribute.attr;
import static com.google.devtools.build.lib.packages.BuildType.LABEL;
import static com.google.devtools.build.lib.packages.BuildType.LICENSE;
import static com.google.devtools.build.lib.packages.BuildType.NODEP_LABEL;
import static com.google.devtools.build.lib.packages.Type.BOOLEAN;
import static com.google.devtools.build.lib.packages.Type.STRING;
import com.google.common.collect.ImmutableList;
import com.google.devtools.build.lib.analysis.BaseRuleClasses;
import com.google.devtools.build.lib.analysis.PlatformConfiguration;
import com.google.devtools.build.lib.analysis.RuleDefinition;
import com.google.devtools.build.lib.analysis.RuleDefinitionEnvironment;
import com.google.devtools.build.lib.analysis.TemplateVariableInfo;
import com.google.devtools.build.lib.analysis.config.HostTransition;
import com.google.devtools.build.lib.cmdline.Label;
import com.google.devtools.build.lib.packages.Attribute.LabelLateBoundDefault;
import com.google.devtools.build.lib.packages.AttributeMap;
import com.google.devtools.build.lib.packages.RuleClass;
import com.google.devtools.build.lib.packages.Type;
/**
* Rule definition for compiler definition.
*/
public final class CcToolchainRule implements RuleDefinition {
public static final String LIBC_TOP_ATTR = ":libc_top";
public static final String TARGET_LIBC_TOP_ATTR = ":target_libc_top";
public static final String FDO_OPTIMIZE_ATTR = ":fdo_optimize";
public static final String FDO_PROFILE_ATTR = ":fdo_profile";
public static final String CSFDO_PROFILE_ATTR = ":csfdo_profile";
public static final String XFDO_PROFILE_ATTR = ":xfdo_profile";
public static final String TOOLCHAIN_CONFIG_ATTR = "toolchain_config";
private static Label getLabel(AttributeMap attributes, String attrName, Label defaultValue) {
if (attributes.has(attrName, LABEL)) {
Label value = attributes.get(attrName, LABEL);
if (value != null) {
return value;
}
}
return defaultValue;
}
private static final LabelLateBoundDefault<?> LIBC_TOP_VALUE =
LabelLateBoundDefault.fromTargetConfiguration(
CppConfiguration.class,
null,
(rule, attributes, cppConfig) -> {
// Is the libcTop directly set via a flag?
Label cppOptionLibcTop = cppConfig.getLibcTopLabel();
if (cppOptionLibcTop != null) {
return cppOptionLibcTop;
}
// Look up the value from the attribute.
// This avoids analyzing the label from the CROSSTOOL if the attribute is set.
return getLabel(attributes, "libc_top", /* defaultValue= */ null);
});
private static final LabelLateBoundDefault<?> TARGET_LIBC_TOP_VALUE =
LabelLateBoundDefault.fromTargetConfiguration(
CppConfiguration.class,
null,
(rule, attributes, cppConfig) -> {
// Is the libcTop directly set via a flag?
Label cppOptionLibcTop = cppConfig.getTargetLibcTopLabel();
if (cppOptionLibcTop != null) {
return cppOptionLibcTop;
}
// Look up the value from the attribute.
// This avoids analyzing the label from the CROSSTOOL if the attribute is set.
return getLabel(attributes, "libc_top", /* defaultValue= */ null);
});
private static final LabelLateBoundDefault<?> FDO_OPTIMIZE_VALUE =
LabelLateBoundDefault.fromTargetConfiguration(
CppConfiguration.class,
null,
(rule, attributes, cppConfig) ->
cppConfig.getFdoOptimizeLabelUnsafeSinceItCanReturnValueFromWrongConfiguration());
private static final LabelLateBoundDefault<?> FDO_PROFILE_VALUE =
LabelLateBoundDefault.fromTargetConfiguration(
CppConfiguration.class,
null,
(rule, attributes, cppConfig) ->
cppConfig.getFdoProfileLabelUnsafeSinceItCanReturnValueFromWrongConfiguration());
private static final LabelLateBoundDefault<?> CSFDO_PROFILE_VALUE =
LabelLateBoundDefault.fromTargetConfiguration(
CppConfiguration.class,
null,
(rule, attributes, cppConfig) -> cppConfig.getCSFdoProfileLabel());
private static final LabelLateBoundDefault<?> XFDO_PROFILE_VALUE =
LabelLateBoundDefault.fromTargetConfiguration(
CppConfiguration.class,
null,
(rule, attributes, cppConfig) ->
cppConfig.getXFdoProfileLabelUnsafeSinceItCanReturnValueFromWrongConfiguration());
private static final LabelLateBoundDefault<?> FDO_PREFETCH_HINTS =
LabelLateBoundDefault.fromTargetConfiguration(
CppConfiguration.class,
null,
(rule, attributes, cppConfig) -> cppConfig.getFdoPrefetchHintsLabel());
/**
* Returns true if zipper should be loaded. We load the zipper executable if FDO optimization is
* enabled through --fdo_optimize or --fdo_profile
*/
private static boolean shouldIncludeZipperInToolchain(CppConfiguration cppConfiguration) {
if (cppConfiguration.isThisHostConfigurationDoNotUseWillBeRemovedFor129045294()) {
// This is actually a bug, because with platforms, and before toolchain-transitions are
// implemented, all toolchains are analyzed in the host configuration. We're betting on
// nobody in the Bazel world actually using zipped fdo profiles and platforms at the same
// time, and if people do, they'll have to wait for toolchain-transitions. This needs to be
// fixed.
// TODO(b/129045294): Fix this once toolchain-transitions are implemented.
return false;
}
return cppConfiguration.getFdoOptimizeLabelUnsafeSinceItCanReturnValueFromWrongConfiguration()
!= null
|| cppConfiguration.getFdoProfileLabelUnsafeSinceItCanReturnValueFromWrongConfiguration()
!= null
|| cppConfiguration.getFdoPathUnsafeSinceItCanReturnValueFromWrongConfiguration() != null;
}
@Override
public RuleClass build(RuleClass.Builder builder, RuleDefinitionEnvironment env) {
final Label zipper = env.getToolsLabel("//tools/zip:zipper");
return builder
.requiresConfigurationFragments(CppConfiguration.class, PlatformConfiguration.class)
.advertiseProvider(TemplateVariableInfo.class)
.add(attr("output_licenses", LICENSE))
/* <!-- #BLAZE_RULE(cc_toolchain).ATTRIBUTE(cpu) -->
Deprecated. Use toolchain_identifier attribute instead. It will be a noop after
<a href="https://github.com/bazelbuild/bazel/issues/5380">CROSSTOOL migration to Starlark
</a>, and will be removed by
<a href="https://github.com/bazelbuild/bazel/issues/7075">#7075</a>.
<p>When set, it will be used to perform crosstool_config.toolchain selection. It will take
precedence over --cpu Bazel option.</p>
<!-- #END_BLAZE_RULE.ATTRIBUTE -->*/
.add(attr("cpu", STRING).nonconfigurable("Used in configuration creation"))
/* <!-- #BLAZE_RULE(cc_toolchain).ATTRIBUTE(compiler) -->
Deprecated. Use <code>toolchain_identifier</code> attribute instead. It will be a noop
after
<a href="https://github.com/bazelbuild/bazel/issues/5380">
CROSSTOOL migration to Starlark
</a>, and will be removed by
<a href="https://github.com/bazelbuild/bazel/issues/7075">#7075</a>.
<p>When set, it will be used to perform crosstool_config.toolchain selection. It will take
precedence over --cpu Bazel option.</p>
<!-- #END_BLAZE_RULE.ATTRIBUTE -->*/
.add(attr("compiler", STRING).nonconfigurable("Used in configuration creation"))
/* <!-- #BLAZE_RULE(cc_toolchain).ATTRIBUTE(all_files) -->
Collection of all cc_toolchain artifacts. These artifacts will be added as inputs to all
rules_cc related actions (with the exception of actions that are using more precise sets of
artifacts from attributes below). Bazel assumes that <code>all_files</code> is a superset
of all other artifact-providing attributes (e.g. linkstamp compilation needs both compile
and link files, so it takes <code>all_files</code>).
<p>
This is what <code>cc_toolchain.files</code> contains, and this is used by all Starlark
rules using C++ toolchain.</p>
<!-- #END_BLAZE_RULE.ATTRIBUTE -->*/
.add(
attr("all_files", LABEL)
.legacyAllowAnyFileType()
.cfg(HostTransition.createFactory())
.mandatory())
/* <!-- #BLAZE_RULE(cc_toolchain).ATTRIBUTE(compiler_files) -->
Collection of all cc_toolchain artifacts required for compile actions.
<p>Currently only used by lto_backend actions, regular compile actions use all_files
(<a href="https://github.com/bazelbuild/bazel/issues/6927">#6927</a>).
</p>
<!-- #END_BLAZE_RULE.ATTRIBUTE -->*/
.add(
attr("compiler_files", LABEL)
.legacyAllowAnyFileType()
.cfg(HostTransition.createFactory())
.mandatory())
/* <!-- #BLAZE_RULE(cc_toolchain).ATTRIBUTE(compiler_files_without_includes) -->
Collection of all cc_toolchain artifacts required for compile actions in case when
input discovery is supported (currently Google-only).
<!-- #END_BLAZE_RULE.ATTRIBUTE -->*/
.add(
attr("compiler_files_without_includes", LABEL)
.legacyAllowAnyFileType()
.cfg(HostTransition.createFactory()))
/* <!-- #BLAZE_RULE(cc_toolchain).ATTRIBUTE(strip_files) -->
Collection of all cc_toolchain artifacts required for strip actions.
<!-- #END_BLAZE_RULE.ATTRIBUTE -->*/
.add(
attr("strip_files", LABEL)
.legacyAllowAnyFileType()
.cfg(HostTransition.createFactory())
.mandatory())
/* <!-- #BLAZE_RULE(cc_toolchain).ATTRIBUTE(objcopy_files) -->
Collection of all cc_toolchain artifacts required for objcopy actions.
<!-- #END_BLAZE_RULE.ATTRIBUTE -->*/
.add(
attr("objcopy_files", LABEL)
.legacyAllowAnyFileType()
.cfg(HostTransition.createFactory())
.mandatory())
/* <!-- #BLAZE_RULE(cc_toolchain).ATTRIBUTE(as_files) -->
<p>Collection of all cc_toolchain artifacts required for assembly actions.</p>
<!-- #END_BLAZE_RULE.ATTRIBUTE -->*/
.add(attr("as_files", LABEL).legacyAllowAnyFileType().cfg(HostTransition.createFactory()))
/* <!-- #BLAZE_RULE(cc_toolchain).ATTRIBUTE(ar_files) -->
<p>Collection of all cc_toolchain artifacts required for archiving actions.</p>
<!-- #END_BLAZE_RULE.ATTRIBUTE -->*/
.add(attr("ar_files", LABEL).legacyAllowAnyFileType().cfg(HostTransition.createFactory()))
/* <!-- #BLAZE_RULE(cc_toolchain).ATTRIBUTE(linker_files) -->
Collection of all cc_toolchain artifacts required for linking actions.
<!-- #END_BLAZE_RULE.ATTRIBUTE -->*/
.add(
attr("linker_files", LABEL)
.legacyAllowAnyFileType()
.cfg(HostTransition.createFactory())
.mandatory())
/* <!-- #BLAZE_RULE(cc_toolchain).ATTRIBUTE(dwp_files) -->
Collection of all cc_toolchain artifacts required for dwp actions.
<!-- #END_BLAZE_RULE.ATTRIBUTE -->*/
.add(
attr("dwp_files", LABEL)
.legacyAllowAnyFileType()
.cfg(HostTransition.createFactory())
.mandatory())
/* <!-- #BLAZE_RULE(cc_toolchain).ATTRIBUTE(coverage_files) -->
Collection of all cc_toolchain artifacts required for coverage actions. If not specified,
all_files are used.
<!-- #END_BLAZE_RULE.ATTRIBUTE -->*/
.add(
attr("coverage_files", LABEL)
.legacyAllowAnyFileType()
.cfg(HostTransition.createFactory()))
/* <!-- #BLAZE_RULE(cc_toolchain).ATTRIBUTE(static_runtime_lib) -->
Static library artifact for the C++ runtime library (e.g. libstdc++.a).
<p>This will be used when 'static_link_cpp_runtimes' feature is enabled, and we're linking
dependencies statically.</p>
<!-- #END_BLAZE_RULE.ATTRIBUTE -->*/
.add(attr("static_runtime_lib", LABEL).legacyAllowAnyFileType())
/* <!-- #BLAZE_RULE(cc_toolchain).ATTRIBUTE(dynamic_runtime_lib) -->
Dynamic library artifact for the C++ runtime library (e.g. libstdc++.so).
<p>This will be used when 'static_link_cpp_runtimes' feature is enabled, and we're linking
dependencies dynamically.</p>
<!-- #END_BLAZE_RULE.ATTRIBUTE -->*/
.add(attr("dynamic_runtime_lib", LABEL).legacyAllowAnyFileType())
/* <!-- #BLAZE_RULE(cc_toolchain).ATTRIBUTE(module_map) -->
Module map artifact to be used for modular builds.
<!-- #END_BLAZE_RULE.ATTRIBUTE -->*/
.add(attr("module_map", LABEL).legacyAllowAnyFileType().cfg(HostTransition.createFactory()))
/* <!-- #BLAZE_RULE(cc_toolchain).ATTRIBUTE(supports_param_files) -->
Set to True when cc_toolchain supports using param files for linking actions.
<!-- #END_BLAZE_RULE.ATTRIBUTE -->*/
.add(attr("supports_param_files", BOOLEAN).value(true))
/* <!-- #BLAZE_RULE(cc_toolchain).ATTRIBUTE(supports_header_parsing) -->
Set to True when cc_toolchain supports header parsing actions.
<!-- #END_BLAZE_RULE.ATTRIBUTE -->*/
.add(attr("supports_header_parsing", BOOLEAN).value(false))
.add(
attr("$interface_library_builder", LABEL)
.cfg(HostTransition.createFactory())
.singleArtifact()
.value(env.getToolsLabel("//tools/cpp:interface_library_builder")))
.add(
attr("$link_dynamic_library_tool", LABEL)
.cfg(HostTransition.createFactory())
.singleArtifact()
.value(env.getToolsLabel("//tools/cpp:link_dynamic_library")))
.add(
attr(CcToolchain.CC_TOOLCHAIN_TYPE_ATTRIBUTE_NAME, NODEP_LABEL)
.value(CppRuleClasses.ccToolchainTypeAttribute(env)))
.add(
attr(":zipper", LABEL)
.cfg(HostTransition.createFactory())
.singleArtifact()
.value(
LabelLateBoundDefault.fromTargetConfiguration(
CppConfiguration.class,
null,
(rule, attributes, cppConfig) ->
shouldIncludeZipperInToolchain(cppConfig) ? zipper : null)))
// TODO(b/78578234): Make this the default and remove the late-bound versions.
/* <!-- #BLAZE_RULE(cc_toolchain).ATTRIBUTE(libc_top) -->
A collection of artifacts for libc passed as inputs to compile/linking actions.
<!-- #END_BLAZE_RULE.ATTRIBUTE -->*/
.add(attr("libc_top", LABEL).allowedFileTypes())
.add(attr(LIBC_TOP_ATTR, LABEL).value(LIBC_TOP_VALUE))
.add(attr(TARGET_LIBC_TOP_ATTR, LABEL).value(TARGET_LIBC_TOP_VALUE))
.add(attr(FDO_OPTIMIZE_ATTR, LABEL).value(FDO_OPTIMIZE_VALUE))
.add(
attr(XFDO_PROFILE_ATTR, LABEL)
.allowedRuleClasses("fdo_profile")
.mandatoryProviders(ImmutableList.of(FdoProfileProvider.PROVIDER.id()))
.value(XFDO_PROFILE_VALUE))
.add(
attr(FDO_PROFILE_ATTR, LABEL)
.allowedRuleClasses("fdo_profile")
.mandatoryProviders(ImmutableList.of(FdoProfileProvider.PROVIDER.id()))
.value(FDO_PROFILE_VALUE))
.add(
attr(CSFDO_PROFILE_ATTR, LABEL)
.allowedRuleClasses("fdo_profile")
.mandatoryProviders(ImmutableList.of(FdoProfileProvider.PROVIDER.id()))
.value(CSFDO_PROFILE_VALUE))
.add(
attr(":fdo_prefetch_hints", LABEL)
.allowedRuleClasses("fdo_prefetch_hints")
.mandatoryProviders(ImmutableList.of(FdoPrefetchHintsProvider.PROVIDER.id()))
.value(FDO_PREFETCH_HINTS))
/* <!-- #BLAZE_RULE(cc_toolchain).ATTRIBUTE(toolchain_identifier) -->
The identifier used to match this cc_toolchain with the corresponding
crosstool_config.toolchain.
<p>
Until issue <a href="https://github.com/bazelbuild/bazel/issues/5380">#5380</a> is fixed
this is the recommended way of associating <code>cc_toolchain</code> with
<code>CROSSTOOL.toolchain</code>. It will be replaced by the <code>toolchain_config</code>
attribute (<a href="https://github.com/bazelbuild/bazel/issues/5380">#5380</a>).</p>
<!-- #END_BLAZE_RULE.ATTRIBUTE -->*/
.add(
attr("toolchain_identifier", Type.STRING)
.nonconfigurable("Used in configuration creation")
.value(""))
/* <!-- #BLAZE_RULE(cc_toolchain).ATTRIBUTE(toolchain_config) -->
The label of the rule providing <code>cc_toolchain_config_info</code>.
<!-- #END_BLAZE_RULE.ATTRIBUTE -->*/
.add(
attr(TOOLCHAIN_CONFIG_ATTR, LABEL)
.allowedFileTypes()
.mandatoryProviders(CcToolchainConfigInfo.PROVIDER.id())
.mandatory())
.build();
}
@Override
public Metadata getMetadata() {
return RuleDefinition.Metadata.builder()
.name("cc_toolchain")
.ancestors(BaseRuleClasses.BaseRule.class)
.factoryClass(CcToolchain.class)
.build();
}
}
/*<!-- #BLAZE_RULE (NAME = cc_toolchain, TYPE = OTHER, FAMILY = C / C++) -->
<p>Represents a C++ toolchain.</p>
<p>
This rule is responsible for:
<ul>
<li>
Collecting all artifacts needed for C++ actions to run. This is done by
attributes such as <code>all_files</code>, <code>compiler_files</code>,
<code>linker_files</code>, or other attributes ending with <code>_files</code>). These are
most commonly filegroups globbing all required files.
</li>
<li>
Generating correct command lines for C++ actions. This is done using
<code>CcToolchainConfigInfo</code> provider (details below).
</li>
</ul>
</p>
<p>
Use <code>toolchain_config</code> attribute to configure the C++ toolchain.
See also this
<a href="https://docs.bazel.build/versions/master/cc-toolchain-config-reference.html">
page
</a> for elaborate C++ toolchain configuration and toolchain selection documentation.
</p>
<!-- #END_BLAZE_RULE -->*/