Skip to content

Commit

Permalink
feat: configurable ldflags
Browse files Browse the repository at this point in the history
  • Loading branch information
jsun-splunk committed Jul 17, 2024
1 parent 9d5727d commit 698b47d
Show file tree
Hide file tree
Showing 5 changed files with 140 additions and 26 deletions.
34 changes: 23 additions & 11 deletions foreign_cc/configure.bzl
Original file line number Diff line number Diff line change
Expand Up @@ -73,20 +73,10 @@ def _create_configure_script(configureParameters):

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 ""
configure_prefix = "{} ".format(expand_locations_and_make_variables(ctx, ctx.attr.configure_prefix, "configure_prefix", data)) if ctx.attr.configure_prefix else ""
configure_options = [expand_locations_and_make_variables(ctx, option, "configure_option", data) for option in ctx.attr.configure_options] if ctx.attr.configure_options else []

for target in ctx.attr.targets:
# Configure will have generated sources into `$BUILD_TMPDIR` so make sure we `cd` there
make_commands.append("{prefix}{make} {target} {args}".format(
prefix = prefix,
make = attrs.make_path,
args = args,
target = target,
))

configure = create_configure_script(
workspace_name = ctx.workspace_name,
tools = tools,
Expand All @@ -107,8 +97,12 @@ def _create_configure_script(configureParameters):
autogen = ctx.attr.autogen,
autogen_command = ctx.attr.autogen_command,
autogen_options = ctx.attr.autogen_options,
make_commands = make_commands,
make_prefix = prefix,
make_path = attrs.make_path,
make_targets = ctx.attr.targets,
make_args = args,
executable_ldflags_vars = ctx.attr.executable_ldflags_vars,
shared_ldflags_vars = ctx.attr.shared_ldflags_vars,
)
return define_install_prefix + configure

Expand Down Expand Up @@ -187,6 +181,15 @@ def _attrs():
"configure_prefix": attr.string(
doc = "A prefix for the call to the `configure_command`.",
),
"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. " +
Expand All @@ -201,6 +204,15 @@ def _attrs():
mandatory = False,
default = "--prefix=",
),
"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 " +
Expand Down
26 changes: 25 additions & 1 deletion foreign_cc/make.bzl
Original file line number Diff line number Diff line change
Expand Up @@ -74,7 +74,13 @@ def _create_make_script(configureParameters):
deps = ctx.attr.deps,
inputs = inputs,
env_vars = user_env,
make_commands = make_commands,
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,
)

def _attrs():
Expand All @@ -83,6 +89,15 @@ def _attrs():
"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. " +
Expand All @@ -91,6 +106,15 @@ def _attrs():
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 " +
Expand Down
21 changes: 19 additions & 2 deletions foreign_cc/private/configure_script.bzl
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
# buildifier: disable=module-docstring
load(":make_env_vars.bzl", "get_make_env_vars")
load(":make_env_vars.bzl", "get_ldflags_make_vars", "get_make_env_vars")
load(":make_script.bzl", "pkgconfig_script")

# buildifier: disable=function-docstring
Expand All @@ -23,8 +23,12 @@ def create_configure_script(
autogen,
autogen_command,
autogen_options,
make_prefix,
make_path,
make_commands):
make_targets,
make_args,
executable_ldflags_vars,
shared_ldflags_vars):
ext_build_dirs = inputs.ext_build_dirs

script = pkgconfig_script(ext_build_dirs)
Expand Down Expand Up @@ -77,6 +81,19 @@ def create_configure_script(
user_options = " ".join(user_options),
))

ldflags_make_vars = get_ldflags_make_vars(executable_ldflags_vars, shared_ldflags_vars, workspace_name, flags, env_vars, deps, inputs)

make_commands = []
for target in make_targets:
# Configure will have generated sources into `$BUILD_TMPDIR` so make sure we `cd` there
make_commands.append("{prefix}{make} {make_vars} {target} {args}".format(
prefix = make_prefix,
make = make_path,
make_vars = ldflags_make_vars,
args = make_args,
target = target,
))

script.extend(make_commands)
script.append("##disable_tracing##")

Expand Down
60 changes: 50 additions & 10 deletions foreign_cc/private/make_env_vars.bzl
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,24 @@ def get_make_env_vars(
return " ".join(["{}=\"{}\""
.format(key, _join_flags_list(workspace_name, vars[key])) for key in vars])

# buildifier: disable=function-docstring
def get_ldflags_make_vars(
executable_ldflags_vars,
shared_ldflags_vars,
workspace_name,
flags,
user_vars,
deps,
inputs):
vars = _get_user_ldflags_variables(executable_ldflags_vars, shared_ldflags_vars, flags, user_vars)

deps_flags = _define_deps_flags(deps, inputs)
for key in vars.keys():
vars[key] = vars[key] + deps_flags.libs

return " ".join(["{}=\"{}\""
.format(key, _join_flags_list(workspace_name, vars[key])) for key in vars])

def _define_deps_flags(deps, inputs):
# It is very important to keep the order for the linker => put them into list
lib_dirs = []
Expand Down Expand Up @@ -96,18 +114,10 @@ _MAKE_TOOLS = {
}

def _get_make_variables(workspace_name, tools, flags, user_env_vars, make_commands):
vars = {}

for flag in _MAKE_FLAGS:
flag_value = getattr(flags, _MAKE_FLAGS[flag])
if flag_value:
vars[flag] = flag_value
vars = _expand_flags(flags, _MAKE_FLAGS)

# Merge flags lists
for user_var in user_env_vars:
toolchain_val = vars.get(user_var)
if toolchain_val:
vars[user_var] = toolchain_val + [user_env_vars[user_var]]
vars = _merge_vars_left(vars, user_env_vars)

tools_dict = {}
for tool in _MAKE_TOOLS:
Expand Down Expand Up @@ -139,6 +149,36 @@ def _get_make_variables(workspace_name, tools, flags, user_env_vars, make_comman

return vars

def _get_user_ldflags_variables(
executable_ldflags_vars,
shared_ldflags_vars,
flags,
user_env_vars):
executable_ldflags = {var: "cxx_linker_executable" for var in executable_ldflags_vars}
shared_ldflags = {var: "cxx_linker_shared" for var in shared_ldflags_vars}

ldflags = {}
for vars in [executable_ldflags, shared_ldflags]:
ldflags.update(vars)

ldflags_vars = _expand_flags(flags, ldflags)
return _merge_vars_left(ldflags_vars, user_env_vars)

def _expand_flags(flags, flags_vars):
vars = {}
for flag in flags_vars:
flag_value = getattr(flags, flags_vars[flag])
if flag_value:
vars[flag] = flag_value
return vars

def _merge_vars_left(vars, vars_to_merge):
for key in vars_to_merge:
value = vars.get(key)
if value:
vars[key] = value + [vars_to_merge[key]]
return vars

def _absolutize(workspace_name, text, force = False):
return absolutize_path_in_str(workspace_name, "$$EXT_BUILD_ROOT$$/", text, force)

Expand Down
25 changes: 23 additions & 2 deletions foreign_cc/private/make_script.bzl
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
"""A module for creating the build script for `make` builds"""

load(":make_env_vars.bzl", "get_make_env_vars")
load(":make_env_vars.bzl", "get_ldflags_make_vars", "get_make_env_vars")

# buildifier: disable=function-docstring
def create_make_script(
Expand All @@ -11,15 +11,36 @@ def create_make_script(
env_vars,
deps,
inputs,
make_commands):
make_prefix,
make_path,
make_targets,
make_args,
make_install_prefix,
executable_ldflags_vars,
shared_ldflags_vars):
ext_build_dirs = inputs.ext_build_dirs

script = pkgconfig_script(ext_build_dirs)

script.append("##symlink_contents_to_dir## $$EXT_BUILD_ROOT$$/{} $$BUILD_TMPDIR$$ False".format(root))

script.append("##enable_tracing##")

ldflags_make_vars = get_ldflags_make_vars(executable_ldflags_vars, shared_ldflags_vars, workspace_name, flags, env_vars, deps, inputs)

make_commands = []
for target in make_targets:
make_commands.append("{prefix}{make} {make_vars} {target} {args} PREFIX={install_prefix}".format(
prefix = make_prefix,
make = make_path,
make_vars = ldflags_make_vars,
args = make_args,
target = target,
install_prefix = make_install_prefix,
))

configure_vars = get_make_env_vars(workspace_name, tools, flags, env_vars, deps, inputs, make_commands)

script.extend(["{env_vars} {command}".format(
env_vars = configure_vars,
command = command,
Expand Down

0 comments on commit 698b47d

Please sign in to comment.