_osx_support.py 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502
  1. """Shared OS X support functions."""
  2. import os
  3. import re
  4. import sys
  5. __all__ = [
  6. 'compiler_fixup',
  7. 'customize_config_vars',
  8. 'customize_compiler',
  9. 'get_platform_osx',
  10. ]
  11. # configuration variables that may contain universal build flags,
  12. # like "-arch" or "-isdkroot", that may need customization for
  13. # the user environment
  14. _UNIVERSAL_CONFIG_VARS = ('CFLAGS', 'LDFLAGS', 'CPPFLAGS', 'BASECFLAGS',
  15. 'BLDSHARED', 'LDSHARED', 'CC', 'CXX',
  16. 'PY_CFLAGS', 'PY_LDFLAGS', 'PY_CPPFLAGS',
  17. 'PY_CORE_CFLAGS')
  18. # configuration variables that may contain compiler calls
  19. _COMPILER_CONFIG_VARS = ('BLDSHARED', 'LDSHARED', 'CC', 'CXX')
  20. # prefix added to original configuration variable names
  21. _INITPRE = '_OSX_SUPPORT_INITIAL_'
  22. def _find_executable(executable, path=None):
  23. """Tries to find 'executable' in the directories listed in 'path'.
  24. A string listing directories separated by 'os.pathsep'; defaults to
  25. os.environ['PATH']. Returns the complete filename or None if not found.
  26. """
  27. if path is None:
  28. path = os.environ['PATH']
  29. paths = path.split(os.pathsep)
  30. base, ext = os.path.splitext(executable)
  31. if (sys.platform == 'win32' or os.name == 'os2') and (ext != '.exe'):
  32. executable = executable + '.exe'
  33. if not os.path.isfile(executable):
  34. for p in paths:
  35. f = os.path.join(p, executable)
  36. if os.path.isfile(f):
  37. # the file exists, we have a shot at spawn working
  38. return f
  39. return None
  40. else:
  41. return executable
  42. def _read_output(commandstring):
  43. """Output from successful command execution or None"""
  44. # Similar to os.popen(commandstring, "r").read(),
  45. # but without actually using os.popen because that
  46. # function is not usable during python bootstrap.
  47. # tempfile is also not available then.
  48. import contextlib
  49. try:
  50. import tempfile
  51. fp = tempfile.NamedTemporaryFile()
  52. except ImportError:
  53. fp = open("/tmp/_osx_support.%s"%(
  54. os.getpid(),), "w+b")
  55. with contextlib.closing(fp) as fp:
  56. cmd = "%s 2>/dev/null >'%s'" % (commandstring, fp.name)
  57. return fp.read().strip() if not os.system(cmd) else None
  58. def _find_build_tool(toolname):
  59. """Find a build tool on current path or using xcrun"""
  60. return (_find_executable(toolname)
  61. or _read_output("/usr/bin/xcrun -find %s" % (toolname,))
  62. or ''
  63. )
  64. _SYSTEM_VERSION = None
  65. def _get_system_version():
  66. """Return the OS X system version as a string"""
  67. # Reading this plist is a documented way to get the system
  68. # version (see the documentation for the Gestalt Manager)
  69. # We avoid using platform.mac_ver to avoid possible bootstrap issues during
  70. # the build of Python itself (distutils is used to build standard library
  71. # extensions).
  72. global _SYSTEM_VERSION
  73. if _SYSTEM_VERSION is None:
  74. _SYSTEM_VERSION = ''
  75. try:
  76. f = open('/System/Library/CoreServices/SystemVersion.plist')
  77. except IOError:
  78. # We're on a plain darwin box, fall back to the default
  79. # behaviour.
  80. pass
  81. else:
  82. try:
  83. m = re.search(r'<key>ProductUserVisibleVersion</key>\s*'
  84. r'<string>(.*?)</string>', f.read())
  85. finally:
  86. f.close()
  87. if m is not None:
  88. _SYSTEM_VERSION = '.'.join(m.group(1).split('.')[:2])
  89. # else: fall back to the default behaviour
  90. return _SYSTEM_VERSION
  91. def _remove_original_values(_config_vars):
  92. """Remove original unmodified values for testing"""
  93. # This is needed for higher-level cross-platform tests of get_platform.
  94. for k in list(_config_vars):
  95. if k.startswith(_INITPRE):
  96. del _config_vars[k]
  97. def _save_modified_value(_config_vars, cv, newvalue):
  98. """Save modified and original unmodified value of configuration var"""
  99. oldvalue = _config_vars.get(cv, '')
  100. if (oldvalue != newvalue) and (_INITPRE + cv not in _config_vars):
  101. _config_vars[_INITPRE + cv] = oldvalue
  102. _config_vars[cv] = newvalue
  103. def _supports_universal_builds():
  104. """Returns True if universal builds are supported on this system"""
  105. # As an approximation, we assume that if we are running on 10.4 or above,
  106. # then we are running with an Xcode environment that supports universal
  107. # builds, in particular -isysroot and -arch arguments to the compiler. This
  108. # is in support of allowing 10.4 universal builds to run on 10.3.x systems.
  109. osx_version = _get_system_version()
  110. if osx_version:
  111. try:
  112. osx_version = tuple(int(i) for i in osx_version.split('.'))
  113. except ValueError:
  114. osx_version = ''
  115. return bool(osx_version >= (10, 4)) if osx_version else False
  116. def _find_appropriate_compiler(_config_vars):
  117. """Find appropriate C compiler for extension module builds"""
  118. # Issue #13590:
  119. # The OSX location for the compiler varies between OSX
  120. # (or rather Xcode) releases. With older releases (up-to 10.5)
  121. # the compiler is in /usr/bin, with newer releases the compiler
  122. # can only be found inside Xcode.app if the "Command Line Tools"
  123. # are not installed.
  124. #
  125. # Furthermore, the compiler that can be used varies between
  126. # Xcode releases. Up to Xcode 4 it was possible to use 'gcc-4.2'
  127. # as the compiler, after that 'clang' should be used because
  128. # gcc-4.2 is either not present, or a copy of 'llvm-gcc' that
  129. # miscompiles Python.
  130. # skip checks if the compiler was overridden with a CC env variable
  131. if 'CC' in os.environ:
  132. return _config_vars
  133. # The CC config var might contain additional arguments.
  134. # Ignore them while searching.
  135. cc = oldcc = _config_vars['CC'].split()[0]
  136. if not _find_executable(cc):
  137. # Compiler is not found on the shell search PATH.
  138. # Now search for clang, first on PATH (if the Command LIne
  139. # Tools have been installed in / or if the user has provided
  140. # another location via CC). If not found, try using xcrun
  141. # to find an uninstalled clang (within a selected Xcode).
  142. # NOTE: Cannot use subprocess here because of bootstrap
  143. # issues when building Python itself (and os.popen is
  144. # implemented on top of subprocess and is therefore not
  145. # usable as well)
  146. cc = _find_build_tool('clang')
  147. elif os.path.basename(cc).startswith('gcc'):
  148. # Compiler is GCC, check if it is LLVM-GCC
  149. data = _read_output("'%s' --version"
  150. % (cc.replace("'", "'\"'\"'"),))
  151. if data and 'llvm-gcc' in data:
  152. # Found LLVM-GCC, fall back to clang
  153. cc = _find_build_tool('clang')
  154. if not cc:
  155. raise SystemError(
  156. "Cannot locate working compiler")
  157. if cc != oldcc:
  158. # Found a replacement compiler.
  159. # Modify config vars using new compiler, if not already explicitly
  160. # overridden by an env variable, preserving additional arguments.
  161. for cv in _COMPILER_CONFIG_VARS:
  162. if cv in _config_vars and cv not in os.environ:
  163. cv_split = _config_vars[cv].split()
  164. cv_split[0] = cc if cv != 'CXX' else cc + '++'
  165. _save_modified_value(_config_vars, cv, ' '.join(cv_split))
  166. return _config_vars
  167. def _remove_universal_flags(_config_vars):
  168. """Remove all universal build arguments from config vars"""
  169. for cv in _UNIVERSAL_CONFIG_VARS:
  170. # Do not alter a config var explicitly overridden by env var
  171. if cv in _config_vars and cv not in os.environ:
  172. flags = _config_vars[cv]
  173. flags = re.sub('-arch\s+\w+\s', ' ', flags)
  174. flags = re.sub('-isysroot [^ \t]*', ' ', flags)
  175. _save_modified_value(_config_vars, cv, flags)
  176. return _config_vars
  177. def _remove_unsupported_archs(_config_vars):
  178. """Remove any unsupported archs from config vars"""
  179. # Different Xcode releases support different sets for '-arch'
  180. # flags. In particular, Xcode 4.x no longer supports the
  181. # PPC architectures.
  182. #
  183. # This code automatically removes '-arch ppc' and '-arch ppc64'
  184. # when these are not supported. That makes it possible to
  185. # build extensions on OSX 10.7 and later with the prebuilt
  186. # 32-bit installer on the python.org website.
  187. # skip checks if the compiler was overridden with a CC env variable
  188. if 'CC' in os.environ:
  189. return _config_vars
  190. if re.search('-arch\s+ppc', _config_vars['CFLAGS']) is not None:
  191. # NOTE: Cannot use subprocess here because of bootstrap
  192. # issues when building Python itself
  193. status = os.system(
  194. """echo 'int main{};' | """
  195. """'%s' -c -arch ppc -x c -o /dev/null /dev/null 2>/dev/null"""
  196. %(_config_vars['CC'].replace("'", "'\"'\"'"),))
  197. if status:
  198. # The compile failed for some reason. Because of differences
  199. # across Xcode and compiler versions, there is no reliable way
  200. # to be sure why it failed. Assume here it was due to lack of
  201. # PPC support and remove the related '-arch' flags from each
  202. # config variables not explicitly overridden by an environment
  203. # variable. If the error was for some other reason, we hope the
  204. # failure will show up again when trying to compile an extension
  205. # module.
  206. for cv in _UNIVERSAL_CONFIG_VARS:
  207. if cv in _config_vars and cv not in os.environ:
  208. flags = _config_vars[cv]
  209. flags = re.sub('-arch\s+ppc\w*\s', ' ', flags)
  210. _save_modified_value(_config_vars, cv, flags)
  211. return _config_vars
  212. def _override_all_archs(_config_vars):
  213. """Allow override of all archs with ARCHFLAGS env var"""
  214. # NOTE: This name was introduced by Apple in OSX 10.5 and
  215. # is used by several scripting languages distributed with
  216. # that OS release.
  217. if 'ARCHFLAGS' in os.environ:
  218. arch = os.environ['ARCHFLAGS']
  219. for cv in _UNIVERSAL_CONFIG_VARS:
  220. if cv in _config_vars and '-arch' in _config_vars[cv]:
  221. flags = _config_vars[cv]
  222. flags = re.sub('-arch\s+\w+\s', ' ', flags)
  223. flags = flags + ' ' + arch
  224. _save_modified_value(_config_vars, cv, flags)
  225. return _config_vars
  226. def _check_for_unavailable_sdk(_config_vars):
  227. """Remove references to any SDKs not available"""
  228. # If we're on OSX 10.5 or later and the user tries to
  229. # compile an extension using an SDK that is not present
  230. # on the current machine it is better to not use an SDK
  231. # than to fail. This is particularly important with
  232. # the standalone Command Line Tools alternative to a
  233. # full-blown Xcode install since the CLT packages do not
  234. # provide SDKs. If the SDK is not present, it is assumed
  235. # that the header files and dev libs have been installed
  236. # to /usr and /System/Library by either a standalone CLT
  237. # package or the CLT component within Xcode.
  238. cflags = _config_vars.get('CFLAGS', '')
  239. m = re.search(r'-isysroot\s+(\S+)', cflags)
  240. if m is not None:
  241. sdk = m.group(1)
  242. if not os.path.exists(sdk):
  243. for cv in _UNIVERSAL_CONFIG_VARS:
  244. # Do not alter a config var explicitly overridden by env var
  245. if cv in _config_vars and cv not in os.environ:
  246. flags = _config_vars[cv]
  247. flags = re.sub(r'-isysroot\s+\S+(?:\s|$)', ' ', flags)
  248. _save_modified_value(_config_vars, cv, flags)
  249. return _config_vars
  250. def compiler_fixup(compiler_so, cc_args):
  251. """
  252. This function will strip '-isysroot PATH' and '-arch ARCH' from the
  253. compile flags if the user has specified one them in extra_compile_flags.
  254. This is needed because '-arch ARCH' adds another architecture to the
  255. build, without a way to remove an architecture. Furthermore GCC will
  256. barf if multiple '-isysroot' arguments are present.
  257. """
  258. stripArch = stripSysroot = False
  259. compiler_so = list(compiler_so)
  260. if not _supports_universal_builds():
  261. # OSX before 10.4.0, these don't support -arch and -isysroot at
  262. # all.
  263. stripArch = stripSysroot = True
  264. else:
  265. stripArch = '-arch' in cc_args
  266. stripSysroot = '-isysroot' in cc_args
  267. if stripArch or 'ARCHFLAGS' in os.environ:
  268. while True:
  269. try:
  270. index = compiler_so.index('-arch')
  271. # Strip this argument and the next one:
  272. del compiler_so[index:index+2]
  273. except ValueError:
  274. break
  275. if 'ARCHFLAGS' in os.environ and not stripArch:
  276. # User specified different -arch flags in the environ,
  277. # see also distutils.sysconfig
  278. compiler_so = compiler_so + os.environ['ARCHFLAGS'].split()
  279. if stripSysroot:
  280. while True:
  281. try:
  282. index = compiler_so.index('-isysroot')
  283. # Strip this argument and the next one:
  284. del compiler_so[index:index+2]
  285. except ValueError:
  286. break
  287. # Check if the SDK that is used during compilation actually exists,
  288. # the universal build requires the usage of a universal SDK and not all
  289. # users have that installed by default.
  290. sysroot = None
  291. if '-isysroot' in cc_args:
  292. idx = cc_args.index('-isysroot')
  293. sysroot = cc_args[idx+1]
  294. elif '-isysroot' in compiler_so:
  295. idx = compiler_so.index('-isysroot')
  296. sysroot = compiler_so[idx+1]
  297. if sysroot and not os.path.isdir(sysroot):
  298. from distutils import log
  299. log.warn("Compiling with an SDK that doesn't seem to exist: %s",
  300. sysroot)
  301. log.warn("Please check your Xcode installation")
  302. return compiler_so
  303. def customize_config_vars(_config_vars):
  304. """Customize Python build configuration variables.
  305. Called internally from sysconfig with a mutable mapping
  306. containing name/value pairs parsed from the configured
  307. makefile used to build this interpreter. Returns
  308. the mapping updated as needed to reflect the environment
  309. in which the interpreter is running; in the case of
  310. a Python from a binary installer, the installed
  311. environment may be very different from the build
  312. environment, i.e. different OS levels, different
  313. built tools, different available CPU architectures.
  314. This customization is performed whenever
  315. distutils.sysconfig.get_config_vars() is first
  316. called. It may be used in environments where no
  317. compilers are present, i.e. when installing pure
  318. Python dists. Customization of compiler paths
  319. and detection of unavailable archs is deferred
  320. until the first extension module build is
  321. requested (in distutils.sysconfig.customize_compiler).
  322. Currently called from distutils.sysconfig
  323. """
  324. if not _supports_universal_builds():
  325. # On Mac OS X before 10.4, check if -arch and -isysroot
  326. # are in CFLAGS or LDFLAGS and remove them if they are.
  327. # This is needed when building extensions on a 10.3 system
  328. # using a universal build of python.
  329. _remove_universal_flags(_config_vars)
  330. # Allow user to override all archs with ARCHFLAGS env var
  331. _override_all_archs(_config_vars)
  332. # Remove references to sdks that are not found
  333. _check_for_unavailable_sdk(_config_vars)
  334. return _config_vars
  335. def customize_compiler(_config_vars):
  336. """Customize compiler path and configuration variables.
  337. This customization is performed when the first
  338. extension module build is requested
  339. in distutils.sysconfig.customize_compiler).
  340. """
  341. # Find a compiler to use for extension module builds
  342. _find_appropriate_compiler(_config_vars)
  343. # Remove ppc arch flags if not supported here
  344. _remove_unsupported_archs(_config_vars)
  345. # Allow user to override all archs with ARCHFLAGS env var
  346. _override_all_archs(_config_vars)
  347. return _config_vars
  348. def get_platform_osx(_config_vars, osname, release, machine):
  349. """Filter values for get_platform()"""
  350. # called from get_platform() in sysconfig and distutils.util
  351. #
  352. # For our purposes, we'll assume that the system version from
  353. # distutils' perspective is what MACOSX_DEPLOYMENT_TARGET is set
  354. # to. This makes the compatibility story a bit more sane because the
  355. # machine is going to compile and link as if it were
  356. # MACOSX_DEPLOYMENT_TARGET.
  357. macver = _config_vars.get('MACOSX_DEPLOYMENT_TARGET', '')
  358. macrelease = _get_system_version() or macver
  359. macver = macver or macrelease
  360. if macver:
  361. release = macver
  362. osname = "macosx"
  363. # Use the original CFLAGS value, if available, so that we
  364. # return the same machine type for the platform string.
  365. # Otherwise, distutils may consider this a cross-compiling
  366. # case and disallow installs.
  367. cflags = _config_vars.get(_INITPRE+'CFLAGS',
  368. _config_vars.get('CFLAGS', ''))
  369. if macrelease:
  370. try:
  371. macrelease = tuple(int(i) for i in macrelease.split('.')[0:2])
  372. except ValueError:
  373. macrelease = (10, 0)
  374. else:
  375. # assume no universal support
  376. macrelease = (10, 0)
  377. if (macrelease >= (10, 4)) and '-arch' in cflags.strip():
  378. # The universal build will build fat binaries, but not on
  379. # systems before 10.4
  380. machine = 'fat'
  381. archs = re.findall('-arch\s+(\S+)', cflags)
  382. archs = tuple(sorted(set(archs)))
  383. if len(archs) == 1:
  384. machine = archs[0]
  385. elif archs == ('i386', 'ppc'):
  386. machine = 'fat'
  387. elif archs == ('i386', 'x86_64'):
  388. machine = 'intel'
  389. elif archs == ('i386', 'ppc', 'x86_64'):
  390. machine = 'fat3'
  391. elif archs == ('ppc64', 'x86_64'):
  392. machine = 'fat64'
  393. elif archs == ('i386', 'ppc', 'ppc64', 'x86_64'):
  394. machine = 'universal'
  395. else:
  396. raise ValueError(
  397. "Don't know machine value for archs=%r" % (archs,))
  398. elif machine == 'i386':
  399. # On OSX the machine type returned by uname is always the
  400. # 32-bit variant, even if the executable architecture is
  401. # the 64-bit variant
  402. if sys.maxint >= 2**32:
  403. machine = 'x86_64'
  404. elif machine in ('PowerPC', 'Power_Macintosh'):
  405. # Pick a sane name for the PPC architecture.
  406. # See 'i386' case
  407. if sys.maxint >= 2**32:
  408. machine = 'ppc64'
  409. else:
  410. machine = 'ppc'
  411. return (osname, release, machine)