2014-05-03 10:01:30 -05:00
|
|
|
# This script is loaded by formula_installer as a separate instance.
|
2011-08-23 23:30:52 +01:00
|
|
|
# Thrown exceptions are propogated back to the parent process over a pipe
|
2011-03-15 22:02:14 -07:00
|
|
|
|
2012-08-22 15:50:27 -04:00
|
|
|
STD_TRAP = trap("INT") { exit! 130 } # no backtrace thanks
|
2009-09-17 21:10:39 +01:00
|
|
|
|
2014-07-29 16:06:06 -05:00
|
|
|
at_exit { main }
|
2012-08-22 15:50:27 -04:00
|
|
|
|
|
|
|
require 'global'
|
2013-07-27 02:07:29 -07:00
|
|
|
require 'cxxstdlib'
|
2012-02-21 01:14:02 -06:00
|
|
|
require 'debrew' if ARGV.debug?
|
2012-08-22 15:50:27 -04:00
|
|
|
|
|
|
|
def main
|
|
|
|
# The main Homebrew process expects to eventually see EOF on the error
|
|
|
|
# pipe in FormulaInstaller#build. However, if any child process fails to
|
|
|
|
# terminate (i.e, fails to close the descriptor), this won't happen, and
|
|
|
|
# the installer will hang. Set close-on-exec to prevent this.
|
|
|
|
# Whether it is *wise* to launch daemons from formulae is a separate
|
|
|
|
# question altogether.
|
|
|
|
if ENV['HOMEBREW_ERROR_PIPE']
|
|
|
|
require 'fcntl'
|
|
|
|
error_pipe = IO.new(ENV['HOMEBREW_ERROR_PIPE'].to_i, 'w')
|
|
|
|
error_pipe.fcntl(Fcntl::F_SETFD, Fcntl::FD_CLOEXEC)
|
|
|
|
end
|
|
|
|
|
|
|
|
raise $! if $! # an exception was already thrown when parsing the formula
|
|
|
|
|
|
|
|
trap("INT", STD_TRAP) # restore default CTRL-C handler
|
|
|
|
|
|
|
|
require 'keg'
|
2013-08-19 12:32:57 -05:00
|
|
|
require 'extend/ENV'
|
2012-08-22 15:50:27 -04:00
|
|
|
|
|
|
|
# Force any future invocations of sudo to require the user's password to be
|
|
|
|
# re-entered. This is in-case any build script call sudo. Certainly this is
|
|
|
|
# can be inconvenient for the user. But we need to be safe.
|
2013-03-02 23:28:27 -06:00
|
|
|
system "/usr/bin/sudo", "-k"
|
2011-08-23 23:30:52 +01:00
|
|
|
|
2014-06-19 21:35:47 -05:00
|
|
|
formula = Formulary.factory($0, ARGV.spec)
|
|
|
|
Build.new(formula).install
|
2012-08-22 15:50:27 -04:00
|
|
|
rescue Exception => e
|
|
|
|
unless error_pipe.nil?
|
2012-02-21 01:14:02 -06:00
|
|
|
e.continuation = nil if ARGV.debug?
|
2012-08-22 15:50:27 -04:00
|
|
|
Marshal.dump(e, error_pipe)
|
|
|
|
error_pipe.close
|
|
|
|
exit! 1
|
|
|
|
else
|
|
|
|
onoe e
|
|
|
|
puts e.backtrace
|
|
|
|
exit! 2
|
2009-10-26 18:13:38 +00:00
|
|
|
end
|
|
|
|
end
|
2009-09-04 15:28:18 +01:00
|
|
|
|
2013-05-25 15:26:55 -05:00
|
|
|
class Build
|
2013-05-25 15:26:55 -05:00
|
|
|
attr_reader :f, :deps, :reqs
|
2012-09-13 09:06:37 -04:00
|
|
|
|
2013-05-25 15:26:55 -05:00
|
|
|
def initialize(f)
|
|
|
|
@f = f
|
2014-03-03 23:47:00 -06:00
|
|
|
|
|
|
|
if ARGV.ignore_deps?
|
|
|
|
@deps = []
|
|
|
|
@reqs = []
|
|
|
|
else
|
|
|
|
@deps = expand_deps
|
|
|
|
@reqs = expand_reqs
|
|
|
|
end
|
2013-05-25 15:26:55 -05:00
|
|
|
end
|
2012-08-28 13:46:29 -04:00
|
|
|
|
2013-05-25 15:26:55 -05:00
|
|
|
def post_superenv_hacks
|
|
|
|
# Only allow Homebrew-approved directories into the PATH, unless
|
|
|
|
# a formula opts-in to allowing the user's path.
|
2013-05-25 15:26:55 -05:00
|
|
|
if f.env.userpaths? || reqs.any? { |rq| rq.env.userpaths? }
|
2013-05-25 15:26:55 -05:00
|
|
|
ENV.userpaths!
|
2013-02-09 13:55:15 -06:00
|
|
|
end
|
2013-05-25 15:26:55 -05:00
|
|
|
end
|
2012-06-08 04:09:29 +02:00
|
|
|
|
2013-05-25 15:26:55 -05:00
|
|
|
def pre_superenv_hacks
|
|
|
|
# Allow a formula to opt-in to the std environment.
|
2014-03-13 16:36:51 -05:00
|
|
|
if (f.env.std? || deps.any? { |d| d.name == "scons" }) && ARGV.env != "super"
|
|
|
|
ARGV.unshift "--env=std"
|
|
|
|
end
|
2013-05-25 15:26:55 -05:00
|
|
|
end
|
2012-02-27 04:06:13 +00:00
|
|
|
|
2013-06-03 15:08:47 -05:00
|
|
|
def expand_reqs
|
|
|
|
f.recursive_requirements do |dependent, req|
|
2013-12-09 14:36:10 -06:00
|
|
|
if (req.optional? || req.recommended?) && dependent.build.without?(req)
|
2013-06-03 15:08:47 -05:00
|
|
|
Requirement.prune
|
|
|
|
elsif req.build? && dependent != f
|
|
|
|
Requirement.prune
|
|
|
|
elsif req.satisfied? && req.default_formula? && (dep = req.to_dependency).installed?
|
2013-12-02 12:44:28 -06:00
|
|
|
deps << dep
|
2013-06-03 15:08:47 -05:00
|
|
|
Requirement.prune
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2013-05-25 15:26:55 -05:00
|
|
|
def expand_deps
|
|
|
|
f.recursive_dependencies do |dependent, dep|
|
2013-12-09 14:36:10 -06:00
|
|
|
if (dep.optional? || dep.recommended?) && dependent.build.without?(dep)
|
2013-07-22 21:36:11 -05:00
|
|
|
Dependency.prune
|
|
|
|
elsif dep.build? && dependent != f
|
|
|
|
Dependency.prune
|
2013-11-13 10:38:14 -06:00
|
|
|
elsif dep.build?
|
|
|
|
Dependency.keep_but_prune_recursive_deps
|
2013-05-25 15:26:55 -05:00
|
|
|
end
|
2013-06-03 15:08:47 -05:00
|
|
|
end
|
2009-09-21 20:22:09 +01:00
|
|
|
end
|
|
|
|
|
2013-05-25 15:26:55 -05:00
|
|
|
def install
|
2013-06-03 15:08:47 -05:00
|
|
|
keg_only_deps = deps.map(&:to_formula).select(&:keg_only?)
|
2013-05-25 15:26:55 -05:00
|
|
|
|
2013-06-10 17:24:44 -05:00
|
|
|
deps.map(&:to_formula).each do |dep|
|
2014-06-09 14:54:49 -05:00
|
|
|
opt = HOMEBREW_PREFIX.join("opt", dep.name)
|
2014-03-03 23:47:00 -06:00
|
|
|
fixopt(dep) unless opt.directory?
|
2013-01-19 20:45:57 -06:00
|
|
|
end
|
superenv: build-environments that just work
1. A minimal build environment, we don't set CFLAGS, CPPFLAGS, LDFLAGS, etc. the rationale being, the less that is set, the less variables we are introducing that can break builds.
2. A set of scripts that replace cc, ld, etc. and inject the -I, -L, etc. flags we need into the args passed to the build-tools.
Because we now have complete control over compiler instantiations we do a variety of clean-up tasks, like removing bad flags, enforcing universal builds and ensuring makefiles don't try to change the order of library and include paths from ones that work to ones that don't.
The previous ENV-system is still available when --env=std is specified.
superenv applies to Xcode >= 4.3 only currently.
2012-08-11 12:30:51 -04:00
|
|
|
|
2014-07-28 23:05:44 -05:00
|
|
|
pre_superenv_hacks
|
|
|
|
ENV.activate_extensions!
|
|
|
|
|
2013-05-25 15:26:55 -05:00
|
|
|
if superenv?
|
2014-03-03 23:51:30 -06:00
|
|
|
ENV.keg_only_deps = keg_only_deps.map(&:name)
|
|
|
|
ENV.deps = deps.map { |d| d.to_formula.name }
|
2013-05-25 15:26:55 -05:00
|
|
|
ENV.x11 = reqs.any? { |rq| rq.kind_of?(X11Dependency) }
|
2013-09-10 23:08:17 -07:00
|
|
|
ENV.setup_build_environment(f)
|
2013-05-25 15:26:55 -05:00
|
|
|
post_superenv_hacks
|
2013-05-25 15:26:55 -05:00
|
|
|
reqs.each(&:modify_build_environment)
|
2013-06-03 15:08:47 -05:00
|
|
|
deps.each(&:modify_build_environment)
|
2013-05-25 15:26:55 -05:00
|
|
|
else
|
2013-09-10 23:08:17 -07:00
|
|
|
ENV.setup_build_environment(f)
|
2013-05-25 15:26:55 -05:00
|
|
|
reqs.each(&:modify_build_environment)
|
2013-06-03 15:08:47 -05:00
|
|
|
deps.each(&:modify_build_environment)
|
2013-05-25 15:26:55 -05:00
|
|
|
|
|
|
|
keg_only_deps.each do |dep|
|
2014-06-09 14:55:01 -05:00
|
|
|
ENV.prepend_path "PATH", dep.opt_bin.to_s
|
|
|
|
ENV.prepend_path "PKG_CONFIG_PATH", "#{dep.opt_lib}/pkgconfig"
|
|
|
|
ENV.prepend_path "PKG_CONFIG_PATH", "#{dep.opt_share}/pkgconfig"
|
|
|
|
ENV.prepend_path "ACLOCAL_PATH", "#{dep.opt_share}/aclocal"
|
|
|
|
ENV.prepend_path "CMAKE_PREFIX_PATH", dep.opt_prefix.to_s
|
|
|
|
ENV.prepend "LDFLAGS", "-L#{dep.opt_lib}" if dep.opt_lib.directory?
|
|
|
|
ENV.prepend "CPPFLAGS", "-I#{dep.opt_include}" if dep.opt_include.directory?
|
2013-05-25 15:26:55 -05:00
|
|
|
end
|
2013-05-20 19:35:07 -05:00
|
|
|
end
|
2012-03-18 13:58:13 -05:00
|
|
|
|
2013-05-25 15:26:55 -05:00
|
|
|
f.brew do
|
2011-08-23 23:30:52 +01:00
|
|
|
if ARGV.flag? '--git'
|
2014-06-20 18:36:18 -05:00
|
|
|
system "git", "init"
|
|
|
|
system "git", "add", "-A"
|
2009-09-04 15:28:18 +01:00
|
|
|
end
|
2013-05-25 15:26:55 -05:00
|
|
|
if ARGV.interactive?
|
|
|
|
ohai "Entering interactive mode"
|
|
|
|
puts "Type `exit' to return and finalize the installation"
|
|
|
|
puts "Install to this prefix: #{f.prefix}"
|
|
|
|
|
|
|
|
if ARGV.flag? '--git'
|
|
|
|
puts "This directory is now a git repo. Make your changes and then use:"
|
|
|
|
puts " git diff | pbcopy"
|
|
|
|
puts "to copy the diff to the clipboard."
|
|
|
|
end
|
2009-09-04 15:28:18 +01:00
|
|
|
|
2013-05-25 15:26:55 -05:00
|
|
|
interactive_shell f
|
|
|
|
else
|
|
|
|
f.prefix.mkpath
|
|
|
|
|
2014-03-13 10:05:55 -05:00
|
|
|
f.resources.each { |r| r.extend(ResourceDebugger) } if ARGV.debug?
|
|
|
|
|
2013-05-25 15:26:55 -05:00
|
|
|
begin
|
|
|
|
f.install
|
2014-08-09 17:47:10 -05:00
|
|
|
stdlibs = detect_stdlibs
|
2014-07-30 20:27:46 -05:00
|
|
|
Tab.create(f, ENV.compiler, stdlibs.first, f.build).write
|
2013-05-25 15:26:55 -05:00
|
|
|
rescue Exception => e
|
|
|
|
if ARGV.debug?
|
|
|
|
debrew e, f
|
|
|
|
else
|
|
|
|
raise e
|
|
|
|
end
|
2012-02-21 01:14:02 -06:00
|
|
|
end
|
2012-09-08 12:18:52 -07:00
|
|
|
|
2013-05-25 15:26:55 -05:00
|
|
|
# Find and link metafiles
|
|
|
|
f.prefix.install_metafiles Pathname.pwd
|
|
|
|
end
|
2010-08-21 12:18:17 -07:00
|
|
|
end
|
2011-07-04 09:31:29 +01:00
|
|
|
end
|
2012-08-27 12:02:57 -04:00
|
|
|
|
2014-08-09 17:47:10 -05:00
|
|
|
def detect_stdlibs
|
|
|
|
keg = Keg.new(f.prefix)
|
|
|
|
# This first test includes executables because we still
|
|
|
|
# want to record the stdlib for something that installs no
|
|
|
|
# dylibs.
|
|
|
|
stdlibs = keg.detect_cxx_stdlibs
|
|
|
|
# This currently only tracks a single C++ stdlib per dep,
|
|
|
|
# though it's possible for different libs/executables in
|
|
|
|
# a given formula to link to different ones.
|
|
|
|
stdlib_in_use = CxxStdlib.create(stdlibs.first, ENV.compiler)
|
|
|
|
begin
|
|
|
|
stdlib_in_use.check_dependencies(f, deps)
|
|
|
|
rescue IncompatibleCxxStdlibs => e
|
|
|
|
opoo e.message
|
|
|
|
end
|
|
|
|
|
|
|
|
# This second check is recorded for checking dependencies,
|
|
|
|
# so executable are irrelevant at this point. If a piece
|
|
|
|
# of software installs an executable that links against libstdc++
|
|
|
|
# and dylibs against libc++, libc++-only dependencies can safely
|
|
|
|
# link against it.
|
|
|
|
stdlibs = keg.detect_cxx_stdlibs :skip_executables => true
|
|
|
|
end
|
|
|
|
|
2014-07-28 21:22:33 -05:00
|
|
|
def fixopt f
|
|
|
|
path = if f.linked_keg.directory? and f.linked_keg.symlink?
|
|
|
|
f.linked_keg.resolved_path
|
|
|
|
elsif f.prefix.directory?
|
|
|
|
f.prefix
|
|
|
|
elsif (kids = f.rack.children).size == 1 and kids.first.directory?
|
|
|
|
kids.first
|
|
|
|
else
|
|
|
|
raise
|
|
|
|
end
|
|
|
|
Keg.new(path).optlink
|
|
|
|
rescue StandardError
|
|
|
|
raise "#{f.opt_prefix} not present or broken\nPlease reinstall #{f}. Sorry :("
|
2012-08-27 12:02:57 -04:00
|
|
|
end
|
|
|
|
end
|