142 lines
3.4 KiB
Ruby
Raw Normal View History

# frozen_string_literal: true
require "os/mac/version"
module OS
module Mac
2020-08-25 00:33:34 +02:00
# Class representing a macOS SDK.
#
# @api private
class SDK
2020-04-07 16:43:32 +01:00
attr_reader :version, :path, :source
2020-04-07 16:43:32 +01:00
def initialize(version, path, source)
2020-10-05 14:46:43 +02:00
@version = version
@path = Pathname.new(path)
2020-04-07 16:43:32 +01:00
@source = source
end
end
2020-08-25 00:33:34 +02:00
# Base class for SDK locators.
#
# @api private
2018-06-12 14:55:31 -07:00
class BaseSDKLocator
class NoSDKError < StandardError; end
def sdk_for(v)
path = sdk_paths[v]
raise NoSDKError if path.nil?
2020-04-07 16:43:32 +01:00
SDK.new v, path, source
end
def latest_sdk
return if sdk_paths.empty?
v, path = sdk_paths.max { |(v1, _), (v2, _)| v1 <=> v2 }
2020-04-07 16:43:32 +01:00
SDK.new v, path, source
end
2020-07-01 16:02:29 +01:00
def all_sdks
sdk_paths.map { |v, p| SDK.new v, p, source }
end
def sdk_if_applicable(v = nil)
2018-07-27 15:44:22 -07:00
sdk = begin
if v.nil?
sdk_for OS::Mac.version
else
sdk_for v
end
2020-08-25 00:33:34 +02:00
rescue NoSDKError
2018-07-27 15:44:22 -07:00
latest_sdk
end
# Only return an SDK older than the OS version if it was specifically requested
2018-07-27 15:44:22 -07:00
return unless v || (!sdk.nil? && sdk.version >= OS::Mac.version)
2018-09-17 02:45:00 +02:00
2018-07-27 15:44:22 -07:00
sdk
end
2020-04-07 16:43:32 +01:00
def source
nil
end
2020-07-01 16:02:29 +01:00
private
2018-06-12 14:55:31 -07:00
def sdk_prefix
""
end
def sdk_paths
@sdk_paths ||= begin
# Bail out if there is no SDK prefix at all
if !File.directory? sdk_prefix
{}
else
paths = {}
Dir[File.join(sdk_prefix, "MacOSX*.sdk")].each do |sdk_path|
2016-07-13 16:19:51 +08:00
version = sdk_path[/MacOSX(\d+\.\d+)u?\.sdk$/, 1]
2020-10-05 14:46:43 +02:00
paths[OS::Mac::Version.new(version)] = sdk_path unless version.nil?
end
paths
end
end
end
end
2020-08-25 00:33:34 +02:00
private_constant :BaseSDKLocator
2018-06-12 14:55:31 -07:00
2020-08-25 00:33:34 +02:00
# Helper class for locating the Xcode SDK.
#
# @api private
2018-06-12 14:55:31 -07:00
class XcodeSDKLocator < BaseSDKLocator
2020-04-07 16:43:32 +01:00
def source
:xcode
end
2020-07-01 16:02:29 +01:00
private
2018-06-12 14:55:31 -07:00
def sdk_prefix
@sdk_prefix ||= begin
# Xcode.prefix is pretty smart, so let's look inside to find the sdk
sdk_prefix = "#{Xcode.prefix}/Platforms/MacOSX.platform/Developer/SDKs"
# Finally query Xcode itself (this is slow, so check it last)
2018-07-27 15:44:22 -07:00
sdk_platform_path = Utils.popen_read(DevelopmentTools.locate("xcrun"), "--show-sdk-platform-path").chomp
sdk_prefix = File.join(sdk_platform_path, "Developer", "SDKs") unless File.directory? sdk_prefix
2018-06-12 14:55:31 -07:00
sdk_prefix
end
end
end
2020-08-25 00:33:34 +02:00
# Helper class for locating the macOS Command Line Tools SDK.
#
# @api private
2018-06-12 14:55:31 -07:00
class CLTSDKLocator < BaseSDKLocator
2020-04-07 16:43:32 +01:00
def source
:clt
end
2020-07-01 16:02:29 +01:00
private
2018-06-12 14:55:31 -07:00
# While CLT SDKs existed prior to Xcode 10, those packages also
# installed a traditional Unix-style header layout and we prefer
# using that
# As of Xcode 10, the Unix-style headers are installed via a
# separate package, so we can't rely on their being present.
# This will only look up SDKs on Xcode 10 or newer, and still
# return nil SDKs for Xcode 9 and older.
def sdk_prefix
@sdk_prefix ||= begin
if !CLT.provides_sdk?
2018-06-12 14:55:31 -07:00
""
else
"#{CLT::PKG_PATH}/SDKs"
end
end
end
end
end
end