rules_foreign_cc/foreign_cc/make.bzl

180 lines
6.8 KiB
Python

"""A rule for building projects using the [GNU Make](https://www.gnu.org/software/make/) build tool"""
load("@bazel_tools//tools/cpp:toolchain_utils.bzl", "find_cpp_toolchain")
load(
"//foreign_cc/private:cc_toolchain_util.bzl",
"get_flags_info",
"get_tools_info",
)
load(
"//foreign_cc/private:detect_root.bzl",
"detect_root",
)
load(
"//foreign_cc/private:framework.bzl",
"CC_EXTERNAL_RULE_ATTRIBUTES",
"CC_EXTERNAL_RULE_FRAGMENTS",
"cc_external_rule_impl",
"create_attrs",
"expand_locations_and_make_variables",
)
load("//foreign_cc/private:make_script.bzl", "create_make_script")
load("//foreign_cc/private:transitions.bzl", "foreign_cc_rule_variant")
load("//toolchains/native_tools:tool_access.bzl", "get_make_data")
def _make(ctx):
make_data = get_make_data(ctx)
tools_data = [make_data]
attrs = create_attrs(
ctx.attr,
configure_name = "Make",
create_configure_script = _create_make_script,
tools_data = tools_data,
make_path = make_data.path,
)
return cc_external_rule_impl(ctx, attrs)
def _create_make_script(configureParameters):
ctx = configureParameters.ctx
attrs = configureParameters.attrs
inputs = configureParameters.inputs
root = detect_root(ctx.attr.lib_source)
tools = get_tools_info(ctx)
flags = get_flags_info(ctx)
data = ctx.attr.data + ctx.attr.build_data
# Generate a list of arguments for make
args = " ".join([
ctx.expand_location(arg, data)
for arg in ctx.attr.args
])
user_env = expand_locations_and_make_variables(ctx, ctx.attr.env, "env", data)
make_commands = []
prefix = "{} ".format(expand_locations_and_make_variables(ctx, attrs.tool_prefix, "tool_prefix", data)) if attrs.tool_prefix else ""
for target in ctx.attr.targets:
make_commands.append("{prefix}{make} {target} {args} PREFIX={install_prefix}".format(
prefix = prefix,
make = attrs.make_path,
args = args,
target = target,
install_prefix = ctx.attr.install_prefix,
))
cc_toolchain = find_cpp_toolchain(ctx)
is_msvc = cc_toolchain.compiler == "msvc-cl"
return create_make_script(
workspace_name = ctx.workspace_name,
tools = tools,
flags = flags,
root = root,
deps = ctx.attr.deps,
inputs = inputs,
env_vars = user_env,
make_prefix = prefix,
make_path = attrs.make_path,
make_targets = ctx.attr.targets,
make_args = args,
make_install_prefix = ctx.attr.install_prefix,
executable_ldflags_vars = ctx.attr.executable_ldflags_vars,
shared_ldflags_vars = ctx.attr.shared_ldflags_vars,
is_msvc = is_msvc,
)
def _attrs():
attrs = dict(CC_EXTERNAL_RULE_ATTRIBUTES)
attrs.update({
"args": attr.string_list(
doc = "A list of arguments to pass to the call to `make`",
),
"executable_ldflags_vars": attr.string_list(
doc = (
"A string list of variable names use as LDFLAGS for executables. These variables " +
"will be passed to the make command as make vars and overwrite what is defined in " +
"the Makefile."
),
mandatory = False,
default = [],
),
"install_prefix": attr.string(
doc = (
"Install prefix, i.e. relative path to where to install the result of the build. " +
"Passed as an arg to \"make\" as PREFIX=<install_prefix>."
),
mandatory = False,
default = "$$INSTALLDIR$$",
),
"shared_ldflags_vars": attr.string_list(
doc = (
"A string list of variable names use as LDFLAGS for shared libraries. These variables " +
"will be passed to the make command as make vars and overwrite what is defined in " +
"the Makefile."
),
mandatory = False,
default = [],
),
"targets": attr.string_list(
doc = (
"A list of targets within the foreign build system to produce. An empty string (`\"\"`) will result in " +
"a call to the underlying build system with no explicit target set. However, in order to extract build " +
"outputs, you must execute at least an equivalent of make install, and have your make file copy the build " +
"outputs into the directory specified by `install_prefix`."
),
mandatory = False,
default = ["", "install"],
),
})
return attrs
make = rule(
doc = (
"Rule for building external libraries with GNU Make. " +
"GNU Make commands (make and make install by default) are invoked with PREFIX=\"install\" " +
"(by default), and other environment variables for compilation and linking, taken from Bazel C/C++ " +
"toolchain and passed dependencies. " +
"Not all Makefiles will work equally well here, and some may require patching." +
"Your Makefile must either support passing the install prefix using the PREFIX flag, or " +
"it needs to have a different way to pass install prefix to it. An equivalent of " +
"make install MUST be specified as one of the targets." +
"This is because all the paths with param names prefixed by out_* are expressed " +
"as relative to INSTALLDIR, not the source directory." +
"That is, if you execute only make, but not make install, this rule will not be able " +
"to pick up any build outputs. Finally, your make install rule must dereference symlinks " +
"to ensure that the installed files don't end up being symlinks to files in the sandbox. " +
"For example, installation lines like `cp $SOURCE $DEST` must become `cp -L $SOURCE $DEST`, " +
"as the -L will ensure that symlinks are dereferenced."
),
attrs = _attrs(),
fragments = CC_EXTERNAL_RULE_FRAGMENTS,
output_to_genfiles = True,
provides = [CcInfo],
implementation = _make,
toolchains = [
"@rules_foreign_cc//toolchains:make_toolchain",
"@rules_foreign_cc//foreign_cc/private/framework:shell_toolchain",
"@bazel_tools//tools/cpp:toolchain_type",
],
)
def make_variant(name, toolchain, **kwargs):
""" Wrapper macro around the make() rule to force usage of the given make variant toolchain.
Args:
name: The target name
toolchain: The desired make variant toolchain to use, e.g. @rules_foreign_cc//toolchains:preinstalled_nmake_toolchain
**kwargs: Remaining keyword arguments
"""
foreign_cc_rule_variant(
name = name,
rule = make,
toolchain = toolchain,
**kwargs
)