Make macOS installed libraries more relocatable (#26608)
* relocate: call install_name_tool less * zstd: fix race condition Multiple times on my mac, trying to install in parallel led to failures from multiple tasks trying to simultaneously create `$PREFIX/lib`. * PackageMeta: simplify callback flush * Relocate: use spack.platforms instead of platform * Relocate: code improvements * fix zstd * Automatically fix rpaths for packages on macOS * Only change library IDs when the path is already in the rpath This restores the hardcoded library path for GCC. * Delete nonexistent rpaths and add more testing * Relocate: Allow @executable_path and @loader_path
This commit is contained in:
		| @@ -645,3 +645,6 @@ def remove_libtool_archives(self): | |||||||
|             fs.mkdirp(os.path.dirname(self._removed_la_files_log)) |             fs.mkdirp(os.path.dirname(self._removed_la_files_log)) | ||||||
|             with open(self._removed_la_files_log, mode='w') as f: |             with open(self._removed_la_files_log, mode='w') as f: | ||||||
|                 f.write('\n'.join(libtool_files)) |                 f.write('\n'.join(libtool_files)) | ||||||
|  | 
 | ||||||
|  |     # On macOS, force rpaths for shared library IDs and remove duplicate rpaths | ||||||
|  |     run_after('install')(PackageBase.apply_macos_rpath_fixups) | ||||||
|   | |||||||
| @@ -110,3 +110,6 @@ def installcheck(self): | |||||||
| 
 | 
 | ||||||
|     # Check that self.prefix is there after installation |     # Check that self.prefix is there after installation | ||||||
|     run_after('install')(PackageBase.sanity_check_prefix) |     run_after('install')(PackageBase.sanity_check_prefix) | ||||||
|  | 
 | ||||||
|  |     # On macOS, force rpaths for shared library IDs and remove duplicate rpaths | ||||||
|  |     run_after('install')(PackageBase.apply_macos_rpath_fixups) | ||||||
|   | |||||||
| @@ -19,7 +19,6 @@ | |||||||
| 
 | 
 | ||||||
| import spack.config | import spack.config | ||||||
| import spack.projections | import spack.projections | ||||||
| import spack.relocate |  | ||||||
| import spack.schema.projections | import spack.schema.projections | ||||||
| import spack.spec | import spack.spec | ||||||
| import spack.store | import spack.store | ||||||
| @@ -74,6 +73,9 @@ def view_copy(src, dst, view, spec=None): | |||||||
|         # TODO: Not sure which one to use... |         # TODO: Not sure which one to use... | ||||||
|         import spack.hooks.sbang as sbang |         import spack.hooks.sbang as sbang | ||||||
| 
 | 
 | ||||||
|  |         # Break a package include cycle | ||||||
|  |         import spack.relocate | ||||||
|  | 
 | ||||||
|         orig_sbang = '#!/bin/bash {0}/bin/sbang'.format(spack.paths.spack_root) |         orig_sbang = '#!/bin/bash {0}/bin/sbang'.format(spack.paths.spack_root) | ||||||
|         new_sbang = sbang.sbang_shebang_line() |         new_sbang = sbang.sbang_shebang_line() | ||||||
| 
 | 
 | ||||||
|   | |||||||
| @@ -286,34 +286,27 @@ def __new__(cls, name, bases, attr_dict): | |||||||
| 
 | 
 | ||||||
|         def _flush_callbacks(check_name): |         def _flush_callbacks(check_name): | ||||||
|             # Name of the attribute I am going to check it exists |             # Name of the attribute I am going to check it exists | ||||||
|             attr_name = PackageMeta.phase_fmt.format(check_name) |             check_attr = PackageMeta.phase_fmt.format(check_name) | ||||||
|             checks = getattr(cls, attr_name) |             checks = getattr(cls, check_attr) | ||||||
|             if checks: |             if checks: | ||||||
|                 for phase_name, funcs in checks.items(): |                 for phase_name, funcs in checks.items(): | ||||||
|  |                     phase_attr = PackageMeta.phase_fmt.format(phase_name) | ||||||
|                     try: |                     try: | ||||||
|                         # Search for the phase in the attribute dictionary |                         # Search for the phase in the attribute dictionary | ||||||
|                         phase = attr_dict[ |                         phase = attr_dict[phase_attr] | ||||||
|                             PackageMeta.phase_fmt.format(phase_name)] |  | ||||||
|                     except KeyError: |                     except KeyError: | ||||||
|                         # If it is not there it's in the bases |                         # If it is not there it's in the bases | ||||||
|                         # and we added a check. We need to copy |                         # and we added a check. We need to copy | ||||||
|                         # and extend |                         # and extend | ||||||
|                         for base in bases: |                         for base in bases: | ||||||
|                             phase = getattr( |                             phase = getattr(base, phase_attr, None) | ||||||
|                                 base, |  | ||||||
|                                 PackageMeta.phase_fmt.format(phase_name), |  | ||||||
|                                 None |  | ||||||
|                             ) |  | ||||||
|                             if phase is not None: |                             if phase is not None: | ||||||
|                                 break |                                 break | ||||||
| 
 | 
 | ||||||
|                         attr_dict[PackageMeta.phase_fmt.format( |                         phase = attr_dict[phase_attr] = phase.copy() | ||||||
|                             phase_name)] = phase.copy() |  | ||||||
|                         phase = attr_dict[ |  | ||||||
|                             PackageMeta.phase_fmt.format(phase_name)] |  | ||||||
|                     getattr(phase, check_name).extend(funcs) |                     getattr(phase, check_name).extend(funcs) | ||||||
|                 # Clear the attribute for the next class |                 # Clear the attribute for the next class | ||||||
|                 setattr(cls, attr_name, {}) |                 setattr(cls, check_attr, {}) | ||||||
| 
 | 
 | ||||||
|         _flush_callbacks('run_before') |         _flush_callbacks('run_before') | ||||||
|         _flush_callbacks('run_after') |         _flush_callbacks('run_after') | ||||||
| @@ -1962,6 +1955,25 @@ def check_paths(path_list, filetype, predicate): | |||||||
|             raise InstallError( |             raise InstallError( | ||||||
|                 "Install failed for %s.  Nothing was installed!" % self.name) |                 "Install failed for %s.  Nothing was installed!" % self.name) | ||||||
| 
 | 
 | ||||||
|  |     def apply_macos_rpath_fixups(self): | ||||||
|  |         """On Darwin, make installed libraries more easily relocatable. | ||||||
|  | 
 | ||||||
|  |         Some build systems (handrolled, autotools, makefiles) can set their own | ||||||
|  |         rpaths that are duplicated by spack's compiler wrapper. Additionally, | ||||||
|  |         many simpler build systems do not link using ``-install_name | ||||||
|  |         @rpath/foo.dylib``, which propagates the library's hardcoded | ||||||
|  |         absolute path into downstream dependencies. This fixup interrogates, | ||||||
|  |         and postprocesses if necessary, all libraries installed by the code. | ||||||
|  | 
 | ||||||
|  |         It should be added as a @run_after to packaging systems (or individual | ||||||
|  |         packages) that do not install relocatable libraries by default. | ||||||
|  |         """ | ||||||
|  |         if 'platform=darwin' not in self.spec: | ||||||
|  |             return | ||||||
|  | 
 | ||||||
|  |         from spack.relocate import fixup_macos_rpaths | ||||||
|  |         fixup_macos_rpaths(self.spec) | ||||||
|  | 
 | ||||||
|     @property |     @property | ||||||
|     def build_log_path(self): |     def build_log_path(self): | ||||||
|         """ |         """ | ||||||
| @@ -2702,6 +2714,8 @@ class Package(PackageBase): | |||||||
|     # This will be used as a registration decorator in user |     # This will be used as a registration decorator in user | ||||||
|     # packages, if need be |     # packages, if need be | ||||||
|     run_after('install')(PackageBase.sanity_check_prefix) |     run_after('install')(PackageBase.sanity_check_prefix) | ||||||
|  |     # On macOS, force rpaths for shared library IDs and remove duplicate rpaths | ||||||
|  |     run_after('install')(PackageBase.apply_macos_rpath_fixups) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def install_dependency_symlinks(pkg, spec, prefix): | def install_dependency_symlinks(pkg, spec, prefix): | ||||||
|   | |||||||
| @@ -4,9 +4,9 @@ | |||||||
| # SPDX-License-Identifier: (Apache-2.0 OR MIT) | # SPDX-License-Identifier: (Apache-2.0 OR MIT) | ||||||
| import multiprocessing.pool | import multiprocessing.pool | ||||||
| import os | import os | ||||||
| import platform |  | ||||||
| import re | import re | ||||||
| import shutil | import shutil | ||||||
|  | from collections import defaultdict | ||||||
| 
 | 
 | ||||||
| import macholib.mach_o | import macholib.mach_o | ||||||
| import macholib.MachO | import macholib.MachO | ||||||
| @@ -20,6 +20,8 @@ | |||||||
| import spack.spec | import spack.spec | ||||||
| import spack.util.executable as executable | import spack.util.executable as executable | ||||||
| 
 | 
 | ||||||
|  | is_macos = (str(spack.platforms.real_host()) == 'darwin') | ||||||
|  | 
 | ||||||
| 
 | 
 | ||||||
| class InstallRootStringError(spack.error.SpackError): | class InstallRootStringError(spack.error.SpackError): | ||||||
|     def __init__(self, file_path, root_path): |     def __init__(self, file_path, root_path): | ||||||
| @@ -82,7 +84,7 @@ def _patchelf(): | |||||||
|     Return None on Darwin or if patchelf cannot be found. |     Return None on Darwin or if patchelf cannot be found. | ||||||
|     """ |     """ | ||||||
|     # Check if patchelf is already in the PATH |     # Check if patchelf is already in the PATH | ||||||
|     patchelf = spack.util.executable.which('patchelf') |     patchelf = executable.which('patchelf') | ||||||
|     if patchelf is not None: |     if patchelf is not None: | ||||||
|         return patchelf.path |         return patchelf.path | ||||||
| 
 | 
 | ||||||
| @@ -94,7 +96,7 @@ def _patchelf(): | |||||||
|         return exe_path |         return exe_path | ||||||
| 
 | 
 | ||||||
|     # Skip darwin |     # Skip darwin | ||||||
|     if str(spack.platforms.host()) == 'darwin': |     if is_macos: | ||||||
|         return None |         return None | ||||||
| 
 | 
 | ||||||
|     # Install the spec and return its path |     # Install the spec and return its path | ||||||
| @@ -197,6 +199,10 @@ def _placeholder(dirname): | |||||||
|     return '@' * len(dirname) |     return '@' * len(dirname) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | def _decode_macho_data(bytestring): | ||||||
|  |     return bytestring.rstrip(b'\x00').decode('ascii') | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| def macho_make_paths_relative(path_name, old_layout_root, | def macho_make_paths_relative(path_name, old_layout_root, | ||||||
|                               rpaths, deps, idpath): |                               rpaths, deps, idpath): | ||||||
|     """ |     """ | ||||||
| @@ -309,21 +315,27 @@ def modify_macho_object(cur_path, rpaths, deps, idpath, | |||||||
|     # avoid error message for libgcc_s |     # avoid error message for libgcc_s | ||||||
|     if 'libgcc_' in cur_path: |     if 'libgcc_' in cur_path: | ||||||
|         return |         return | ||||||
|     install_name_tool = executable.Executable('install_name_tool') |     args = [] | ||||||
| 
 | 
 | ||||||
|     if idpath: |     if idpath: | ||||||
|         new_idpath = paths_to_paths.get(idpath, None) |         new_idpath = paths_to_paths.get(idpath, None) | ||||||
|         if new_idpath and not idpath == new_idpath: |         if new_idpath and not idpath == new_idpath: | ||||||
|             install_name_tool('-id', new_idpath, str(cur_path)) |             args += ['-id', new_idpath] | ||||||
|     for dep in deps: |     for dep in deps: | ||||||
|         new_dep = paths_to_paths.get(dep) |         new_dep = paths_to_paths.get(dep) | ||||||
|         if new_dep and dep != new_dep: |         if new_dep and dep != new_dep: | ||||||
|             install_name_tool('-change', dep, new_dep, str(cur_path)) |             args += ['-change', dep, new_dep] | ||||||
| 
 | 
 | ||||||
|     for orig_rpath in rpaths: |     for orig_rpath in rpaths: | ||||||
|         new_rpath = paths_to_paths.get(orig_rpath) |         new_rpath = paths_to_paths.get(orig_rpath) | ||||||
|         if new_rpath and not orig_rpath == new_rpath: |         if new_rpath and not orig_rpath == new_rpath: | ||||||
|             install_name_tool('-rpath', orig_rpath, new_rpath, str(cur_path)) |             args += ['-rpath', orig_rpath, new_rpath] | ||||||
|  | 
 | ||||||
|  |     if args: | ||||||
|  |         args.append(str(cur_path)) | ||||||
|  |         install_name_tool = executable.Executable('install_name_tool') | ||||||
|  |         install_name_tool(*args) | ||||||
|  | 
 | ||||||
|     return |     return | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| @@ -339,14 +351,7 @@ def modify_object_macholib(cur_path, paths_to_paths): | |||||||
|     """ |     """ | ||||||
| 
 | 
 | ||||||
|     dll = macholib.MachO.MachO(cur_path) |     dll = macholib.MachO.MachO(cur_path) | ||||||
| 
 |     dll.rewriteLoadCommands(paths_to_paths.get) | ||||||
|     changedict = paths_to_paths |  | ||||||
| 
 |  | ||||||
|     def changefunc(path): |  | ||||||
|         npath = changedict.get(path, None) |  | ||||||
|         return npath |  | ||||||
| 
 |  | ||||||
|     dll.rewriteLoadCommands(changefunc) |  | ||||||
| 
 | 
 | ||||||
|     try: |     try: | ||||||
|         f = open(dll.filename, 'rb+') |         f = open(dll.filename, 'rb+') | ||||||
| @@ -363,30 +368,35 @@ def changefunc(path): | |||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def macholib_get_paths(cur_path): | def macholib_get_paths(cur_path): | ||||||
|  |     """Get rpaths, dependent libraries, and library id of mach-o objects. | ||||||
|     """ |     """ | ||||||
|     Get rpaths, dependencies and id of mach-o objects |     headers = macholib.MachO.MachO(cur_path).headers | ||||||
|     using python macholib package |     if not headers: | ||||||
|     """ |         tty.warn("Failed to read Mach-O headers: {0}".format(cur_path)) | ||||||
|     dll = macholib.MachO.MachO(cur_path) |         commands = [] | ||||||
|  |     else: | ||||||
|  |         if len(headers) > 1: | ||||||
|  |             # Reproduce original behavior of only returning the last mach-O | ||||||
|  |             # header section | ||||||
|  |             tty.warn("Encountered fat binary: {0}".format(cur_path)) | ||||||
|  |         commands = headers[-1].commands | ||||||
|  | 
 | ||||||
|  |     LC_ID_DYLIB = macholib.mach_o.LC_ID_DYLIB | ||||||
|  |     LC_LOAD_DYLIB = macholib.mach_o.LC_LOAD_DYLIB | ||||||
|  |     LC_RPATH = macholib.mach_o.LC_RPATH | ||||||
| 
 | 
 | ||||||
|     ident = None |     ident = None | ||||||
|     rpaths = list() |     rpaths = [] | ||||||
|     deps = list() |     deps = [] | ||||||
|     for header in dll.headers: |     for load_command, dylib_command, data in commands: | ||||||
|         rpaths = [data.rstrip(b'\0').decode('utf-8') |         cmd = load_command.cmd | ||||||
|                   for load_command, dylib_command, data in header.commands if |         if cmd == LC_RPATH: | ||||||
|                   load_command.cmd == macholib.mach_o.LC_RPATH] |             rpaths.append(_decode_macho_data(data)) | ||||||
|         deps = [data.rstrip(b'\0').decode('utf-8') |         elif cmd == LC_LOAD_DYLIB: | ||||||
|                 for load_command, dylib_command, data in header.commands if |             deps.append(_decode_macho_data(data)) | ||||||
|                 load_command.cmd == macholib.mach_o.LC_LOAD_DYLIB] |         elif cmd == LC_ID_DYLIB: | ||||||
|         idents = [data.rstrip(b'\0').decode('utf-8') |             ident = _decode_macho_data(data) | ||||||
|                   for load_command, dylib_command, data in header.commands if | 
 | ||||||
|                   load_command.cmd == macholib.mach_o.LC_ID_DYLIB] |  | ||||||
|         if len(idents) == 1: |  | ||||||
|             ident = idents[0] |  | ||||||
|     tty.debug('ident: %s' % ident) |  | ||||||
|     tty.debug('deps: %s' % deps) |  | ||||||
|     tty.debug('rpaths: %s' % rpaths) |  | ||||||
|     return (rpaths, deps, ident) |     return (rpaths, deps, ident) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| @@ -539,7 +549,7 @@ def relocate_macho_binaries(path_names, old_layout_root, new_layout_root, | |||||||
|                                                   rpaths, deps, |                                                   rpaths, deps, | ||||||
|                                                   idpath) |                                                   idpath) | ||||||
|             # replace the relativized paths with normalized paths |             # replace the relativized paths with normalized paths | ||||||
|             if platform.system().lower() == 'darwin': |             if is_macos: | ||||||
|                 modify_macho_object(path_name, rpaths, deps, |                 modify_macho_object(path_name, rpaths, deps, | ||||||
|                                     idpath, rel_to_orig) |                                     idpath, rel_to_orig) | ||||||
|             else: |             else: | ||||||
| @@ -552,7 +562,7 @@ def relocate_macho_binaries(path_names, old_layout_root, new_layout_root, | |||||||
|                                               old_layout_root, |                                               old_layout_root, | ||||||
|                                               prefix_to_prefix) |                                               prefix_to_prefix) | ||||||
|             # replace the old paths with new paths |             # replace the old paths with new paths | ||||||
|             if platform.system().lower() == 'darwin': |             if is_macos: | ||||||
|                 modify_macho_object(path_name, rpaths, deps, |                 modify_macho_object(path_name, rpaths, deps, | ||||||
|                                     idpath, paths_to_paths) |                                     idpath, paths_to_paths) | ||||||
|             else: |             else: | ||||||
| @@ -565,7 +575,7 @@ def relocate_macho_binaries(path_names, old_layout_root, new_layout_root, | |||||||
|                                                        new_layout_root, |                                                        new_layout_root, | ||||||
|                                                        rpaths, deps, idpath) |                                                        rpaths, deps, idpath) | ||||||
|             # replace the new paths with relativized paths in the new prefix |             # replace the new paths with relativized paths in the new prefix | ||||||
|             if platform.system().lower() == 'darwin': |             if is_macos: | ||||||
|                 modify_macho_object(path_name, rpaths, deps, |                 modify_macho_object(path_name, rpaths, deps, | ||||||
|                                     idpath, paths_to_paths) |                                     idpath, paths_to_paths) | ||||||
|             else: |             else: | ||||||
| @@ -579,7 +589,7 @@ def relocate_macho_binaries(path_names, old_layout_root, new_layout_root, | |||||||
|                                               old_layout_root, |                                               old_layout_root, | ||||||
|                                               prefix_to_prefix) |                                               prefix_to_prefix) | ||||||
|             # replace the old paths with new paths |             # replace the old paths with new paths | ||||||
|             if platform.system().lower() == 'darwin': |             if is_macos: | ||||||
|                 modify_macho_object(path_name, rpaths, deps, |                 modify_macho_object(path_name, rpaths, deps, | ||||||
|                                     idpath, paths_to_paths) |                                     idpath, paths_to_paths) | ||||||
|             else: |             else: | ||||||
| @@ -695,18 +705,15 @@ def make_macho_binaries_relative(cur_path_names, orig_path_names, | |||||||
|     """ |     """ | ||||||
|     Replace old RPATHs with paths relative to old_dir in binary files |     Replace old RPATHs with paths relative to old_dir in binary files | ||||||
|     """ |     """ | ||||||
|  |     if not is_macos: | ||||||
|  |         return | ||||||
|  | 
 | ||||||
|     for cur_path, orig_path in zip(cur_path_names, orig_path_names): |     for cur_path, orig_path in zip(cur_path_names, orig_path_names): | ||||||
|         rpaths = set() |         (rpaths, deps, idpath) = macholib_get_paths(cur_path) | ||||||
|         deps = set() |         paths_to_paths = macho_make_paths_relative( | ||||||
|         idpath = None |             orig_path, old_layout_root, rpaths, deps, idpath | ||||||
|         if platform.system().lower() == 'darwin': |         ) | ||||||
|             (rpaths, deps, idpath) = macholib_get_paths(cur_path) |         modify_macho_object(cur_path, rpaths, deps, idpath, paths_to_paths) | ||||||
|             paths_to_paths = macho_make_paths_relative(orig_path, |  | ||||||
|                                                        old_layout_root, |  | ||||||
|                                                        rpaths, deps, idpath) |  | ||||||
|             modify_macho_object(cur_path, |  | ||||||
|                                 rpaths, deps, idpath, |  | ||||||
|                                 paths_to_paths) |  | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def make_elf_binaries_relative(new_binaries, orig_binaries, orig_layout_root): | def make_elf_binaries_relative(new_binaries, orig_binaries, orig_layout_root): | ||||||
| @@ -915,11 +922,6 @@ def file_is_relocatable(filename, paths_to_relocate=None): | |||||||
|     default_paths_to_relocate = [spack.store.layout.root, spack.paths.prefix] |     default_paths_to_relocate = [spack.store.layout.root, spack.paths.prefix] | ||||||
|     paths_to_relocate = paths_to_relocate or default_paths_to_relocate |     paths_to_relocate = paths_to_relocate or default_paths_to_relocate | ||||||
| 
 | 
 | ||||||
|     if not (platform.system().lower() == 'darwin' |  | ||||||
|             or platform.system().lower() == 'linux'): |  | ||||||
|         msg = 'function currently implemented only for linux and macOS' |  | ||||||
|         raise NotImplementedError(msg) |  | ||||||
| 
 |  | ||||||
|     if not os.path.exists(filename): |     if not os.path.exists(filename): | ||||||
|         raise ValueError('{0} does not exist'.format(filename)) |         raise ValueError('{0} does not exist'.format(filename)) | ||||||
| 
 | 
 | ||||||
| @@ -935,11 +937,11 @@ def file_is_relocatable(filename, paths_to_relocate=None): | |||||||
|     if m_type == 'application': |     if m_type == 'application': | ||||||
|         tty.debug('{0},{1}'.format(m_type, m_subtype)) |         tty.debug('{0},{1}'.format(m_type, m_subtype)) | ||||||
| 
 | 
 | ||||||
|     if platform.system().lower() == 'linux': |     if not is_macos: | ||||||
|         if m_subtype == 'x-executable' or m_subtype == 'x-sharedlib': |         if m_subtype == 'x-executable' or m_subtype == 'x-sharedlib': | ||||||
|             rpaths = ':'.join(_elf_rpaths_for(filename)) |             rpaths = ':'.join(_elf_rpaths_for(filename)) | ||||||
|             set_of_strings.discard(rpaths) |             set_of_strings.discard(rpaths) | ||||||
|     if platform.system().lower() == 'darwin': |     else: | ||||||
|         if m_subtype == 'x-mach-binary': |         if m_subtype == 'x-mach-binary': | ||||||
|             rpaths, deps, idpath = macholib_get_paths(filename) |             rpaths, deps, idpath = macholib_get_paths(filename) | ||||||
|             set_of_strings.discard(set(rpaths)) |             set_of_strings.discard(set(rpaths)) | ||||||
| @@ -978,6 +980,14 @@ def is_binary(filename): | |||||||
|     return False |     return False | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | @llnl.util.lang.memoized | ||||||
|  | def _get_mime_type(): | ||||||
|  |     file_cmd = executable.which('file') | ||||||
|  |     for arg in ['-b', '-h', '--mime-type']: | ||||||
|  |         file_cmd.add_default_arg(arg) | ||||||
|  |     return file_cmd | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| @llnl.util.lang.memoized | @llnl.util.lang.memoized | ||||||
| def mime_type(filename): | def mime_type(filename): | ||||||
|     """Returns the mime type and subtype of a file. |     """Returns the mime type and subtype of a file. | ||||||
| @@ -988,13 +998,159 @@ def mime_type(filename): | |||||||
|     Returns: |     Returns: | ||||||
|         Tuple containing the MIME type and subtype |         Tuple containing the MIME type and subtype | ||||||
|     """ |     """ | ||||||
|     file_cmd = executable.Executable('file') |     output = _get_mime_type()(filename, output=str, error=str).strip() | ||||||
|     output = file_cmd( |     tty.debug('==> ' + output) | ||||||
|         '-b', '-h', '--mime-type', filename, output=str, error=str) |     type, _, subtype = output.partition('/') | ||||||
|     tty.debug('[MIME_TYPE] {0} -> {1}'.format(filename, output.strip())) |     return type, subtype | ||||||
|     # In corner cases the output does not contain a subtype prefixed with a / | 
 | ||||||
|     # In those cases add the / so the tuple can be formed. | 
 | ||||||
|     if '/' not in output: | # Memoize this due to repeated calls to libraries in the same directory. | ||||||
|         output += '/' | @llnl.util.lang.memoized | ||||||
|     split_by_slash = output.strip().split('/') | def _exists_dir(dirname): | ||||||
|     return split_by_slash[0], "/".join(split_by_slash[1:]) |     return os.path.isdir(dirname) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def fixup_macos_rpath(root, filename): | ||||||
|  |     """Apply rpath fixups to the given file. | ||||||
|  | 
 | ||||||
|  |     Args: | ||||||
|  |         root: absolute path to the parent directory | ||||||
|  |         filename: relative path to the library or binary | ||||||
|  | 
 | ||||||
|  |     Returns: | ||||||
|  |         True if fixups were applied, else False | ||||||
|  |     """ | ||||||
|  |     abspath = os.path.join(root, filename) | ||||||
|  |     if mime_type(abspath) != ('application', 'x-mach-binary'): | ||||||
|  |         return False | ||||||
|  | 
 | ||||||
|  |     # Get Mach-O header commands | ||||||
|  |     (rpath_list, deps, id_dylib) = macholib_get_paths(abspath) | ||||||
|  | 
 | ||||||
|  |     # Convert rpaths list to (name -> number of occurrences) | ||||||
|  |     add_rpaths = set() | ||||||
|  |     del_rpaths = set() | ||||||
|  |     rpaths = defaultdict(int) | ||||||
|  |     for rpath in rpath_list: | ||||||
|  |         rpaths[rpath] += 1 | ||||||
|  | 
 | ||||||
|  |     args = [] | ||||||
|  | 
 | ||||||
|  |     # Check dependencies for non-rpath entries | ||||||
|  |     spack_root = spack.store.layout.root | ||||||
|  |     for name in deps: | ||||||
|  |         if name.startswith(spack_root): | ||||||
|  |             tty.debug("Spack-installed dependency for {0}: {1}" | ||||||
|  |                       .format(abspath, name)) | ||||||
|  |             (dirname, basename) = os.path.split(name) | ||||||
|  |             if dirname != root or dirname in rpaths: | ||||||
|  |                 # Only change the rpath if it's a dependency *or* if the root | ||||||
|  |                 # rpath was already added to the library (this is to prevent | ||||||
|  |                 # GCC or similar getting rpaths when they weren't at all | ||||||
|  |                 # configured) | ||||||
|  |                 args += ['-change', name, '@rpath/' + basename] | ||||||
|  |                 add_rpaths.add(dirname.rstrip('/')) | ||||||
|  | 
 | ||||||
|  |     # Check for nonexistent rpaths (often added by spack linker overzealousness | ||||||
|  |     # with both lib/ and lib64/) and duplicate rpaths | ||||||
|  |     for (rpath, count) in rpaths.items(): | ||||||
|  |         if (rpath.startswith('@loader_path') | ||||||
|  |                 or rpath.startswith('@executable_path')): | ||||||
|  |             # Allowable relative paths | ||||||
|  |             pass | ||||||
|  |         elif not _exists_dir(rpath): | ||||||
|  |             tty.debug("Nonexistent rpath in {0}: {1}".format(abspath, rpath)) | ||||||
|  |             del_rpaths.add(rpath) | ||||||
|  |         elif count > 1: | ||||||
|  |             # Rpath should only be there once, but it can sometimes be | ||||||
|  |             # duplicated between Spack's compiler and libtool. If there are | ||||||
|  |             # more copies of the same one, something is very odd.... | ||||||
|  |             tty_debug = tty.debug if count == 2 else tty.warn | ||||||
|  |             tty_debug("Rpath appears {0} times in {1}: {2}".format( | ||||||
|  |                 count, abspath, rpath | ||||||
|  |             )) | ||||||
|  |             del_rpaths.add(rpath) | ||||||
|  | 
 | ||||||
|  |     # Check for relocatable ID | ||||||
|  |     if id_dylib is None: | ||||||
|  |         tty.debug("No dylib ID is set for {0}".format(abspath)) | ||||||
|  |     elif not id_dylib.startswith('@'): | ||||||
|  |         tty.debug("Non-relocatable dylib ID for {0}: {1}" | ||||||
|  |                   .format(abspath, id_dylib)) | ||||||
|  |         if root in rpaths or root in add_rpaths: | ||||||
|  |             args += ['-id', '@rpath/' + filename] | ||||||
|  |         else: | ||||||
|  |             tty.debug("Allowing hardcoded dylib ID because its rpath " | ||||||
|  |                       "is *not* in the library already") | ||||||
|  | 
 | ||||||
|  |     # Delete bad rpaths | ||||||
|  |     for rpath in del_rpaths: | ||||||
|  |         args += ['-delete_rpath', rpath] | ||||||
|  | 
 | ||||||
|  |     # Add missing rpaths that are not set for deletion | ||||||
|  |     for rpath in add_rpaths - del_rpaths - set(rpaths): | ||||||
|  |         args += ['-add_rpath', rpath] | ||||||
|  | 
 | ||||||
|  |     if not args: | ||||||
|  |         # No fixes needed | ||||||
|  |         return False | ||||||
|  | 
 | ||||||
|  |     args.append(abspath) | ||||||
|  |     executable.Executable('install_name_tool')(*args) | ||||||
|  |     return True | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def fixup_macos_rpaths(spec): | ||||||
|  |     """Remove duplicate rpaths and make shared library IDs relocatable. | ||||||
|  | 
 | ||||||
|  |     Some autotools packages write their own ``-rpath`` entries in addition to | ||||||
|  |     those implicitly added by the Spack compiler wrappers. On Linux these | ||||||
|  |     duplicate rpaths are eliminated, but on macOS they result in multiple | ||||||
|  |     entries which makes it harder to adjust with ``install_name_tool | ||||||
|  |     -delete_rpath``. | ||||||
|  | 
 | ||||||
|  |     Furthermore, many autotools programs (on macOS) set a library's install | ||||||
|  |     paths to use absolute paths rather than relative paths. | ||||||
|  |     """ | ||||||
|  |     if spec.external or spec.virtual: | ||||||
|  |         tty.warn('external or virtual package cannot be fixed up: {0!s}' | ||||||
|  |                  .format(spec)) | ||||||
|  |         return False | ||||||
|  | 
 | ||||||
|  |     if 'platform=darwin' not in spec: | ||||||
|  |         raise NotImplementedError('fixup_macos_rpaths requires macOS') | ||||||
|  | 
 | ||||||
|  |     applied = 0 | ||||||
|  | 
 | ||||||
|  |     libs = frozenset(['lib', 'lib64', 'libexec', 'plugins', | ||||||
|  |                       'Library', 'Frameworks']) | ||||||
|  |     prefix = spec.prefix | ||||||
|  | 
 | ||||||
|  |     if not os.path.exists(prefix): | ||||||
|  |         raise RuntimeError( | ||||||
|  |             'Could not fix up install prefix spec {0} because it does ' | ||||||
|  |             'not exist: {1!s}'.format(prefix, spec.name) | ||||||
|  |         ) | ||||||
|  | 
 | ||||||
|  |     # Explore the installation prefix of the spec | ||||||
|  |     for root, dirs, files in os.walk(prefix, topdown=True): | ||||||
|  |         dirs[:] = set(dirs) & libs | ||||||
|  |         for name in files: | ||||||
|  |             try: | ||||||
|  |                 needed_fix = fixup_macos_rpath(root, name) | ||||||
|  |             except Exception as e: | ||||||
|  |                 tty.warn("Failed to apply library fixups to: {0}/{1}: {2!s}" | ||||||
|  |                          .format(root, name, e)) | ||||||
|  |                 needed_fix = False | ||||||
|  |             if needed_fix: | ||||||
|  |                 applied += 1 | ||||||
|  | 
 | ||||||
|  |     specname = spec.format('{name}{/hash:7}') | ||||||
|  |     if applied: | ||||||
|  |         tty.info('Fixed rpaths for {0:d} {1} installed to {2}'.format( | ||||||
|  |             applied, | ||||||
|  |             "binary" if applied == 1 else "binaries", | ||||||
|  |             specname | ||||||
|  |         )) | ||||||
|  |     else: | ||||||
|  |         tty.debug('No rpath fixup needed for ' + specname) | ||||||
|   | |||||||
| @@ -4,7 +4,6 @@ | |||||||
| # SPDX-License-Identifier: (Apache-2.0 OR MIT) | # SPDX-License-Identifier: (Apache-2.0 OR MIT) | ||||||
| import collections | import collections | ||||||
| import os.path | import os.path | ||||||
| import platform |  | ||||||
| import re | import re | ||||||
| import shutil | import shutil | ||||||
| 
 | 
 | ||||||
| @@ -14,6 +13,7 @@ | |||||||
| 
 | 
 | ||||||
| import spack.concretize | import spack.concretize | ||||||
| import spack.paths | import spack.paths | ||||||
|  | import spack.platforms | ||||||
| import spack.relocate | import spack.relocate | ||||||
| import spack.spec | import spack.spec | ||||||
| import spack.store | import spack.store | ||||||
| @@ -21,6 +21,13 @@ | |||||||
| import spack.util.executable | import spack.util.executable | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | def skip_unless_linux(f): | ||||||
|  |     return pytest.mark.skipif( | ||||||
|  |         str(spack.platforms.real_host()) != 'linux', | ||||||
|  |         reason='implementation currently requires linux' | ||||||
|  |     )(f) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| def rpaths_for(new_binary): | def rpaths_for(new_binary): | ||||||
|     """Return the RPATHs or RUNPATHs of a binary.""" |     """Return the RPATHs or RUNPATHs of a binary.""" | ||||||
|     patchelf = spack.util.executable.which('patchelf') |     patchelf = spack.util.executable.which('patchelf') | ||||||
| @@ -144,6 +151,66 @@ def _factory(rpaths, message="Hello world!"): | |||||||
|     return _factory |     return _factory | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | @pytest.fixture() | ||||||
|  | def make_dylib(tmpdir_factory): | ||||||
|  |     """Create a shared library with unfriendly qualities. | ||||||
|  | 
 | ||||||
|  |     - Writes the same rpath twice | ||||||
|  |     - Writes its install path as an absolute path | ||||||
|  |     """ | ||||||
|  |     cc = spack.util.executable.which('cc') | ||||||
|  | 
 | ||||||
|  |     def _factory(abs_install_name="abs", extra_rpaths=[]): | ||||||
|  |         assert all(extra_rpaths) | ||||||
|  | 
 | ||||||
|  |         tmpdir = tmpdir_factory.mktemp( | ||||||
|  |             abs_install_name + '-'.join(extra_rpaths).replace('/', '') | ||||||
|  |         ) | ||||||
|  |         src = tmpdir.join('foo.c') | ||||||
|  |         src.write("int foo() { return 1; }\n") | ||||||
|  | 
 | ||||||
|  |         filename = 'foo.dylib' | ||||||
|  |         lib = tmpdir.join(filename) | ||||||
|  | 
 | ||||||
|  |         args = ['-shared', str(src), '-o', str(lib)] | ||||||
|  |         rpaths = list(extra_rpaths) | ||||||
|  |         if abs_install_name.startswith('abs'): | ||||||
|  |             args += ['-install_name', str(lib)] | ||||||
|  |         else: | ||||||
|  |             args += ['-install_name', '@rpath/' + filename] | ||||||
|  | 
 | ||||||
|  |         if abs_install_name.endswith('rpath'): | ||||||
|  |             rpaths.append(str(tmpdir)) | ||||||
|  | 
 | ||||||
|  |         args.extend('-Wl,-rpath,' + s for s in rpaths) | ||||||
|  | 
 | ||||||
|  |         cc(*args) | ||||||
|  | 
 | ||||||
|  |         return (str(tmpdir), filename) | ||||||
|  | 
 | ||||||
|  |     return _factory | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | @pytest.fixture() | ||||||
|  | def make_object_file(tmpdir): | ||||||
|  |     cc = spack.util.executable.which('cc') | ||||||
|  | 
 | ||||||
|  |     def _factory(): | ||||||
|  |         src = tmpdir.join('bar.c') | ||||||
|  |         src.write("int bar() { return 2; }\n") | ||||||
|  | 
 | ||||||
|  |         filename = 'bar.o' | ||||||
|  |         lib = tmpdir.join(filename) | ||||||
|  | 
 | ||||||
|  |         args = ['-c', str(src), '-o', str(lib)] | ||||||
|  | 
 | ||||||
|  |         cc(*args) | ||||||
|  | 
 | ||||||
|  |         return (str(tmpdir), filename) | ||||||
|  | 
 | ||||||
|  |     return _factory | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| @pytest.fixture() | @pytest.fixture() | ||||||
| def copy_binary(): | def copy_binary(): | ||||||
|     """Returns a function that copies a binary somewhere and |     """Returns a function that copies a binary somewhere and | ||||||
| @@ -179,10 +246,7 @@ def test_patchelf_is_relocatable(): | |||||||
|     assert spack.relocate.file_is_relocatable(patchelf) |     assert spack.relocate.file_is_relocatable(patchelf) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| @pytest.mark.skipif( | @skip_unless_linux | ||||||
|     platform.system().lower() != 'linux', |  | ||||||
|     reason='implementation for MacOS still missing' |  | ||||||
| ) |  | ||||||
| def test_file_is_relocatable_errors(tmpdir): | def test_file_is_relocatable_errors(tmpdir): | ||||||
|     # The file passed in as argument must exist... |     # The file passed in as argument must exist... | ||||||
|     with pytest.raises(ValueError) as exc_info: |     with pytest.raises(ValueError) as exc_info: | ||||||
| @@ -199,10 +263,7 @@ def test_file_is_relocatable_errors(tmpdir): | |||||||
|         assert 'is not an absolute path' in str(exc_info.value) |         assert 'is not an absolute path' in str(exc_info.value) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| @pytest.mark.skipif( | @skip_unless_linux | ||||||
|     platform.system().lower() != 'linux', |  | ||||||
|     reason='implementation for MacOS still missing' |  | ||||||
| ) |  | ||||||
| def test_search_patchelf(expected_patchelf_path): | def test_search_patchelf(expected_patchelf_path): | ||||||
|     current = spack.relocate._patchelf() |     current = spack.relocate._patchelf() | ||||||
|     assert current == expected_patchelf_path |     assert current == expected_patchelf_path | ||||||
| @@ -272,10 +333,7 @@ def test_set_elf_rpaths_warning(mock_patchelf): | |||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| @pytest.mark.requires_executables('patchelf', 'strings', 'file', 'gcc') | @pytest.mark.requires_executables('patchelf', 'strings', 'file', 'gcc') | ||||||
| @pytest.mark.skipif( | @skip_unless_linux | ||||||
|     platform.system().lower() != 'linux', |  | ||||||
|     reason='implementation for MacOS still missing' |  | ||||||
| ) |  | ||||||
| def test_replace_prefix_bin(hello_world): | def test_replace_prefix_bin(hello_world): | ||||||
|     # Compile an "Hello world!" executable and set RPATHs |     # Compile an "Hello world!" executable and set RPATHs | ||||||
|     executable = hello_world(rpaths=['/usr/lib', '/usr/lib64']) |     executable = hello_world(rpaths=['/usr/lib', '/usr/lib64']) | ||||||
| @@ -288,10 +346,7 @@ def test_replace_prefix_bin(hello_world): | |||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| @pytest.mark.requires_executables('patchelf', 'strings', 'file', 'gcc') | @pytest.mark.requires_executables('patchelf', 'strings', 'file', 'gcc') | ||||||
| @pytest.mark.skipif( | @skip_unless_linux | ||||||
|     platform.system().lower() != 'linux', |  | ||||||
|     reason='implementation for MacOS still missing' |  | ||||||
| ) |  | ||||||
| def test_relocate_elf_binaries_absolute_paths( | def test_relocate_elf_binaries_absolute_paths( | ||||||
|         hello_world, copy_binary, tmpdir |         hello_world, copy_binary, tmpdir | ||||||
| ): | ): | ||||||
| @@ -316,10 +371,7 @@ def test_relocate_elf_binaries_absolute_paths( | |||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| @pytest.mark.requires_executables('patchelf', 'strings', 'file', 'gcc') | @pytest.mark.requires_executables('patchelf', 'strings', 'file', 'gcc') | ||||||
| @pytest.mark.skipif( | @skip_unless_linux | ||||||
|     platform.system().lower() != 'linux', |  | ||||||
|     reason='implementation for MacOS still missing' |  | ||||||
| ) |  | ||||||
| def test_relocate_elf_binaries_relative_paths(hello_world, copy_binary): | def test_relocate_elf_binaries_relative_paths(hello_world, copy_binary): | ||||||
|     # Create an executable, set some RPATHs, copy it to another location |     # Create an executable, set some RPATHs, copy it to another location | ||||||
|     orig_binary = hello_world(rpaths=['lib', 'lib64', '/opt/local/lib']) |     orig_binary = hello_world(rpaths=['lib', 'lib64', '/opt/local/lib']) | ||||||
| @@ -340,10 +392,7 @@ def test_relocate_elf_binaries_relative_paths(hello_world, copy_binary): | |||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| @pytest.mark.requires_executables('patchelf', 'strings', 'file', 'gcc') | @pytest.mark.requires_executables('patchelf', 'strings', 'file', 'gcc') | ||||||
| @pytest.mark.skipif( | @skip_unless_linux | ||||||
|     platform.system().lower() != 'linux', |  | ||||||
|     reason='implementation for MacOS still missing' |  | ||||||
| ) |  | ||||||
| def test_make_elf_binaries_relative(hello_world, copy_binary, tmpdir): | def test_make_elf_binaries_relative(hello_world, copy_binary, tmpdir): | ||||||
|     orig_binary = hello_world(rpaths=[ |     orig_binary = hello_world(rpaths=[ | ||||||
|         str(tmpdir.mkdir('lib')), str(tmpdir.mkdir('lib64')), '/opt/local/lib' |         str(tmpdir.mkdir('lib')), str(tmpdir.mkdir('lib64')), '/opt/local/lib' | ||||||
| @@ -367,10 +416,7 @@ def test_raise_if_not_relocatable(monkeypatch): | |||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| @pytest.mark.requires_executables('patchelf', 'strings', 'file', 'gcc') | @pytest.mark.requires_executables('patchelf', 'strings', 'file', 'gcc') | ||||||
| @pytest.mark.skipif( | @skip_unless_linux | ||||||
|     platform.system().lower() != 'linux', |  | ||||||
|     reason='implementation for MacOS still missing' |  | ||||||
| ) |  | ||||||
| def test_relocate_text_bin(hello_world, copy_binary, tmpdir): | def test_relocate_text_bin(hello_world, copy_binary, tmpdir): | ||||||
|     orig_binary = hello_world(rpaths=[ |     orig_binary = hello_world(rpaths=[ | ||||||
|         str(tmpdir.mkdir('lib')), str(tmpdir.mkdir('lib64')), '/opt/local/lib' |         str(tmpdir.mkdir('lib')), str(tmpdir.mkdir('lib64')), '/opt/local/lib' | ||||||
| @@ -406,3 +452,59 @@ def test_relocate_text_bin_raise_if_new_prefix_is_longer(tmpdir): | |||||||
|         spack.relocate.relocate_text_bin( |         spack.relocate.relocate_text_bin( | ||||||
|             [fpath], {short_prefix: long_prefix} |             [fpath], {short_prefix: long_prefix} | ||||||
|         ) |         ) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | @pytest.mark.requires_executables('install_name_tool', 'file', 'cc') | ||||||
|  | def test_fixup_macos_rpaths(make_dylib, make_object_file): | ||||||
|  |     # For each of these tests except for the "correct" case, the first fixup | ||||||
|  |     # should make changes, and the second fixup should be a null-op. | ||||||
|  |     fixup_rpath = spack.relocate.fixup_macos_rpath | ||||||
|  | 
 | ||||||
|  |     no_rpath = [] | ||||||
|  |     duplicate_rpaths = ['/usr', '/usr'] | ||||||
|  |     bad_rpath = ['/nonexistent/path'] | ||||||
|  | 
 | ||||||
|  |     # Non-relocatable library id and duplicate rpaths | ||||||
|  |     (root, filename) = make_dylib("abs", duplicate_rpaths) | ||||||
|  |     assert fixup_rpath(root, filename) | ||||||
|  |     assert not fixup_rpath(root, filename) | ||||||
|  | 
 | ||||||
|  |     # Bad but relocatable library id | ||||||
|  |     (root, filename) = make_dylib("abs_with_rpath", no_rpath) | ||||||
|  |     assert fixup_rpath(root, filename) | ||||||
|  |     assert not fixup_rpath(root, filename) | ||||||
|  | 
 | ||||||
|  |     # Library id uses rpath but there are extra duplicate rpaths | ||||||
|  |     (root, filename) = make_dylib("rpath", duplicate_rpaths) | ||||||
|  |     assert fixup_rpath(root, filename) | ||||||
|  |     assert not fixup_rpath(root, filename) | ||||||
|  | 
 | ||||||
|  |     # Shared library was constructed with relocatable id from the get-go | ||||||
|  |     (root, filename) = make_dylib("rpath", no_rpath) | ||||||
|  |     assert not fixup_rpath(root, filename) | ||||||
|  | 
 | ||||||
|  |     # Non-relocatable library id | ||||||
|  |     (root, filename) = make_dylib("abs", no_rpath) | ||||||
|  |     assert not fixup_rpath(root, filename) | ||||||
|  | 
 | ||||||
|  |     # Relocatable with executable paths and loader paths | ||||||
|  |     (root, filename) = make_dylib("rpath", ['@executable_path/../lib', | ||||||
|  |                                             '@loader_path']) | ||||||
|  |     assert not fixup_rpath(root, filename) | ||||||
|  | 
 | ||||||
|  |     # Non-relocatable library id but nonexistent rpath | ||||||
|  |     (root, filename) = make_dylib("abs", bad_rpath) | ||||||
|  |     assert fixup_rpath(root, filename) | ||||||
|  |     assert not fixup_rpath(root, filename) | ||||||
|  | 
 | ||||||
|  |     # Duplicate nonexistent rpath will need *two* passes | ||||||
|  |     (root, filename) = make_dylib("rpath", bad_rpath * 2) | ||||||
|  |     assert fixup_rpath(root, filename) | ||||||
|  |     assert fixup_rpath(root, filename) | ||||||
|  |     assert not fixup_rpath(root, filename) | ||||||
|  | 
 | ||||||
|  |     # Test on an object file, which *also* has type 'application/x-mach-binary' | ||||||
|  |     # but should be ignored (no ID headers, no RPATH) | ||||||
|  |     # (this is a corner case for GCC installation) | ||||||
|  |     (root, filename) = make_object_file() | ||||||
|  |     assert not fixup_rpath(root, filename) | ||||||
|   | |||||||
| @@ -37,10 +37,10 @@ class Zstd(MakefilePackage): | |||||||
|     depends_on('lzma', when='+programs') |     depends_on('lzma', when='+programs') | ||||||
|     depends_on('lz4', when='+programs') |     depends_on('lz4', when='+programs') | ||||||
| 
 | 
 | ||||||
|     def _make(self, *args): |     def _make(self, *args, **kwargs): | ||||||
|         # PREFIX must be defined on macOS even when building the library, since |         # PREFIX must be defined on macOS even when building the library, since | ||||||
|         # it gets hardcoded into the library's install_path |         # it gets hardcoded into the library's install_path | ||||||
|         make('VERBOSE=1', 'PREFIX=' + self.prefix, '-C', *args) |         make('VERBOSE=1', 'PREFIX=' + self.prefix, '-C', *args, **kwargs) | ||||||
| 
 | 
 | ||||||
|     def build(self, spec, prefix): |     def build(self, spec, prefix): | ||||||
|         self._make('lib') |         self._make('lib') | ||||||
| @@ -48,6 +48,6 @@ def build(self, spec, prefix): | |||||||
|             self._make('programs') |             self._make('programs') | ||||||
| 
 | 
 | ||||||
|     def install(self, spec, prefix): |     def install(self, spec, prefix): | ||||||
|         self._make('lib', 'install') |         self._make('lib', 'install', parallel=False) | ||||||
|         if spec.variants['programs'].value: |         if spec.variants['programs'].value: | ||||||
|             self._make('programs', 'install') |             self._make('programs', 'install') | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 Seth R. Johnson
					Seth R. Johnson