require "hardware" require "os/mac/version" require "os/mac/xcode" require "os/mac/xquartz" require "os/mac/pathname" require "os/mac/sdk" require "os/mac/keg" module OS module Mac extend self ::MacOS = self # compatibility # This can be compared to numerics, strings, or symbols # using the standard Ruby Comparable methods. def version return @version ||= Version.new("0") unless OS.mac? @version ||= Version.new(full_version.to_s[/10\.\d+/]) end # This can be compared to numerics, strings, or symbols # using the standard Ruby Comparable methods. def full_version return @full_version ||= Version.new("0") unless OS.mac? @full_version ||= Version.new(`/usr/bin/sw_vers -productVersion`.chomp) end def prerelease? # TODO: bump version when new OS is released version >= "10.12" end def outdated_release? # TODO: bump version when new OS is released version < "10.9" end def cat version.to_sym end def locate(tool) # Don't call tools (cc, make, strip, etc.) directly! # Give the name of the binary you look for as a string to this method # in order to get the full path back as a Pathname. (@locate ||= {}).fetch(tool) do |key| @locate[key] = if File.executable?(path = "/usr/bin/#{tool}") Pathname.new path # Homebrew GCCs most frequently; much faster to check this before xcrun elsif (path = HOMEBREW_PREFIX/"bin/#{tool}").executable? path elsif OS.mac? path = Utils.popen_read("/usr/bin/xcrun", "-no-cache", "-find", tool).chomp Pathname.new(path) if File.executable?(path) end end end # Locates a (working) copy of install_name_tool, guaranteed to function # whether the user has developer tools installed or not. def install_name_tool if (path = HOMEBREW_PREFIX/"opt/cctools/bin/install_name_tool").executable? path else locate("install_name_tool") end end # Locates a (working) copy of otool, guaranteed to function whether the user # has developer tools installed or not. def otool if (path = HOMEBREW_PREFIX/"opt/cctools/bin/otool").executable? path else locate("otool") end end # Checks if the user has any developer tools installed, either via Xcode # or the CLT. Convenient for guarding against formula builds when building # is impossible. def has_apple_developer_tools? Xcode.installed? || CLT.installed? end def active_developer_dir return "" unless OS.mac? @active_developer_dir ||= Utils.popen_read("/usr/bin/xcode-select", "-print-path").strip end # If a specific SDK is requested # a) The requested SDK is returned, if it's installed. # b) If the requested SDK is not installed, the newest SDK (if any SDKs # are available) is returned. # c) If no SDKs are available, nil is returned. # If no specific SDK is requested # a) For Xcode >= 7, the latest SDK is returned even if the latest SDK is # named after a newer OS version than the running OS. The # MACOSX_DEPLOYMENT_TARGET must be set to the OS for which you're # actually building (usually the running OS version). # https://github.com/Homebrew/homebrew/pull/50355 # https://developer.apple.com/library/ios/documentation/DeveloperTools/Conceptual/WhatsNewXcode/Articles/Introduction.html#//apple_ref/doc/uid/TP40004626 # Section "About SDKs and Simulator" # b) For Xcode < 7, proceed as if the SDK for the running OS version had # specifically been requested according to the rules above. def sdk(v = nil) return nil unless OS.mac? @locator ||= SDKLocator.new begin sdk = if v.nil? Xcode.version.to_i >= 7 ? @locator.latest_sdk : @locator.sdk_for(version) else @locator.sdk_for v end rescue SDKLocator::NoSDKError sdk = @locator.latest_sdk end # Only return an SDK older than the OS version if it was specifically requested sdk if v || (!sdk.nil? && sdk.version >= version) end # Returns the path to an SDK or nil, following the rules set by #sdk. def sdk_path(v = nil) s = sdk(v) s.path unless s.nil? end def default_cc cc = locate "cc" cc.realpath.basename.to_s rescue nil end def default_compiler case default_cc when /^gcc-4.0/ then :gcc_4_0 when /^gcc/ then :gcc when /^llvm/ then :llvm when "clang" then :clang else # guess :( if Xcode.version >= "4.3" :clang elsif Xcode.version >= "4.2" :llvm else :gcc end end end def gcc_40_build_version @gcc_40_build_version ||= if (path = locate("gcc-4.0")) `#{path} --version 2>/dev/null`[/build (\d{4,})/, 1].to_i end end alias_method :gcc_4_0_build_version, :gcc_40_build_version def gcc_42_build_version @gcc_42_build_version ||= begin gcc = MacOS.locate("gcc-4.2") || HOMEBREW_PREFIX.join("opt/apple-gcc42/bin/gcc-4.2") if gcc.exist? && !gcc.realpath.basename.to_s.start_with?("llvm") `#{gcc} --version 2>/dev/null`[/build (\d{4,})/, 1].to_i end end end alias_method :gcc_build_version, :gcc_42_build_version def llvm_build_version @llvm_build_version ||= if (path = locate("llvm-gcc")) && !path.realpath.basename.to_s.start_with?("clang") `#{path} --version`[/LLVM build (\d{4,})/, 1].to_i end end def clang_version @clang_version ||= if (path = locate("clang")) `#{path} --version`[/(?:clang|LLVM) version (\d\.\d)/, 1] end end def clang_build_version @clang_build_version ||= if (path = locate("clang")) `#{path} --version`[/clang-(\d{2,})/, 1].to_i end end def non_apple_gcc_version(cc) (@non_apple_gcc_version ||= {}).fetch(cc) do path = HOMEBREW_PREFIX.join("opt", "gcc", "bin", cc) path = locate(cc) unless path.exist? path = locate(cc.delete("-.")) if OS.linux? && !path version = `#{path} --version`[/gcc(?:-\d(?:\.\d)?)? \(.+\) (\d\.\d\.\d)/, 1] if path @non_apple_gcc_version[cc] = version end end def clear_version_cache @gcc_40_build_version = @gcc_42_build_version = @llvm_build_version = nil @clang_version = @clang_build_version = nil @non_apple_gcc_version = {} end # See these issues for some history: # https://github.com/Homebrew/homebrew/issues/13 # https://github.com/Homebrew/homebrew/issues/41 # https://github.com/Homebrew/homebrew/issues/48 def macports_or_fink paths = [] # First look in the path because MacPorts is relocatable and Fink # may become relocatable in the future. %w[port fink].each do |ponk| path = which(ponk) paths << path unless path.nil? end # Look in the standard locations, because even if port or fink are # not in the path they can still break builds if the build scripts # have these paths baked in. %w[/sw/bin/fink /opt/local/bin/port].each do |ponk| path = Pathname.new(ponk) paths << path if path.exist? end # Finally, some users make their MacPorts or Fink directorie # read-only in order to try out Homebrew, but this doens't work as # some build scripts error out when trying to read from these now # unreadable paths. %w[/sw /opt/local].map { |p| Pathname.new(p) }.each do |path| paths << path if path.exist? && !path.readable? end paths.uniq end def prefer_64_bit? Hardware::CPU.is_64_bit? && (version > :leopard || OS.linux?) end def preferred_arch if prefer_64_bit? Hardware::CPU.arch_64_bit else Hardware::CPU.arch_32_bit end end STANDARD_COMPILERS = { "2.5" => { :gcc_40_build => 5370 }, "3.1.4" => { :gcc_40_build => 5493, :gcc_42_build => 5577 }, "3.2.6" => { :gcc_40_build => 5494, :gcc_42_build => 5666, :llvm_build => 2335, :clang => "1.7", :clang_build => 77 }, "4.0" => { :gcc_40_build => 5494, :gcc_42_build => 5666, :llvm_build => 2335, :clang => "2.0", :clang_build => 137 }, "4.0.1" => { :gcc_40_build => 5494, :gcc_42_build => 5666, :llvm_build => 2335, :clang => "2.0", :clang_build => 137 }, "4.0.2" => { :gcc_40_build => 5494, :gcc_42_build => 5666, :llvm_build => 2335, :clang => "2.0", :clang_build => 137 }, "4.2" => { :llvm_build => 2336, :clang => "3.0", :clang_build => 211 }, "4.3" => { :llvm_build => 2336, :clang => "3.1", :clang_build => 318 }, "4.3.1" => { :llvm_build => 2336, :clang => "3.1", :clang_build => 318 }, "4.3.2" => { :llvm_build => 2336, :clang => "3.1", :clang_build => 318 }, "4.3.3" => { :llvm_build => 2336, :clang => "3.1", :clang_build => 318 }, "4.4" => { :llvm_build => 2336, :clang => "4.0", :clang_build => 421 }, "4.4.1" => { :llvm_build => 2336, :clang => "4.0", :clang_build => 421 }, "4.5" => { :llvm_build => 2336, :clang => "4.1", :clang_build => 421 }, "4.5.1" => { :llvm_build => 2336, :clang => "4.1", :clang_build => 421 }, "4.5.2" => { :llvm_build => 2336, :clang => "4.1", :clang_build => 421 }, "4.6" => { :llvm_build => 2336, :clang => "4.2", :clang_build => 425 }, "4.6.1" => { :llvm_build => 2336, :clang => "4.2", :clang_build => 425 }, "4.6.2" => { :llvm_build => 2336, :clang => "4.2", :clang_build => 425 }, "4.6.3" => { :llvm_build => 2336, :clang => "4.2", :clang_build => 425 }, "5.0" => { :clang => "5.0", :clang_build => 500 }, "5.0.1" => { :clang => "5.0", :clang_build => 500 }, "5.0.2" => { :clang => "5.0", :clang_build => 500 }, "5.1" => { :clang => "5.1", :clang_build => 503 }, "5.1.1" => { :clang => "5.1", :clang_build => 503 }, "6.0" => { :clang => "6.0", :clang_build => 600 }, "6.0.1" => { :clang => "6.0", :clang_build => 600 }, "6.1" => { :clang => "6.0", :clang_build => 600 }, "6.1.1" => { :clang => "6.0", :clang_build => 600 }, "6.2" => { :clang => "6.0", :clang_build => 600 }, "6.3" => { :clang => "6.1", :clang_build => 602 }, "6.3.1" => { :clang => "6.1", :clang_build => 602 }, "6.3.2" => { :clang => "6.1", :clang_build => 602 }, "6.4" => { :clang => "6.1", :clang_build => 602 }, "7.0" => { :clang => "7.0", :clang_build => 700 }, "7.0.1" => { :clang => "7.0", :clang_build => 700 }, "7.1" => { :clang => "7.0", :clang_build => 700 }, "7.1.1" => { :clang => "7.0", :clang_build => 700 }, "7.2" => { :clang => "7.0", :clang_build => 700 }, "7.2.1" => { :clang => "7.0", :clang_build => 700 }, } def compilers_standard? return true unless OS.mac? STANDARD_COMPILERS.fetch(Xcode.version.to_s).all? do |method, build| send(:"#{method}_version") == build end rescue IndexError onoe <<-EOS.undent Homebrew doesn't know what compiler versions ship with your version of Xcode (#{Xcode.version}). Please `brew update` and if that doesn't help, file an issue with the output of `brew --config`: https://github.com/Homebrew/homebrew/issues Note that we only track stable, released versions of Xcode. Thanks! EOS end def app_with_bundle_id(*ids) path = mdfind(*ids).first Pathname.new(path) unless path.nil? || path.empty? end def mdfind(*ids) return [] unless OS.mac? (@mdfind ||= {}).fetch(ids) do @mdfind[ids] = Utils.popen_read("/usr/bin/mdfind", mdfind_query(*ids)).split("\n") end end def pkgutil_info(id) return "" unless OS.mac? (@pkginfo ||= {}).fetch(id) do |key| @pkginfo[key] = Utils.popen_read("/usr/sbin/pkgutil", "--pkg-info", key).strip end end def mdfind_query(*ids) ids.map! { |id| "kMDItemCFBundleIdentifier == #{id}" }.join(" || ") end end end