aboutsummaryrefslogtreecommitdiff
path: root/mesonbuild
diff options
context:
space:
mode:
authorMichael Hirsch, Ph.D <scivision@users.noreply.github.com>2019-11-10 10:14:01 -0500
committerJussi Pakkanen <jpakkane@gmail.com>2019-11-28 12:51:58 +0200
commitc02593fddc70402d8a3f50b8ae476e1778d5cd3c (patch)
treeec6106c602800cb3f771b03074ba570c82f13b8a /mesonbuild
parentdd15c47ea8c8cde4b954dad1ea9625c360412f14 (diff)
downloadmeson-c02593fddc70402d8a3f50b8ae476e1778d5cd3c.zip
meson-c02593fddc70402d8a3f50b8ae476e1778d5cd3c.tar.gz
meson-c02593fddc70402d8a3f50b8ae476e1778d5cd3c.tar.bz2
HDF5: make much more robust across platforms
This addresses various real-world problems with HDF5 pkg-config, including * hdf*.pc with package versions as part of the filename * malformed hdf*.pc missing the commonly-used HDF5 HL module --- Additionally, this refactors more complicated dependencies such as HDF5 and OpenMPI. This may help us deduplicate internal dependency code in the future. HDF5 selftest: improve platform-agnostic test ci: init demo github action for HDF5 framework ci Actions: hold off on MSYS2 for now [skip ci] hdf5: ensure C libraries always included ci: mac hdf5--use clang+gfortran
Diffstat (limited to 'mesonbuild')
-rw-r--r--mesonbuild/dependencies/__init__.py3
-rw-r--r--mesonbuild/dependencies/hdf5.py120
-rw-r--r--mesonbuild/dependencies/misc.py47
3 files changed, 122 insertions, 48 deletions
diff --git a/mesonbuild/dependencies/__init__.py b/mesonbuild/dependencies/__init__.py
index c2c5e6d..1b7c03f 100644
--- a/mesonbuild/dependencies/__init__.py
+++ b/mesonbuild/dependencies/__init__.py
@@ -14,6 +14,7 @@
from .boost import BoostDependency
from .cuda import CudaDependency
+from .hdf5 import HDF5Dependency
from .base import ( # noqa: F401
Dependency, DependencyException, DependencyMethods, ExternalProgram, EmptyExternalProgram, NonExistingExternalProgram,
ExternalDependency, NotFoundDependency, ExternalLibrary, ExtraFrameworkDependency, InternalDependency,
@@ -21,7 +22,7 @@ from .base import ( # noqa: F401
from .dev import GMockDependency, GTestDependency, LLVMDependency, ValgrindDependency
from .coarrays import CoarrayDependency
from .mpi import MPIDependency
-from .misc import (BlocksDependency, HDF5Dependency, NetCDFDependency, OpenMPDependency, Python3Dependency, ThreadDependency, PcapDependency, CupsDependency, LibWmfDependency, LibGCryptDependency, GpgmeDependency, ShadercDependency)
+from .misc import (BlocksDependency, NetCDFDependency, OpenMPDependency, Python3Dependency, ThreadDependency, PcapDependency, CupsDependency, LibWmfDependency, LibGCryptDependency, GpgmeDependency, ShadercDependency)
from .platform import AppleFrameworks
from .ui import GLDependency, GnuStepDependency, Qt4Dependency, Qt5Dependency, SDL2Dependency, WxDependency, VulkanDependency
diff --git a/mesonbuild/dependencies/hdf5.py b/mesonbuild/dependencies/hdf5.py
new file mode 100644
index 0000000..b73e424
--- /dev/null
+++ b/mesonbuild/dependencies/hdf5.py
@@ -0,0 +1,120 @@
+# Copyright 2013-2019 The Meson development team
+
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+
+# http://www.apache.org/licenses/LICENSE-2.0
+
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+
+# This file contains the detection logic for miscellaneous external dependencies.
+
+import subprocess
+from pathlib import Path
+
+from .. import mlog
+from ..mesonlib import split_args
+from .base import DependencyException, ExternalDependency, ExternalProgram, PkgConfigDependency
+
+class HDF5Dependency(ExternalDependency):
+
+ def __init__(self, environment, kwargs):
+ language = kwargs.get('language', 'c')
+ super().__init__('hdf5', environment, language, kwargs)
+ kwargs['required'] = False
+ kwargs['silent'] = True
+ self.is_found = False
+
+ # 1. pkg-config
+ pkgconfig_files = ['hdf5', 'hdf5-serial']
+ # some distros put hdf5-1.2.3.pc with version number in .pc filename.
+ ret = subprocess.run(['pkg-config', '--list-all'], stdout=subprocess.PIPE, stderr=subprocess.DEVNULL,
+ universal_newlines=True)
+ if ret.returncode == 0:
+ for pkg in ret.stdout.split('\n'):
+ if pkg.startswith(('hdf5')):
+ pkgconfig_files.append(pkg.split(' ', 1)[0])
+ pkgconfig_files = list(set(pkgconfig_files)) # dedupe
+
+ if language not in ('c', 'cpp', 'fortran'):
+ raise DependencyException('Language {} is not supported with HDF5.'.format(language))
+
+ for pkg in pkgconfig_files:
+ pkgdep = PkgConfigDependency(pkg, environment, kwargs, language=self.language)
+ if not pkgdep.found():
+ continue
+
+ self.compile_args = pkgdep.get_compile_args()
+ # some broken pkgconfig don't actually list the full path to the needed includes
+ newinc = []
+ for arg in self.compile_args:
+ if arg.startswith('-I'):
+ stem = 'static' if kwargs.get('static', False) else 'shared'
+ if (Path(arg[2:]) / stem).is_dir():
+ newinc.append('-I' + str(Path(arg[2:]) / stem))
+ self.compile_args += newinc
+
+ # derive needed libraries by language
+ pd_link_args = pkgdep.get_link_args()
+ link_args = []
+ for larg in pd_link_args:
+ lpath = Path(larg)
+ # some pkg-config hdf5.pc (e.g. Ubuntu) don't include the commonly-used HL HDF5 libraries,
+ # so let's add them if they exist
+ # additionally, some pkgconfig HDF5 HL files are malformed so let's be sure to find HL anyway
+ if lpath.is_file():
+ hl = []
+ if language == 'cpp':
+ hl += ['_hl_cpp', '_cpp']
+ elif language == 'fortran':
+ hl += ['_hl_fortran', 'hl_fortran', '_fortran']
+ hl += ['_hl'] # C HL library, always needed
+
+ suffix = '.' + lpath.name.split('.', 1)[1] # in case of .dll.a
+ for h in hl:
+ hlfn = lpath.parent / (lpath.name.split('.', 1)[0] + h + suffix)
+ if hlfn.is_file():
+ link_args.append(str(hlfn))
+ # HDF5 C libs are required by other HDF5 languages
+ link_args.append(larg)
+ else:
+ link_args.append(larg)
+
+ self.link_args = link_args
+ self.version = pkgdep.get_version()
+ self.is_found = True
+ self.pcdep = pkgdep
+ return
+
+ # 2. compiler wrapper fallback
+ wrappers = {'c': 'h5cc', 'cpp': 'h5c++', 'fortran': 'h5fc'}
+ comp_args = []
+ link_args = []
+ # have to always do C as well as desired language
+ for lang in set([language, 'c']):
+ prog = ExternalProgram(wrappers[lang], silent=True)
+ if not prog.found():
+ return
+ cmd = prog.get_command() + ['-show']
+ p = subprocess.run(cmd, stdout=subprocess.PIPE, stderr=subprocess.PIPE, universal_newlines=True, timeout=15)
+ if p.returncode != 0:
+ mlog.debug('Command', mlog.bold(cmd), 'failed to run:')
+ mlog.debug(mlog.bold('Standard output\n'), p.stdout)
+ mlog.debug(mlog.bold('Standard error\n'), p.stderr)
+ return
+ args = split_args(p.stdout)
+ for arg in args[1:]:
+ if arg.startswith(('-I', '-f', '-D')) or arg == '-pthread':
+ comp_args.append(arg)
+ elif arg.startswith(('-L', '-l', '-Wl')):
+ link_args.append(arg)
+ elif Path(arg).is_file():
+ link_args.append(arg)
+ self.compile_args = comp_args
+ self.link_args = link_args
+ self.is_found = True
diff --git a/mesonbuild/dependencies/misc.py b/mesonbuild/dependencies/misc.py
index 789015a..bfd450c 100644
--- a/mesonbuild/dependencies/misc.py
+++ b/mesonbuild/dependencies/misc.py
@@ -30,53 +30,6 @@ from .base import (
)
-class HDF5Dependency(ExternalDependency):
-
- def __init__(self, environment, kwargs):
- language = kwargs.get('language', 'c')
- super().__init__('hdf5', environment, language, kwargs)
- kwargs['required'] = False
- kwargs['silent'] = True
- self.is_found = False
-
- pkgconfig_files = ['hdf5']
-
- if language not in ('c', 'cpp', 'fortran'):
- raise DependencyException('Language {} is not supported with HDF5.'.format(language))
-
- for pkg in pkgconfig_files:
- try:
- pkgdep = PkgConfigDependency(pkg, environment, kwargs, language=self.language)
- if pkgdep.found():
- self.compile_args = pkgdep.get_compile_args()
- # derive needed libraries by language
- pd_link_args = pkgdep.get_link_args()
- link_args = []
- for larg in pd_link_args:
- lpath = Path(larg)
- if lpath.is_file():
- if language == 'cpp':
- link_args.append(str(lpath.parent / (lpath.stem + '_hl_cpp' + lpath.suffix)))
- link_args.append(str(lpath.parent / (lpath.stem + '_cpp' + lpath.suffix)))
- elif language == 'fortran':
- link_args.append(str(lpath.parent / (lpath.stem + 'hl_fortran' + lpath.suffix)))
- link_args.append(str(lpath.parent / (lpath.stem + '_fortran' + lpath.suffix)))
-
- # HDF5 C libs are required by other HDF5 languages
- link_args.append(str(lpath.parent / (lpath.stem + '_hl' + lpath.suffix)))
- link_args.append(larg)
- else:
- link_args.append(larg)
-
- self.link_args = link_args
- self.version = pkgdep.get_version()
- self.is_found = True
- self.pcdep = pkgdep
- break
- except Exception:
- pass
-
-
class NetCDFDependency(ExternalDependency):
def __init__(self, environment, kwargs):