diff --git a/docs/common_settings_doc.md b/docs/common_settings_doc.md index 7de79659..b224f529 100755 --- a/docs/common_settings_doc.md +++ b/docs/common_settings_doc.md @@ -50,7 +50,7 @@ A bool-typed build setting that cannot be set on the command line ## int_flag
-int_flag(name)
+int_flag(name, make_variable)
 
An int-typed build setting that can be set on the command line @@ -61,6 +61,7 @@ An int-typed build setting that can be set on the command line | Name | Description | Type | Mandatory | Default | | :------------- | :------------- | :------------- | :------------- | :------------- | | name | A unique name for this target. | Name | required | | +| make_variable | If set, the build setting's value will be available as a Make variable with this name in the attributes of rules that list this build setting in their 'toolchains' attribute. | String | optional | "" | @@ -68,7 +69,7 @@ An int-typed build setting that can be set on the command line ## int_setting
-int_setting(name)
+int_setting(name, make_variable)
 
An int-typed build setting that cannot be set on the command line @@ -79,6 +80,7 @@ An int-typed build setting that cannot be set on the command line | Name | Description | Type | Mandatory | Default | | :------------- | :------------- | :------------- | :------------- | :------------- | | name | A unique name for this target. | Name | required | | +| make_variable | If set, the build setting's value will be available as a Make variable with this name in the attributes of rules that list this build setting in their 'toolchains' attribute. | String | optional | "" | @@ -86,7 +88,7 @@ An int-typed build setting that cannot be set on the command line ## string_flag
-string_flag(name, values)
+string_flag(name, make_variable, values)
 
A string-typed build setting that can be set on the command line @@ -97,6 +99,7 @@ A string-typed build setting that can be set on the command line | Name | Description | Type | Mandatory | Default | | :------------- | :------------- | :------------- | :------------- | :------------- | | name | A unique name for this target. | Name | required | | +| make_variable | If set, the build setting's value will be available as a Make variable with this name in the attributes of rules that list this build setting in their 'toolchains' attribute. | String | optional | "" | | values | The list of allowed values for this setting. An error is raised if any other value is given. | List of strings | optional | [] | @@ -141,7 +144,7 @@ A string list-typed build setting that cannot be set on the command line ## string_setting
-string_setting(name, values)
+string_setting(name, make_variable, values)
 
A string-typed build setting that cannot be set on the command line @@ -152,6 +155,7 @@ A string-typed build setting that cannot be set on the command line | Name | Description | Type | Mandatory | Default | | :------------- | :------------- | :------------- | :------------- | :------------- | | name | A unique name for this target. | Name | required | | +| make_variable | If set, the build setting's value will be available as a Make variable with this name in the attributes of rules that list this build setting in their 'toolchains' attribute. | String | optional | "" | | values | The list of allowed values for this setting. An error is raised if any other value is given. | List of strings | optional | [] | diff --git a/rules/common_settings.bzl b/rules/common_settings.bzl index eb057cce..58a15aee 100644 --- a/rules/common_settings.bzl +++ b/rules/common_settings.bzl @@ -30,18 +30,52 @@ BuildSettingInfo = provider( }, ) +_MAKE_VARIABLE_ATTR = attr.string( + doc = "If set, the build setting's value will be available as a Make variable with this " + + "name in the attributes of rules that list this build setting in their 'toolchains' " + + "attribute.", +) + +def _is_valid_make_variable_char(c): + # Restrict make variable names for consistency with predefined ones. There are no enforced + # restrictions on make variable names, but when they contain e.g. spaces or braces, they + # aren't expanded by e.g. cc_binary. + return c == "_" or c.isdigit() or (c.isalpha() and c.isupper()) + +def _get_template_variable_info(ctx): + make_variable = getattr(ctx.attr, "make_variable", None) + if not make_variable: + return [] + + if not all([_is_valid_make_variable_char(c) for c in make_variable.elems()]): + fail("Error setting " + _no_at_str(ctx.label) + ": invalid make variable name '" + make_variable + "'. Make variable names may only contain uppercase letters, digits, and underscores.") + + return [ + platform_common.TemplateVariableInfo({ + make_variable: str(ctx.build_setting_value), + }), + ] + def _impl(ctx): - return BuildSettingInfo(value = ctx.build_setting_value) + return [ + BuildSettingInfo(value = ctx.build_setting_value), + ] + _get_template_variable_info(ctx) int_flag = rule( implementation = _impl, build_setting = config.int(flag = True), + attrs = { + "make_variable": _MAKE_VARIABLE_ATTR, + }, doc = "An int-typed build setting that can be set on the command line", ) int_setting = rule( implementation = _impl, build_setting = config.int(), + attrs = { + "make_variable": _MAKE_VARIABLE_ATTR, + }, doc = "An int-typed build setting that cannot be set on the command line", ) @@ -82,7 +116,7 @@ def _string_impl(ctx): allowed_values = ctx.attr.values value = ctx.build_setting_value if len(allowed_values) == 0 or value in ctx.attr.values: - return BuildSettingInfo(value = value) + return [BuildSettingInfo(value = value)] + _get_template_variable_info(ctx) else: fail("Error setting " + _no_at_str(ctx.label) + ": invalid value '" + value + "'. Allowed values are " + str(allowed_values)) @@ -93,6 +127,7 @@ string_flag = rule( "values": attr.string_list( doc = "The list of allowed values for this setting. An error is raised if any other value is given.", ), + "make_variable": _MAKE_VARIABLE_ATTR, }, doc = "A string-typed build setting that can be set on the command line", ) @@ -104,6 +139,7 @@ string_setting = rule( "values": attr.string_list( doc = "The list of allowed values for this setting. An error is raised if any other value is given.", ), + "make_variable": _MAKE_VARIABLE_ATTR, }, doc = "A string-typed build setting that cannot be set on the command line", ) diff --git a/tests/BUILD b/tests/BUILD index bbab077a..7f056d2b 100644 --- a/tests/BUILD +++ b/tests/BUILD @@ -1,6 +1,7 @@ load("//:bzl_library.bzl", "bzl_library") load(":build_test_tests.bzl", "build_test_test_suite") load(":collections_tests.bzl", "collections_test_suite") +load(":common_settings_tests.bzl", "common_settings_test_suite") load(":dicts_tests.bzl", "dicts_test_suite") load(":new_sets_tests.bzl", "new_sets_test_suite") load(":partial_tests.bzl", "partial_test_suite") @@ -24,6 +25,8 @@ build_test_test_suite() collections_test_suite() +common_settings_test_suite() + dicts_test_suite() new_sets_test_suite() diff --git a/tests/common_settings/BUILD b/tests/common_settings/BUILD new file mode 100644 index 00000000..bbf32d55 --- /dev/null +++ b/tests/common_settings/BUILD @@ -0,0 +1,25 @@ +load("//rules:common_settings.bzl", "int_flag", "string_flag") + +int_flag( + name = "my_int_flag", + build_setting_default = 42, + make_variable = "MY_INT_FLAG", +) + +string_flag( + name = "my_string_flag", + build_setting_default = "foo", + make_variable = "MY_STRING_FLAG", +) + +sh_test( + name = "make_variable_test", + srcs = ["make_variable_test.sh"], + env = { + "MESSAGE": "Hello, $(MY_STRING_FLAG)! My name is $(MY_INT_FLAG).", + }, + toolchains = [ + ":my_int_flag", + ":my_string_flag", + ], +) diff --git a/tests/common_settings/make_variable_test.sh b/tests/common_settings/make_variable_test.sh new file mode 100755 index 00000000..546c53d5 --- /dev/null +++ b/tests/common_settings/make_variable_test.sh @@ -0,0 +1,17 @@ +#!/usr/bin/env bash + +# Copyright 2023 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. + +[[ "$MESSAGE" == "Hello, foo! My name is 42." ]] diff --git a/tests/common_settings_tests.bzl b/tests/common_settings_tests.bzl new file mode 100644 index 00000000..efbe0e0b --- /dev/null +++ b/tests/common_settings_tests.bzl @@ -0,0 +1,167 @@ +# Copyright 2023 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. + +"""Analysis tests for common_settings.bzl.""" + +load("//lib:unittest.bzl", "analysistest", "asserts") +load("//rules:common_settings.bzl", "int_flag", "int_setting", "string_flag", "string_setting") + +def _template_variable_info_contents_test_impl(ctx): + env = analysistest.begin(ctx) + + target_under_test = analysistest.target_under_test(env) + if ctx.attr.expected: + asserts.equals( + env, + expected = ctx.attr.expected, + actual = target_under_test[platform_common.TemplateVariableInfo].variables, + ) + else: + asserts.false(env, platform_common.TemplateVariableInfo in target_under_test) + + return analysistest.end(env) + +_template_variable_info_contents_test = analysistest.make( + _template_variable_info_contents_test_impl, + attrs = { + "expected": attr.string_dict(), + }, +) + +def _test_template_variable_info_contents(): + int_flag( + name = "my_int_flag", + build_setting_default = 42, + make_variable = "MY_INT_1", + ) + + _template_variable_info_contents_test( + name = "my_int_flag_test", + target_under_test = ":my_int_flag", + expected = { + "MY_INT_1": "42", + }, + ) + + int_setting( + name = "my_int_setting", + build_setting_default = 21, + make_variable = "MY_INT_2", + ) + + _template_variable_info_contents_test( + name = "my_int_setting_test", + target_under_test = ":my_int_setting", + expected = { + "MY_INT_2": "21", + }, + ) + + string_flag( + name = "my_string_flag", + build_setting_default = "foo", + make_variable = "MY_STRING_1", + ) + + _template_variable_info_contents_test( + name = "my_string_flag_test", + target_under_test = ":my_string_flag", + expected = { + "MY_STRING_1": "foo", + }, + ) + + string_setting( + name = "my_string_setting", + build_setting_default = "bar", + make_variable = "MY_STRING_2", + ) + + _template_variable_info_contents_test( + name = "my_string_setting_test", + target_under_test = ":my_string_setting", + expected = { + "MY_STRING_2": "bar", + }, + ) + + string_flag( + name = "my_string_flag_without_make_variable", + build_setting_default = "foo", + ) + + _template_variable_info_contents_test( + name = "my_string_flag_without_make_variable_test", + target_under_test = ":my_string_flag_without_make_variable", + expected = {}, + ) + +def _failure_test_impl(ctx): + env = analysistest.begin(ctx) + + asserts.expect_failure(env, ctx.attr.expected_failure) + + return analysistest.end(env) + +_failure_test = analysistest.make( + _failure_test_impl, + attrs = { + "expected_failure": attr.string(), + }, + expect_failure = True, +) + +def _test_make_variable_name_failures(): + int_flag( + name = "my_failing_int_flag", + build_setting_default = 42, + make_variable = "my_int_1", + tags = ["manual"], + ) + + _failure_test( + name = "my_failing_int_flag_test", + target_under_test = ":my_failing_int_flag", + expected_failure = "Error setting //tests:my_failing_int_flag: invalid make variable name 'my_int_1'. Make variable names may only contain uppercase letters, digits, and underscores.", + ) + + string_flag( + name = "my_failing_string_flag", + build_setting_default = "foo", + make_variable = "MY STRING", + tags = ["manual"], + ) + + _failure_test( + name = "my_failing_string_flag_test", + target_under_test = ":my_failing_string_flag", + expected_failure = "Error setting //tests:my_failing_string_flag: invalid make variable name 'MY STRING'. Make variable names may only contain uppercase letters, digits, and underscores.", + ) + +def common_settings_test_suite(name = "common_settings_test_suite"): + _test_template_variable_info_contents() + _test_make_variable_name_failures() + + native.test_suite( + name = "common_settings_test_suite", + tests = [ + "my_int_flag_test", + "my_int_setting_test", + "my_string_flag_test", + "my_string_setting_test", + "my_string_flag_without_make_variable_test", + "my_failing_int_flag_test", + "my_failing_string_flag_test", + ], + ) diff --git a/tests/subpackages_tests.bzl b/tests/subpackages_tests.bzl index 07a52e97..3c494d68 100644 --- a/tests/subpackages_tests.bzl +++ b/tests/subpackages_tests.bzl @@ -21,6 +21,7 @@ def _all_test(env): """Unit tests for subpackages.all.""" all_pkgs = [ + "common_settings", "copy_directory", "copy_file", "diff_test", @@ -38,6 +39,7 @@ def _all_test(env): # These exist in all cases filtered_pkgs = [ + "common_settings", "copy_directory", "copy_file", "expand_template",