
[g]libtoolize looks for m4 using the environment variable M4. When that isn't set, it looks for m4 in PATH. However, when libtool is a build dependency but m4 is not, m4 will not be found in PATH. Since it is currently not set in the environment by superenv, this causes some builds to fail. Closes https://github.com/Homebrew/homebrew-core/pull/73932.
369 lines
9.1 KiB
Ruby
369 lines
9.1 KiB
Ruby
# typed: true
|
|
# frozen_string_literal: true
|
|
|
|
require "extend/ENV/shared"
|
|
require "development_tools"
|
|
|
|
# ### Why `superenv`?
|
|
#
|
|
# 1. Only specify the environment we need (*NO* LDFLAGS for cmake)
|
|
# 2. Only apply compiler-specific options when we are calling that compiler
|
|
# 3. Force all incpaths and libpaths into the cc instantiation (fewer bugs)
|
|
# 4. Cater toolchain usage to specific Xcode versions
|
|
# 5. Remove flags that we don't want or that will break builds
|
|
# 6. Simpler code
|
|
# 7. Simpler formulae that *just work*
|
|
# 8. Build-system agnostic configuration of the toolchain
|
|
module Superenv
|
|
extend T::Sig
|
|
|
|
include SharedEnvExtension
|
|
|
|
# @private
|
|
attr_accessor :keg_only_deps, :deps, :run_time_deps, :x11
|
|
|
|
sig { params(base: Superenv).void }
|
|
def self.extended(base)
|
|
base.keg_only_deps = []
|
|
base.deps = []
|
|
base.run_time_deps = []
|
|
end
|
|
|
|
# @private
|
|
sig { returns(T.nilable(Pathname)) }
|
|
def self.bin; end
|
|
|
|
sig { void }
|
|
def reset
|
|
super
|
|
# Configure scripts generated by autoconf 2.61 or later export as_nl, which
|
|
# we use as a heuristic for running under configure
|
|
delete("as_nl")
|
|
end
|
|
|
|
# @private
|
|
sig {
|
|
params(
|
|
formula: T.nilable(Formula),
|
|
cc: T.nilable(String),
|
|
build_bottle: T.nilable(T::Boolean),
|
|
bottle_arch: T.nilable(String),
|
|
testing_formula: T::Boolean,
|
|
).void
|
|
}
|
|
def setup_build_environment(formula: nil, cc: nil, build_bottle: false, bottle_arch: nil, testing_formula: false)
|
|
super
|
|
send(compiler)
|
|
|
|
self["HOMEBREW_ENV"] = "super"
|
|
self["MAKEFLAGS"] ||= "-j#{determine_make_jobs}"
|
|
self["PATH"] = determine_path
|
|
self["PKG_CONFIG_PATH"] = determine_pkg_config_path
|
|
self["PKG_CONFIG_LIBDIR"] = determine_pkg_config_libdir
|
|
self["HOMEBREW_CCCFG"] = determine_cccfg
|
|
self["HOMEBREW_OPTIMIZATION_LEVEL"] = "Os"
|
|
self["HOMEBREW_BREW_FILE"] = HOMEBREW_BREW_FILE.to_s
|
|
self["HOMEBREW_PREFIX"] = HOMEBREW_PREFIX.to_s
|
|
self["HOMEBREW_CELLAR"] = HOMEBREW_CELLAR.to_s
|
|
self["HOMEBREW_OPT"] = "#{HOMEBREW_PREFIX}/opt"
|
|
self["HOMEBREW_TEMP"] = HOMEBREW_TEMP.to_s
|
|
self["HOMEBREW_OPTFLAGS"] = determine_optflags
|
|
self["HOMEBREW_ARCHFLAGS"] = ""
|
|
self["CMAKE_PREFIX_PATH"] = determine_cmake_prefix_path
|
|
self["CMAKE_FRAMEWORK_PATH"] = determine_cmake_frameworks_path
|
|
self["CMAKE_INCLUDE_PATH"] = determine_cmake_include_path
|
|
self["CMAKE_LIBRARY_PATH"] = determine_cmake_library_path
|
|
self["ACLOCAL_PATH"] = determine_aclocal_path
|
|
self["M4"] = "#{HOMEBREW_PREFIX}/opt/m4/bin/m4" if deps.any? { |d| d.name == "libtool" }
|
|
self["HOMEBREW_ISYSTEM_PATHS"] = determine_isystem_paths
|
|
self["HOMEBREW_INCLUDE_PATHS"] = determine_include_paths
|
|
self["HOMEBREW_LIBRARY_PATHS"] = determine_library_paths
|
|
self["HOMEBREW_DEPENDENCIES"] = determine_dependencies
|
|
self["HOMEBREW_FORMULA_PREFIX"] = @formula.prefix unless @formula.nil?
|
|
|
|
# The HOMEBREW_CCCFG ENV variable is used by the ENV/cc tool to control
|
|
# compiler flag stripping. It consists of a string of characters which act
|
|
# as flags. Some of these flags are mutually exclusive.
|
|
#
|
|
# O - Enables argument refurbishing. Only active under the
|
|
# make/bsdmake wrappers currently.
|
|
# x - Enable C++11 mode.
|
|
# g - Enable "-stdlib=libc++" for clang.
|
|
# h - Enable "-stdlib=libstdc++" for clang.
|
|
# K - Don't strip -arch <arch>, -m32, or -m64
|
|
# w - Pass -no_weak_imports to the linker
|
|
#
|
|
# These flags will also be present:
|
|
# s - apply fix for sed's Unicode support
|
|
# a - apply fix for apr-1-config path
|
|
end
|
|
alias generic_setup_build_environment setup_build_environment
|
|
|
|
private
|
|
|
|
sig { params(val: T.any(String, Pathname)).returns(String) }
|
|
def cc=(val)
|
|
self["HOMEBREW_CC"] = super
|
|
end
|
|
|
|
sig { params(val: T.any(String, Pathname)).returns(String) }
|
|
def cxx=(val)
|
|
self["HOMEBREW_CXX"] = super
|
|
end
|
|
|
|
sig { returns(String) }
|
|
def determine_cxx
|
|
determine_cc.to_s.gsub("gcc", "g++").gsub("clang", "clang++")
|
|
end
|
|
|
|
sig { returns(T::Array[Pathname]) }
|
|
def homebrew_extra_paths
|
|
[]
|
|
end
|
|
|
|
sig { returns(T.nilable(PATH)) }
|
|
def determine_path
|
|
path = PATH.new(Superenv.bin)
|
|
|
|
# Formula dependencies can override standard tools.
|
|
path.append(deps.map(&:opt_bin))
|
|
path.append(homebrew_extra_paths)
|
|
path.append("/usr/bin", "/bin", "/usr/sbin", "/sbin")
|
|
|
|
begin
|
|
path.append(gcc_version_formula(T.must(homebrew_cc)).opt_bin) if homebrew_cc&.match?(GNU_GCC_REGEXP)
|
|
rescue FormulaUnavailableError
|
|
# Don't fail and don't add these formulae to the path if they don't exist.
|
|
nil
|
|
end
|
|
|
|
path.existing
|
|
end
|
|
|
|
sig { returns(T::Array[Pathname]) }
|
|
def homebrew_extra_pkg_config_paths
|
|
[]
|
|
end
|
|
|
|
sig { returns(T.nilable(PATH)) }
|
|
def determine_pkg_config_path
|
|
PATH.new(
|
|
deps.map { |d| d.opt_lib/"pkgconfig" },
|
|
deps.map { |d| d.opt_share/"pkgconfig" },
|
|
).existing
|
|
end
|
|
|
|
sig { returns(T.nilable(PATH)) }
|
|
def determine_pkg_config_libdir
|
|
PATH.new(
|
|
homebrew_extra_pkg_config_paths,
|
|
).existing
|
|
end
|
|
|
|
sig { returns(T::Array[Pathname]) }
|
|
def homebrew_extra_aclocal_paths
|
|
[]
|
|
end
|
|
|
|
sig { returns(T.nilable(PATH)) }
|
|
def determine_aclocal_path
|
|
PATH.new(
|
|
keg_only_deps.map { |d| d.opt_share/"aclocal" },
|
|
HOMEBREW_PREFIX/"share/aclocal",
|
|
homebrew_extra_aclocal_paths,
|
|
).existing
|
|
end
|
|
|
|
sig { returns(T::Array[Pathname]) }
|
|
def homebrew_extra_isystem_paths
|
|
[]
|
|
end
|
|
|
|
sig { returns(T.nilable(PATH)) }
|
|
def determine_isystem_paths
|
|
PATH.new(
|
|
HOMEBREW_PREFIX/"include",
|
|
homebrew_extra_isystem_paths,
|
|
).existing
|
|
end
|
|
|
|
sig { returns(T.nilable(PATH)) }
|
|
def determine_include_paths
|
|
PATH.new(keg_only_deps.map(&:opt_include)).existing
|
|
end
|
|
|
|
sig { returns(T::Array[Pathname]) }
|
|
def homebrew_extra_library_paths
|
|
[]
|
|
end
|
|
|
|
sig { returns(T.nilable(PATH)) }
|
|
def determine_library_paths
|
|
paths = [
|
|
keg_only_deps.map(&:opt_lib),
|
|
HOMEBREW_PREFIX/"lib",
|
|
]
|
|
paths += homebrew_extra_library_paths
|
|
PATH.new(paths).existing
|
|
end
|
|
|
|
sig { returns(String) }
|
|
def determine_dependencies
|
|
deps.map(&:name).join(",")
|
|
end
|
|
|
|
sig { returns(T.nilable(PATH)) }
|
|
def determine_cmake_prefix_path
|
|
PATH.new(
|
|
keg_only_deps.map(&:opt_prefix),
|
|
HOMEBREW_PREFIX.to_s,
|
|
).existing
|
|
end
|
|
|
|
sig { returns(T::Array[Pathname]) }
|
|
def homebrew_extra_cmake_include_paths
|
|
[]
|
|
end
|
|
|
|
sig { returns(T.nilable(PATH)) }
|
|
def determine_cmake_include_path
|
|
PATH.new(homebrew_extra_cmake_include_paths).existing
|
|
end
|
|
|
|
sig { returns(T::Array[Pathname]) }
|
|
def homebrew_extra_cmake_library_paths
|
|
[]
|
|
end
|
|
|
|
sig { returns(T.nilable(PATH)) }
|
|
def determine_cmake_library_path
|
|
PATH.new(homebrew_extra_cmake_library_paths).existing
|
|
end
|
|
|
|
sig { returns(T::Array[Pathname]) }
|
|
def homebrew_extra_cmake_frameworks_paths
|
|
[]
|
|
end
|
|
|
|
sig { returns(T.nilable(PATH)) }
|
|
def determine_cmake_frameworks_path
|
|
PATH.new(
|
|
deps.map(&:opt_frameworks),
|
|
homebrew_extra_cmake_frameworks_paths,
|
|
).existing
|
|
end
|
|
|
|
sig { returns(String) }
|
|
def determine_make_jobs
|
|
Homebrew::EnvConfig.make_jobs
|
|
end
|
|
|
|
sig { returns(String) }
|
|
def determine_optflags
|
|
Hardware::CPU.optimization_flags.fetch(effective_arch)
|
|
end
|
|
|
|
sig { returns(String) }
|
|
def determine_cccfg
|
|
""
|
|
end
|
|
|
|
public
|
|
|
|
# Removes the MAKEFLAGS environment variable, causing make to use a single job.
|
|
# This is useful for makefiles with race conditions.
|
|
# When passed a block, MAKEFLAGS is removed only for the duration of the block and is restored after its completion.
|
|
sig { params(block: T.proc.returns(T.untyped)).returns(T.untyped) }
|
|
def deparallelize(&block)
|
|
old = delete("MAKEFLAGS")
|
|
if block
|
|
begin
|
|
yield
|
|
ensure
|
|
self["MAKEFLAGS"] = old
|
|
end
|
|
end
|
|
|
|
old
|
|
end
|
|
|
|
sig { returns(Integer) }
|
|
def make_jobs
|
|
self["MAKEFLAGS"] =~ /-\w*j(\d+)/
|
|
[Regexp.last_match(1).to_i, 1].max
|
|
end
|
|
|
|
sig { void }
|
|
def universal_binary
|
|
odisabled "ENV.universal_binary"
|
|
|
|
check_for_compiler_universal_support
|
|
|
|
self["HOMEBREW_ARCHFLAGS"] = Hardware::CPU.universal_archs.as_arch_flags
|
|
end
|
|
|
|
sig { void }
|
|
def permit_arch_flags
|
|
append_to_cccfg "K"
|
|
end
|
|
|
|
sig { void }
|
|
def m32
|
|
odisabled "ENV.m32"
|
|
|
|
append "HOMEBREW_ARCHFLAGS", "-m32"
|
|
end
|
|
|
|
sig { void }
|
|
def m64
|
|
odisabled "ENV.m64"
|
|
|
|
append "HOMEBREW_ARCHFLAGS", "-m64"
|
|
end
|
|
|
|
sig { void }
|
|
def cxx11
|
|
append_to_cccfg "x"
|
|
append_to_cccfg "g" if homebrew_cc == "clang"
|
|
end
|
|
|
|
sig { void }
|
|
def libcxx
|
|
append_to_cccfg "g" if compiler == :clang
|
|
end
|
|
|
|
sig { void }
|
|
def libstdcxx
|
|
odisabled "ENV.libstdcxx"
|
|
|
|
append_to_cccfg "h" if compiler == :clang
|
|
end
|
|
|
|
# @private
|
|
sig { void }
|
|
def refurbish_args
|
|
append_to_cccfg "O"
|
|
end
|
|
|
|
%w[O3 O2 Os].each do |opt|
|
|
define_method opt do
|
|
odisabled "ENV.#{opt}"
|
|
|
|
send(:[]=, "HOMEBREW_OPTIMIZATION_LEVEL", opt)
|
|
end
|
|
end
|
|
|
|
%w[O1 O0].each do |opt|
|
|
define_method opt do
|
|
send(:[]=, "HOMEBREW_OPTIMIZATION_LEVEL", opt)
|
|
end
|
|
end
|
|
|
|
sig { void }
|
|
def set_x11_env_if_installed
|
|
odisabled "ENV.set_x11_env_if_installed"
|
|
end
|
|
end
|
|
|
|
require "extend/os/extend/ENV/super"
|