# typed: true # rubocop:todo Sorbet/StrictSigil # 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 include SharedEnvExtension attr_accessor :keg_only_deps, :deps, :run_time_deps sig { params(base: Superenv).void } def self.extended(base) base.keg_only_deps = [] base.deps = [] base.run_time_deps = [] end # The location of Homebrew's shims. # # @api public sig { returns(Pathname) } def self.shims_path HOMEBREW_SHIMS_PATH/"super" end 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 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, debug_symbols: T.nilable(T::Boolean), ).void } def setup_build_environment(formula: nil, cc: nil, build_bottle: false, bottle_arch: nil, testing_formula: false, debug_symbols: false) super send(compiler) self["HOMEBREW_ENV"] = "super" self["MAKEFLAGS"] ||= "-j#{determine_make_jobs}" self["RUSTFLAGS"] = Hardware.rustflags_target_cpu(effective_arch) 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["HOMEBREW_MAKE_JOBS"] = determine_make_jobs.to_s 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? # Prevent the OpenSSL rust crate from building a vendored OpenSSL. # https://github.com/sfackler/rust-openssl/blob/994e5ff8c63557ab2aa85c85cc6956b0b0216ca7/openssl/src/lib.rs#L65 self["OPENSSL_NO_VENDOR"] = "1" # Prevent Go from automatically downloading a newer toolchain than the one that we have. # https://tip.golang.org/doc/toolchain self["GOTOOLCHAIN"] = "local" # Prevent Python packages from using bundled libraries by default. # Currently for hidapi, pyzmq and pynacl self["HIDAPI_SYSTEM_HIDAPI"] = "1" self["PYZMQ_NO_BUNDLE"] = "1" self["SODIUM_INSTALL"] = "system" set_debug_symbols if debug_symbols # 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 , -m32, or -m64 # d - Don't strip -march=. Use only in formulae that # have runtime detection of CPU features. # D - Generate debugging information # w - Pass `-no_weak_imports` to the linker # f - Pass `-no_fixup_chains` to `ld` whenever it # is invoked with `-undefined dynamic_lookup` # o - Pass `-oso_prefix` to `ld` whenever it is invoked # c - Pass `-ld_classic` to `ld` whenever it is invoked # with `-dead_strip_dylibs` # b - Pass `-mbranch-protection=standard` to the compiler # # These flags will also be present: # 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 # Reverse sort by version so that we prefer the newest when there are multiple. deps.select { |d| d.name.match? Version.formula_optionally_versioned_regex(:python) } .sort_by(&:version) .reverse .map { |d| d.opt_libexec/"bin" } end alias generic_homebrew_extra_paths homebrew_extra_paths 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.nilable(PATH)) } def determine_aclocal_path PATH.new( keg_only_deps.map { |d| d.opt_share/"aclocal" }, HOMEBREW_PREFIX/"share/aclocal", ).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 = [] if compiler.match?(GNU_GCC_REGEXP) # Add path to GCC runtime libs for version being used to compile, # so that the linker will find those libs before any that may be linked in $HOMEBREW_PREFIX/lib. # https://github.com/Homebrew/brew/pull/11459#issuecomment-851075936 begin f = gcc_version_formula(compiler.to_s) rescue FormulaUnavailableError nil else paths << (f.opt_lib/"gcc"/f.version.major) if f.any_version_installed? end end # Don't add `llvm` to library paths; this leads to undesired linkage to LLVM's `libunwind` paths << keg_only_deps.reject { |dep| dep.name.match?(/^llvm(@\d+)?$/) } .map(&:opt_lib) paths << (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) rescue KeyError odebug "Building a bottle for custom architecture (#{effective_arch})..." Hardware::CPU.arch_flag(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.nilable(T.proc.returns(T.untyped))).returns(T.untyped) } def deparallelize(&block) old_makeflags = delete("MAKEFLAGS") old_make_jobs = delete("HOMEBREW_MAKE_JOBS") self["HOMEBREW_MAKE_JOBS"] = "1" if block begin yield ensure self["MAKEFLAGS"] = old_makeflags self["HOMEBREW_MAKE_JOBS"] = old_make_jobs end end old_makeflags end sig { returns(Integer) } def make_jobs self["MAKEFLAGS"] =~ /-\w*j(\d+)/ [Regexp.last_match(1).to_i, 1].max end sig { void } def permit_arch_flags append_to_cccfg "K" end sig { void } def runtime_cpu_detection append_to_cccfg "d" 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 set_debug_symbols append_to_cccfg "D" end sig { void } def refurbish_args append_to_cccfg "O" end # rubocop: disable Naming/MethodName # Fixes style error `Naming/MethodName: Use snake_case for method names.` sig { params(block: T.nilable(T.proc.void)).void } def O0(&block) if block with_env(HOMEBREW_OPTIMIZATION_LEVEL: "O0", &block) else self["HOMEBREW_OPTIMIZATION_LEVEL"] = "O0" end end sig { params(block: T.nilable(T.proc.void)).void } def O1(&block) if block with_env(HOMEBREW_OPTIMIZATION_LEVEL: "O1", &block) else self["HOMEBREW_OPTIMIZATION_LEVEL"] = "O1" end end sig { params(block: T.nilable(T.proc.void)).void } def O3(&block) if block with_env(HOMEBREW_OPTIMIZATION_LEVEL: "O3", &block) else self["HOMEBREW_OPTIMIZATION_LEVEL"] = "O3" end end # rubocop: enable Naming/MethodName end require "extend/os/extend/ENV/super"