
## Summary Compilers stop being a *node attribute*, and become a *build-only* dependency. Packages may declare a dependency on the `c`, `cxx`, or `fortran` languages, which are now treated as virtuals, and compilers would be *providers* for one or more of those languages. Compilers can also inject runtime dependency, on the node being compiled. An example graph for something as simple as `zlib-ng` is the following: <p align="center"> <img src="https://github.com/user-attachments/assets/ee6471cb-09fd-4127-9f16-b9fe6d1338ac" alt="zlib-ng DAG" width="80%" height="auto"> </p> Here `gcc` is used for both the `c`, and `cxx` languages. Edges are annotated with the virtuals they satisfy (`c`, `cxx`, `libc`). `gcc` injects `gcc-runtime` on the nodes being compiled. `glibc` is also injected for packages that require `c`. The `compiler-wrapper` is explicitly represented as a node in the DAG, and is included in the hash. This change in the model has implications on the semantics of the `%` sigil, as discussed in #44379, and requires a version bump for our `Specfile`, `Database`, and `Lockfile` formats. ## Breaking changes Breaking changes below may impact users of this branch. ### 1. Custom, non-numeric version of compilers are not supported Currently, users can assign to compilers any custom version they want, and Spack will try to recover the "real version" whenever the custom version fails some operation. To deduce the "real version" Spack must run the compiler, which can add needless overhead to common operations. Since any information that a version like `gcc@foo` might give to the user, can also be suffixed while retaining the correct numeric version, e.g. `gcc@10.5.0-foo`, Spack will **not try** anymore to deduce real versions for compilers. Said otherwise, users should have no expectation that `gcc@foo` behaves as `gcc@X.Y.Z` internally. ### 2. The `%` sigil in the spec syntax means "direct build dependency" The `%` sigil in the spec syntax means *"direct build dependency"*, and is not a node attribute anymore. This means that: ```python node.satisfies("%gcc") ``` is true only if `gcc` is a direct build dependency of the node. *Nodes without a compiler dependency are allowed.* ### `parent["child"]`, and `node in spec`, will now only inspect the link/run sub-DAG and direct build dependencies The subscript notation for `Spec`: ```python parent["child"] ``` will look for a `child` node only in the link/run transitive graph of `parent`, and in its direct build dependencies. This means that to reach a transitive build dependency, we must first pass through the node it is associated with. Assuming `parent` does not depend on `cmake`, but depends on a `CMakePackage`, e.g. `hdf5`, then we have the following situation: ```python # This one raises an Exception, since "parent" does not depend on cmake parent["cmake"] # This one is ok cmake = parent["hdf5"]["cmake"] ``` ### 3. Externals differing by just the compiler attribute Externals are nodes where dependencies are trimmed, and that _is not planned to change_ in this branch. Currently, on `develop` it is ok to write: ```yaml packages: hdf5: externals: - spec: hdf5@1.12 %gcc prefix: /prefix/gcc - spec: hdf5@1.12 %clang prefix: /prefix/clang ``` and Spack will account for the compiler node attribute when computing the optimal spec. In this branch, using externals with a compiler specified is allowed only if any compiler in the dag matches the constraints specified on the external. _The external will be still represented as a single node without dependencies_. ### 4. Spec matrices enforcing a compiler Currently we can have matrices of the form: ```yaml matrix: - [x, y, z] - [%gcc, %clang] ``` to get the cross-product of specs and compilers. We can disregard the nature of the packages in the first row, since the compiler is a node attribute required on each node. In this branch, instead, we require a spec to depend on `c`, `cxx`, or `fortran` for the `%` to have any meaning. If any of the specs in the first row doesn't depend on these languages, there will be a concretization error. ## Deprecations * The entire `compilers` section in the configuration (i.e., `compilers.yaml`) has been deprecated, and current entries will be removed in v1.2.0. For the time being, if Spack finds any `compilers` configuration, it will try to convert it automatically to a set of external packages. * The `packages:compiler` soft-preference has been deprecated. It will be removed in v1.1.0. ## Other notable changes * The tokens `{compiler}`, `{compiler.version}`, and `{compiler.name}` in `Spec.format` expand to `"none"` if a Spec does not depend on C, C++, or Fortran. * The default install tree layout is now `"{architecture.platform}-{architecture.target}/{name}-{version}-{hash}"` ## Known limitations The major known limitations of this branch that we intend to fix before v1.0 is that compilers cannot be bootstrapped directly. In this branch we can build a new compiler using an existing external compiler, for instance: ``` $ spack install gcc@14 %gcc@10.5.0 ``` where `gcc@10.5.0` is external, and `gcc@14` is to be built. What we can't do at the moment is use a yet to be built compiler, and expect it will be bootstrapped, e.g. : ``` spack install hdf5 %gcc@14 ``` We plan to tackle this issue in a following PR. --------- Signed-off-by: Massimiliano Culpo <massimiliano.culpo@gmail.com> Signed-off-by: Todd Gamblin <tgamblin@llnl.gov> Signed-off-by: Harmen Stoppels <me@harmenstoppels.nl> Co-authored-by: Harmen Stoppels <me@harmenstoppels.nl> Co-authored-by: Todd Gamblin <tgamblin@llnl.gov>
141 lines
4.2 KiB
Bash
Executable File
141 lines
4.2 KiB
Bash
Executable File
#!/bin/bash -e
|
|
#
|
|
# Copyright Spack Project Developers. See COPYRIGHT file for details.
|
|
#
|
|
# SPDX-License-Identifier: (Apache-2.0 OR MIT)
|
|
|
|
#
|
|
# Description:
|
|
# Common setup code to be sourced by Spack's test scripts.
|
|
#
|
|
|
|
QA_DIR="$(dirname ${BASH_SOURCE[0]})"
|
|
export SPACK_ROOT=$(realpath "$QA_DIR/../../..")
|
|
|
|
# Source the setup script
|
|
. "$SPACK_ROOT/share/spack/setup-env.sh"
|
|
|
|
# by default coverage is off.
|
|
coverage=""
|
|
coverage_run=""
|
|
|
|
# Set up some variables for running coverage tests.
|
|
if [[ "$COVERAGE" == "true" ]]; then
|
|
# these set up coverage for Python
|
|
coverage=coverage
|
|
coverage_run="coverage run"
|
|
|
|
# bash coverage depends on some other factors
|
|
mkdir -p coverage
|
|
bashcov=$(realpath ${QA_DIR}/bashcov)
|
|
|
|
# instrument scripts requiring shell coverage
|
|
if [ "$(uname -o)" != "Darwin" ]; then
|
|
# On darwin, #! interpreters must be binaries, so no sbang for bashcov
|
|
sed -i "s@#\!/bin/sh@#\!${bashcov}@" "$SPACK_ROOT/bin/sbang"
|
|
fi
|
|
fi
|
|
|
|
#
|
|
# Description:
|
|
# Check to see if dependencies are installed.
|
|
# If not, warn the user and tell them how to
|
|
# install these dependencies.
|
|
#
|
|
# Usage:
|
|
# check-deps <dep> ...
|
|
#
|
|
# Options:
|
|
# One or more dependencies. Must use name of binary.
|
|
check_dependencies() {
|
|
for dep in "$@"; do
|
|
if ! which $dep &> /dev/null; then
|
|
# Map binary name to package name
|
|
case $dep in
|
|
sphinx-apidoc|sphinx-build)
|
|
spack_package=py-sphinx
|
|
pip_package=sphinx
|
|
;;
|
|
coverage)
|
|
spack_package=py-coverage
|
|
pip_package=coverage
|
|
;;
|
|
flake8)
|
|
spack_package=py-flake8
|
|
pip_package=flake8
|
|
;;
|
|
mypy)
|
|
spack_package=py-mypy
|
|
pip_package=mypy
|
|
;;
|
|
dot)
|
|
spack_package=graphviz
|
|
;;
|
|
git)
|
|
spack_package=git
|
|
;;
|
|
hg)
|
|
spack_package=mercurial
|
|
pip_package=mercurial
|
|
;;
|
|
kcov)
|
|
spack_package=kcov
|
|
;;
|
|
svn)
|
|
spack_package=subversion
|
|
;;
|
|
*)
|
|
spack_package=$dep
|
|
pip_package=$dep
|
|
;;
|
|
esac
|
|
|
|
echo "ERROR: $dep is required to run this script."
|
|
echo
|
|
|
|
if [[ $spack_package ]]; then
|
|
echo "To install with Spack, run:"
|
|
echo " $ spack install $spack_package"
|
|
fi
|
|
|
|
if [[ $pip_package ]]; then
|
|
echo "To install with pip, run:"
|
|
echo " $ pip install $pip_package"
|
|
fi
|
|
|
|
if [[ $spack_package || $pip_package ]]; then
|
|
echo "Then add the bin directory to your PATH."
|
|
fi
|
|
|
|
exit 1
|
|
fi
|
|
|
|
# Flake8 and Sphinx require setuptools in order to run.
|
|
# Otherwise, they print out this error message:
|
|
#
|
|
# Traceback (most recent call last):
|
|
# File: "/usr/bin/flake8", line 5, in <module>
|
|
# from pkg_resources import load_entry_point
|
|
# ImportError: No module named pkg_resources
|
|
#
|
|
# Print a more useful error message if setuptools not found.
|
|
if [[ $dep == flake8 || $dep == sphinx* ]]; then
|
|
# Find which Python is being run
|
|
# Spack-installed packages have a hard-coded shebang
|
|
python_cmd=$(head -n 1 $(which $dep) | cut -c 3-)
|
|
# May not have a shebang
|
|
if [[ $python_cmd != *python* ]]; then
|
|
python_cmd=python
|
|
fi
|
|
# Check if setuptools is in the PYTHONPATH
|
|
if ! $python_cmd -c "import setuptools" 2> /dev/null; then
|
|
echo "ERROR: setuptools is required to run $dep."
|
|
echo "Please add it to your PYTHONPATH."
|
|
|
|
exit 1
|
|
fi
|
|
fi
|
|
done
|
|
echo "Dependencies found."
|
|
}
|