spack/var/spack/repos/builtin/packages/gcc/package.py
Harmen Stoppels e1686eef7c
gcc: use -idirafter for libc headers (#44081)
GCC C++ headers like cstdlib use `#include_next <stdlib.h>` to wrap libc
headers. We're using `-isystem` for libc, which puts those headers too
early in the search path. `-idirafter` fixes this so `include_next`
works.
2024-05-08 20:45:04 +02:00

1196 lines
51 KiB
Python

# Copyright 2013-2024 Lawrence Livermore National Security, LLC and other
# Spack Project Developers. See the top-level COPYRIGHT file for details.
#
# SPDX-License-Identifier: (Apache-2.0 OR MIT)
import glob
import itertools
import os
import sys
from archspec.cpu import UnsupportedMicroarchitecture
import llnl.util.tty as tty
from llnl.util.symlink import readlink
import spack.platforms
import spack.util.executable
import spack.util.libc
from spack.operating_systems.mac_os import macos_sdk_path, macos_version
from spack.package import *
class Gcc(AutotoolsPackage, GNUMirrorPackage, CompilerPackage):
"""The GNU Compiler Collection includes front ends for C, C++, Objective-C,
Fortran, Ada, and Go, as well as libraries for these languages."""
homepage = "https://gcc.gnu.org"
gnu_mirror_path = "gcc/gcc-9.2.0/gcc-9.2.0.tar.xz"
git = "git://gcc.gnu.org/git/gcc.git"
list_url = "https://ftp.gnu.org/gnu/gcc/"
list_depth = 1
keep_werror = "all"
maintainers("michaelkuhn", "alalazo")
license("GPL-2.0-or-later AND LGPL-2.1-or-later")
version("master", branch="master")
version("14.1.0", sha256="e283c654987afe3de9d8080bc0bd79534b5ca0d681a73a11ff2b5d3767426840")
version("13.2.0", sha256="e275e76442a6067341a27f04c5c6b83d8613144004c0413528863dc6b5c743da")
version("13.1.0", sha256="61d684f0aa5e76ac6585ad8898a2427aade8979ed5e7f85492286c4dfc13ee86")
version("12.3.0", sha256="949a5d4f99e786421a93b532b22ffab5578de7321369975b91aec97adfda8c3b")
version("12.2.0", sha256="e549cf9cf3594a00e27b6589d4322d70e0720cdd213f39beb4181e06926230ff")
version("12.1.0", sha256="62fd634889f31c02b64af2c468f064b47ad1ca78411c45abe6ac4b5f8dd19c7b")
version("11.4.0", sha256="3f2db222b007e8a4a23cd5ba56726ef08e8b1f1eb2055ee72c1402cea73a8dd9")
version("11.3.0", sha256="b47cf2818691f5b1e21df2bb38c795fac2cfbd640ede2d0a5e1c89e338a3ac39")
version("11.2.0", sha256="d08edc536b54c372a1010ff6619dd274c0f1603aa49212ba20f7aa2cda36fa8b")
version("11.1.0", sha256="4c4a6fb8a8396059241c2e674b85b351c26a5d678274007f076957afa1cc9ddf")
version("10.5.0", sha256="25109543fdf46f397c347b5d8b7a2c7e5694a5a51cce4b9c6e1ea8a71ca307c1")
version("10.4.0", sha256="c9297d5bcd7cb43f3dfc2fed5389e948c9312fd962ef6a4ce455cff963ebe4f1")
version("10.3.0", sha256="64f404c1a650f27fc33da242e1f2df54952e3963a49e06e73f6940f3223ac344")
version("10.2.0", sha256="b8dd4368bb9c7f0b98188317ee0254dd8cc99d1e3a18d0ff146c855fe16c1d8c")
version("10.1.0", sha256="b6898a23844b656f1b68691c5c012036c2e694ac4b53a8918d4712ad876e7ea2")
version("9.5.0", sha256="27769f64ef1d4cd5e2be8682c0c93f9887983e6cfd1a927ce5a0a2915a95cf8f")
version("9.4.0", sha256="c95da32f440378d7751dd95533186f7fc05ceb4fb65eb5b85234e6299eb9838e")
version("9.3.0", sha256="71e197867611f6054aa1119b13a0c0abac12834765fe2d81f35ac57f84f742d1")
version("9.2.0", sha256="ea6ef08f121239da5695f76c9b33637a118dcf63e24164422231917fa61fb206")
version("9.1.0", sha256="79a66834e96a6050d8fe78db2c3b32fb285b230b855d0a66288235bc04b327a0")
version("8.5.0", sha256="d308841a511bb830a6100397b0042db24ce11f642dab6ea6ee44842e5325ed50")
version("8.4.0", sha256="e30a6e52d10e1f27ed55104ad233c30bd1e99cfb5ff98ab022dc941edd1b2dd4")
version("8.3.0", sha256="64baadfe6cc0f4947a84cb12d7f0dfaf45bb58b7e92461639596c21e02d97d2c")
version("8.2.0", sha256="196c3c04ba2613f893283977e6011b2345d1cd1af9abeac58e916b1aab3e0080")
version("8.1.0", sha256="1d1866f992626e61349a1ccd0b8d5253816222cdc13390dcfaa74b093aa2b153")
version("7.5.0", sha256="b81946e7f01f90528a1f7352ab08cc602b9ccc05d4e44da4bd501c5a189ee661")
version("7.4.0", sha256="eddde28d04f334aec1604456e536416549e9b1aa137fc69204e65eb0c009fe51")
version("7.3.0", sha256="832ca6ae04636adbb430e865a1451adf6979ab44ca1c8374f61fba65645ce15c")
version("7.2.0", sha256="1cf7adf8ff4b5aa49041c8734bbcf1ad18cc4c94d0029aae0f4e48841088479a")
version("7.1.0", sha256="8a8136c235f64c6fef69cac0d73a46a1a09bb250776a050aec8f9fc880bebc17")
version("6.5.0", sha256="7ef1796ce497e89479183702635b14bb7a46b53249209a5e0f999bebf4740945")
version("6.4.0", sha256="850bf21eafdfe5cd5f6827148184c08c4a0852a37ccf36ce69855334d2c914d4")
version("6.3.0", sha256="f06ae7f3f790fbf0f018f6d40e844451e6bc3b7bc96e128e63b09825c1f8b29f")
version("6.2.0", sha256="9944589fc722d3e66308c0ce5257788ebd7872982a718aa2516123940671b7c5")
version("6.1.0", sha256="09c4c85cabebb971b1de732a0219609f93fc0af5f86f6e437fd8d7f832f1a351")
version("5.5.0", sha256="530cea139d82fe542b358961130c69cfde8b3d14556370b65823d2f91f0ced87")
version("5.4.0", sha256="608df76dec2d34de6558249d8af4cbee21eceddbcb580d666f7a5a583ca3303a")
version("5.3.0", sha256="b84f5592e9218b73dbae612b5253035a7b34a9a1f7688d2e1bfaaf7267d5c4db")
version("5.2.0", sha256="5f835b04b5f7dd4f4d2dc96190ec1621b8d89f2dc6f638f9f8bc1b1014ba8cad")
version("5.1.0", sha256="b7dafdf89cbb0e20333dbf5b5349319ae06e3d1a30bf3515b5488f7e89dca5ad")
version("4.9.4", sha256="6c11d292cd01b294f9f84c9a59c230d80e9e4a47e5c6355f046bb36d4f358092")
version("4.9.3", sha256="2332b2a5a321b57508b9031354a8503af6fdfb868b8c1748d33028d100a8b67e")
version("4.9.2", sha256="2020c98295856aa13fda0f2f3a4794490757fc24bcca918d52cc8b4917b972dd")
version("4.9.1", sha256="d334781a124ada6f38e63b545e2a3b8c2183049515a1abab6d513f109f1d717e")
version("4.8.5", sha256="22fb1e7e0f68a63cee631d85b20461d1ea6bda162f03096350e38c8d427ecf23")
version("4.8.4", sha256="4a80aa23798b8e9b5793494b8c976b39b8d9aa2e53cd5ed5534aff662a7f8695")
version("4.7.4", sha256="92e61c6dc3a0a449e62d72a38185fda550168a86702dea07125ebd3ec3996282")
version("4.6.4", sha256="35af16afa0b67af9b8eb15cafb76d2bc5f568540552522f5dc2c88dd45d977e8")
version("4.5.4", sha256="eef3f0456db8c3d992cbb51d5d32558190bc14f3bc19383dd93acc27acc6befc")
# We specifically do not add 'all' variant here because:
# (i) Ada, D, Go, Jit, and Objective-C++ are not default languages.
# In that respect, the name 'all' is rather misleading.
# (ii) Languages other than c,c++,fortran are prone to configure bug in GCC
# For example, 'java' appears to ignore custom location of zlib
# (iii) meaning of 'all' changes with GCC version, i.e. 'java' is not part
# of gcc7. Correctly specifying conflicts() and depends_on() in such a
# case is a PITA.
#
# Also note that some languages get enabled by the configure scripts even if not listed in the
# arguments. For example, c++ is enabled when the bootstrapping is enabled and lto is enabled
# when the link time optimization support is enabled.
variant(
"languages",
default="c,c++,fortran",
values=(
"ada",
"brig",
"c",
"c++",
"d",
"fortran",
"go",
"java",
"jit",
"lto",
"objc",
"obj-c++",
),
multi=True,
description="Compilers and runtime libraries to build",
)
variant("binutils", default=False, description="Build via binutils")
variant(
"piclibs", default=False, description="Build PIC versions of libgfortran.a and libstdc++.a"
)
variant("strip", default=False, description="Strip executables to reduce installation size")
variant("nvptx", default=False, description="Target nvptx offloading to NVIDIA GPUs")
variant("bootstrap", default=True, description="Enable 3-stage bootstrap")
variant(
"graphite", default=False, description="Enable Graphite loop optimizations (requires ISL)"
)
variant(
"build_type",
default="RelWithDebInfo",
values=("Debug", "Release", "RelWithDebInfo", "MinSizeRel"),
description="CMake-like build type. "
"Debug: -O0 -g; Release: -O3; "
"RelWithDebInfo: -O2 -g; MinSizeRel: -Os",
)
variant(
"profiled",
default=False,
description="Use Profile Guided Optimization",
when="+bootstrap %gcc",
)
depends_on("flex", type="build", when="@master")
# https://gcc.gnu.org/install/prerequisites.html
depends_on("gmp@4.3.2:")
# mawk is not sufficient for go support
depends_on("gawk@3.1.5:", type="build")
depends_on("texinfo@4.7:", type="build")
depends_on("libtool", type="build")
# dependencies required for git versions
depends_on("m4@1.4.6:", when="@master", type="build")
depends_on("automake@1.15.1:", when="@master", type="build")
depends_on("autoconf@2.69:", when="@master", type="build")
depends_on("gmake@3.80:", type="build")
depends_on("perl@5", type="build")
# GCC 7.3 does not compile with newer releases on some platforms, see
# https://github.com/spack/spack/issues/6902#issuecomment-433030376
depends_on("mpfr@2.4.2:3.1.6", when="@:9.9")
depends_on("mpfr@3.1.0:", when="@10:")
depends_on("mpc@1.0.1:", when="@4.5:")
# Already released GCC versions do not support any newer version of ISL
# GCC 5.4 https://github.com/spack/spack/issues/6902#issuecomment-433072097
# GCC 7.3 https://github.com/spack/spack/issues/6902#issuecomment-433030376
# GCC 9+ https://gcc.gnu.org/bugzilla/show_bug.cgi?id=86724
with when("+graphite"):
depends_on("isl@0.14", when="@5.0:5.2")
depends_on("isl@0.15", when="@5.3:5.9")
depends_on("isl@0.15:0.18", when="@6:8.9")
depends_on("isl@0.15:0.20", when="@9:9.9")
depends_on("isl@0.15:", when="@10:")
depends_on("zlib-api", when="@6:")
depends_on("zstd", when="@10:")
depends_on("diffutils", type="build")
depends_on("iconv", when="platform=darwin")
depends_on("gnat", when="languages=ada")
depends_on(
"binutils+gas+ld+plugins~libiberty", when="+binutils", type=("build", "link", "run")
)
depends_on("zip", type="build", when="languages=java")
# The server is sometimes a bit slow to respond
timeout = {"timeout": 60}
# TODO: integrate these libraries.
# depends_on('ppl')
# depends_on('cloog')
# https://gcc.gnu.org/install/test.html
depends_on("dejagnu@1.4.4", type="test")
depends_on("expect", type="test")
depends_on("tcl", type="test")
depends_on("autogen@5.5.4:", type="test")
depends_on("guile@1.4.1:", type="test")
# See https://go.dev/doc/install/gccgo#Releases
with when("languages=go"):
provides("go-or-gccgo-bootstrap@:1.0", when="@4.7.1:")
provides("go-or-gccgo-bootstrap@:1.2", when="@4.9:")
provides("go-or-gccgo-bootstrap@:1.4", when="@5:")
provides("go-or-gccgo-bootstrap@:1.6.1", when="@6:")
provides("go-or-gccgo-bootstrap@:1.8.1", when="@7:")
provides("go-or-gccgo-bootstrap@:1.10.1", when="@8:")
provides("go-or-gccgo-bootstrap@:1.12.2", when="@9:")
provides("go-or-gccgo-bootstrap@:1.14.6", when="@10:")
provides("go-or-gccgo-bootstrap@1.16.3:1.16.5", when="@11:")
provides("golang@:1.0", when="@4.7.1:")
provides("golang@:1.2", when="@4.9:")
provides("golang@:1.4", when="@5:")
provides("golang@:1.6.1", when="@6:")
provides("golang@:1.8.1", when="@7:")
provides("golang@:1.10.1", when="@8:")
provides("golang@:1.12.2", when="@9:")
provides("golang@:1.14.6", when="@10:")
provides("golang@1.16.3:1.16.5", when="@11:")
# GCC 4.7.1 added full support for the Go 1.x programming language.
conflicts("@:4.7.0")
# Go is not supported on macOS
conflicts("platform=darwin", msg="GCC cannot build Go support on MacOS")
# For a list of valid languages for a specific release,
# run the following command in the GCC source directory:
# $ grep ^language= gcc/*/config-lang.in
# See https://gcc.gnu.org/install/configure.html
# Support for processing BRIG 1.0 files was added in GCC 7
# BRIG is a binary format for HSAIL:
# (Heterogeneous System Architecture Intermediate Language).
# See https://gcc.gnu.org/gcc-7/changes.html
conflicts("languages=brig", when="@:6")
# BRIG does not seem to be supported on macOS
conflicts("languages=brig", when="platform=darwin")
# GCC 4.8 added a 'c' language. I'm sure C was always built,
# but this is the first version that accepts 'c' as a valid language.
conflicts("languages=c", when="@:4.7")
# The GCC Java frontend and associated libjava runtime library
# have been removed from GCC as of GCC 7.
# See https://gcc.gnu.org/gcc-7/changes.html
conflicts("languages=java", when="@7:")
# GCC 5 added the ability to build GCC as a Just-In-Time compiler.
# See https://gcc.gnu.org/gcc-5/changes.html
conflicts("languages=jit", when="@:4")
with when("languages=d"):
# The very first version of GDC that became part of GCC already supported version 2.076 of
# the language and runtime.
# See https://wiki.dlang.org/GDC#Status
provides("D@2")
# Support for the D programming language has been added to GCC 9.
# See https://gcc.gnu.org/gcc-9/changes.html#d
conflicts("@:8", msg="support for D has been added in GCC 9.1")
# Versions of GDC prior to 12 can be built with an ISO C++11 compiler. Starting version 12,
# the D frontend requires a working GDC. Moreover, it is strongly recommended to use an
# older version of GDC to build GDC.
# See https://gcc.gnu.org/install/prerequisites.html#GDC-prerequisite
with when("@12:"):
# All versions starting 12 have to be built GCC:
requires("%gcc")
# And it has to be GCC older than the version we build:
vv = ["11", "12.1.0", "12.2.0"]
for prev_v, curr_v in zip(vv, vv[1:]):
conflicts(
"%gcc@{0}:".format(curr_v),
when="@{0}".format(curr_v),
msg="'gcc@{0} languages=d' requires '%gcc@:{1}' "
"with the D language support".format(curr_v, prev_v),
)
# In principle, it is possible to have GDC even with GCC 5.
# See https://github.com/D-Programming-GDC/gdc
# We, however, require at least the oldest version that officially supports GDC. It is
# also a good opportunity to tell the users that they need a working GDC:
conflicts(
"%gcc@:8",
msg="'gcc@12: languages=d' requires '%gcc@9:' with the D language support",
)
with when("+nvptx"):
depends_on("cuda")
resource(
name="newlib",
url="ftp://sourceware.org/pub/newlib/newlib-3.0.0.20180831.tar.gz",
sha256="3ad3664f227357df15ff34e954bfd9f501009a647667cd307bf0658aefd6eb5b",
destination="newlibsource",
fetch_options=timeout,
)
nvptx_tools_ver = "2023-09-13"
depends_on("nvptx-tools@" + nvptx_tools_ver, type="build")
# NVPTX offloading supported in 7 and later by limited languages
conflicts("@:6", msg="NVPTX only supported in gcc 7 and above")
conflicts("languages=ada")
conflicts("languages=brig")
conflicts("languages=go")
conflicts("languages=java")
conflicts("languages=jit")
conflicts("languages=objc")
conflicts("languages=obj-c++")
conflicts("languages=d")
# NVPTX build disables bootstrap
conflicts("+bootstrap")
# Binutils can't build ld on macOS
conflicts("+binutils", when="platform=darwin")
# Bootstrap comparison failure:
# see https://github.com/spack/spack/issues/23296
# https://gcc.gnu.org/bugzilla/show_bug.cgi?id=100340
# on XCode 12.5
conflicts("+bootstrap", when="@:11.1 %apple-clang@12.0.5")
# aarch64/M1 is supported in GCC 11.3-12.2 and 13
requires(
"@11.3,12.2,13.1:",
when="target=aarch64: platform=darwin",
msg="Only GCC 11.3-12.2, 13.1+ support macOS M1 (aarch64)",
)
# Newer binutils than RHEL's is required to run `as` on some instructions
# generated by new GCC (see https://github.com/spack/spack/issues/12235)
conflicts("~binutils", when="@7: os=rhel6", msg="New GCC cannot use system assembler on RHEL6")
# Ditto for RHEL7/8: OpenBLAS uses flags which the RHEL system-binutils don't have:
# https://github.com/xianyi/OpenBLAS/issues/3805#issuecomment-1319878852
conflicts(
"~binutils", when="@10: os=rhel7", msg="gcc: Add +binutils - preinstalled as might be old"
)
conflicts(
"~binutils", when="@10: os=rhel8", msg="gcc: Add +binutils - preinstalled as might be old"
)
# GCC 11 requires GCC 4.8 or later (https://gcc.gnu.org/gcc-11/changes.html)
conflicts("%gcc@:4.7", when="@11:")
# https://github.com/iains/gcc-12-branch/issues/6
conflicts("@:12", when="%apple-clang@14:14.0")
if sys.platform == "darwin":
# Fix parallel build on APFS filesystem
# https://gcc.gnu.org/bugzilla/show_bug.cgi?id=81797
if macos_version() >= Version("10.13"):
patch("darwin/apfs.patch", when="@5.5.0,6.1:6.4,7.1:7.3")
# from homebrew via macports
# https://trac.macports.org/ticket/56502#no1
# see also: https://gcc.gnu.org/bugzilla/show_bug.cgi?id=83531
patch("darwin/headers-10.13-fix.patch", when="@5.5.0")
if macos_version() >= Version("10.14"):
# Fix system headers for Mojave SDK:
# https://github.com/Homebrew/homebrew-core/pull/39041
patch(
"https://raw.githubusercontent.com/Homebrew/formula-patches/b8b8e65e/gcc/8.3.0-xcode-bug-_Atomic-fix.patch",
sha256="33ee92bf678586357ee8ab9d2faddf807e671ad37b97afdd102d5d153d03ca84",
when="@6:8.3",
)
if macos_version() >= Version("10.15"):
# Fix system headers for Catalina SDK
# (otherwise __OSX_AVAILABLE_STARTING ends up undefined)
patch(
"https://raw.githubusercontent.com/Homebrew/formula-patches/b8b8e65e/gcc/9.2.0-catalina.patch",
sha256="0b8d14a7f3c6a2f0d2498526e86e088926671b5da50a554ffa6b7f73ac4f132b",
when="@9.2.0",
)
# See https://raw.githubusercontent.com/Homebrew/homebrew-core/3b7db4457ac64a31e3bbffc54b04c4bd824a4a4a/Formula/gcc.rb
patch(
"https://github.com/iains/gcc-darwin-arm64/commit/20f61faaed3b335d792e38892d826054d2ac9f15.patch?full_index=1",
sha256="c0605179a856ca046d093c13cea4d2e024809ec2ad4bf3708543fc3d2e60504b",
when="@11.2.0",
)
# Apple M1 support, created from branch of Darwin maintainer for GCC:
# https://github.com/iains/gcc-11-branch
patch(
"https://raw.githubusercontent.com/Homebrew/formula-patches/22dec3fc/gcc/gcc-11.3.0-arm.diff",
sha256="e02006b7ec917cc1390645d95735a6a866caed0dfe506d5bef742f7862cab218",
when="@11.3.0 target=aarch64:",
)
# https://github.com/iains/gcc-12-branch
patch(
"https://raw.githubusercontent.com/Homebrew/formula-patches/76677f2b/gcc/gcc-12.1.0-arm.diff",
sha256="a000f1d9cb1dd98c7c4ef00df31435cd5d712d2f9d037ddc044f8bf82a16cf35",
when="@12.1.0 target=aarch64:",
)
patch(
"https://raw.githubusercontent.com/Homebrew/formula-patches/1d184289/gcc/gcc-12.2.0-arm.diff",
sha256="a7843b5c6bf1401e40c20c72af69c8f6fc9754ae980bb4a5f0540220b3dcb62d",
when="@12.2.0 target=aarch64:",
)
patch(
"https://raw.githubusercontent.com/Homebrew/formula-patches/5c206c47/gcc/gcc-13.1.0.diff",
sha256="cb4e8a89387f748a744da0273025d0dc2e3c76780cc390b18ada704676afea11",
when="@13.1.0 target=aarch64:",
)
patch(
"https://raw.githubusercontent.com/Homebrew/formula-patches/3c5cbc8e9cf444a1967786af48e430588e1eb481/gcc/gcc-13.2.0.diff",
sha256="2df7ef067871a30b2531a2013b3db661ec9e61037341977bfc451e30bf2c1035",
when="@13.2.0 target=aarch64:",
)
conflicts("+bootstrap", when="@11.3.0,13.1: target=aarch64:")
# Use -headerpad_max_install_names in the build,
# otherwise updated load commands won't fit in the Mach-O header.
# This is needed because `gcc` avoids the superenv shim.
patch("darwin/gcc-7.1.0-headerpad.patch", when="@5:11.2")
patch("darwin/gcc-6.1.0-jit.patch", when="@5:7")
patch("darwin/gcc-4.9.patch1", when="@4.9.0:4.9.3")
patch("darwin/gcc-4.9.patch2", when="@4.9.0:4.9.3")
# https://gcc.gnu.org/bugzilla/show_bug.cgi?id=92061
patch("darwin/clang13.patch", when="@:11.1 %apple-clang@13")
patch("piclibs.patch", when="+piclibs")
patch("gcc-backport.patch", when="@4.7:4.9.3,5:5.3")
# Backport libsanitizer patch for glibc >= 2.31 and 5.3.0 <= gcc <= 9.2.0
# https://bugs.gentoo.org/708346
patch("glibc-2.31-libsanitizer-1.patch", when="@7.1.0:7.5.0,8.1.0:8.3.0,9.0.0:9.2.0")
patch("glibc-2.31-libsanitizer-1-gcc-6.patch", when="@5.3.0:5.5.0,6.1.0:6.5.0")
patch("glibc-2.31-libsanitizer-2.patch", when="@8.1.0:8.3.0,9.0.0:9.2.0")
patch("glibc-2.31-libsanitizer-2-gcc-6.patch", when="@5.3.0:5.5.0,6.1.0:6.5.0")
patch("glibc-2.31-libsanitizer-2-gcc-7.patch", when="@7.1.0:7.5.0")
patch(
"patch-2b40941d23b1570cdd90083b58fa0f66aa58c86e.patch",
when="@6.5.0,7.4.0:7.5.0,8.2.0:9.3.0",
)
patch("patch-745dae5923aba02982563481d75a21595df22ff8.patch", when="@10.1.0:10.3.0,11.1.0")
# Backport libsanitizer patch for glibc >= 2.36
# https://reviews.llvm.org/D129471
patch("glibc-2.36-libsanitizer-gcc-5-9.patch", when="@5:9")
patch("glibc-2.36-libsanitizer-gcc-10-12.patch", when="@10:10.4,11:11.3,12.1.0")
# Older versions do not compile with newer versions of glibc
# https://gcc.gnu.org/bugzilla/show_bug.cgi?id=81712
patch("ucontext_t.patch", when="@4.9,5.1:5.4,6.1:6.4,7.1")
patch("ucontext_t-java.patch", when="@4.9,5.1:5.4,6.1:6.4 languages=java")
# https://gcc.gnu.org/bugzilla/show_bug.cgi?id=81066
patch("stack_t-4.9.patch", when="@4.9")
patch("stack_t.patch", when="@5.1:5.4,6.1:6.4,7.1")
# https://bugs.busybox.net/show_bug.cgi?id=10061
patch("signal.patch", when="@4.9,5.1:5.4")
# https://gcc.gnu.org/bugzilla/show_bug.cgi?id=85835
patch("sys_ustat.h.patch", when="@5.0:6.4,7.0:7.3,8.1")
patch("sys_ustat-4.9.patch", when="@4.9")
# this patch removes cylades support from gcc-5 and allows gcc-5 to be built
# with newer glibc versions.
patch("glibc-2.31-libsanitizer-3-gcc-5.patch", when="@5.3.0:5.5.0")
# https://gcc.gnu.org/bugzilla/show_bug.cgi?id=95005
patch("zstd.patch", when="@10")
# https://gcc.gnu.org/bugzilla/show_bug.cgi?id=100102
patch("patch-fc930b3010bd0de899a3da3209eab20664ddb703.patch", when="@10.1:10.3")
patch("patch-f1feb74046e0feb0596b93bbb822fae02940a90e.patch", when="@11.1")
# libstdc++: Fix inconsistent noexcept-specific for valarray begin/end
patch(
"https://github.com/gcc-mirror/gcc/commit/423cd47cfc9640ba3d6811b780e8a0b94b704dcb.patch?full_index=1",
sha256="0d136226eb07bc43f1b15284f48bd252e3748a0426b5d7ac9084ebc406e15490",
when="@9.5.0:10.4.0,11.1.0:11.2.0",
)
# patch ICE on aarch64 in tree-vect-slp, cf: https://gcc.gnu.org/bugzilla/show_bug.cgi?id=111478
# patch taken from releases/gcc-12 branch
patch(
"https://github.com/gcc-mirror/gcc/commit/9d033155254ac6df5f47ab32896dbf336f991589.patch?full_index=1",
sha256="8b76fe575ef095b48ac45e8b56544c331663f840ce4b63abdb61510bf3647597",
when="@12.3.0 target=aarch64:",
)
# patch taken from releases/gcc-13 branch
patch(
"https://github.com/gcc-mirror/gcc/commit/7c67939ec384425a3d7383dfb4fb39aa7e9ad20a.patch?full_index=1",
sha256="f0826d7a9c9808af40f3434918f24ad942f1c6a6daec73f11cf52c544cf5fc01",
when="@13.2.0 target=aarch64:",
)
build_directory = "spack-build"
compiler_languages = ["c", "cxx", "fortran", "d", "go"]
@property
def supported_languages(self):
# This weirdness is because it could be called on an abstract spec
if "languages" not in self.spec.variants:
return self.compiler_languages
return [x for x in self.compiler_languages if x in self.spec.variants["languages"].value]
c_names = ["gcc"]
cxx_names = ["g++"]
fortran_names = ["gfortran"]
d_names = ["gdc"]
go_names = ["gccgo"]
compiler_prefixes = [r"\w+-\w+-\w+-"]
compiler_suffixes = [r"-mp-\d+(?:\.\d+)?", r"-\d+(?:\.\d+)?", r"\d\d"]
compiler_version_regex = r"(?<!clang version)\s?([0-9.]+)"
compiler_version_argument = ("-dumpfullversion", "-dumpversion")
@classmethod
def determine_version(cls, exe):
try:
output = spack.compiler.get_compiler_version_output(exe, "--version")
except Exception:
output = ""
# Apple's gcc is actually apple clang, so skip it.
if "Apple" in output:
return None
return super().determine_version(exe)
@classmethod
def filter_detected_exes(cls, prefix, exes_in_prefix):
result = []
for exe in exes_in_prefix:
# On systems like Ubuntu we might get multiple executables
# with the string "gcc" in them. See:
# https://helpmanual.io/packages/apt/gcc/
basename = os.path.basename(exe)
substring_to_be_filtered = [
"c99-gcc",
"c89-gcc",
"-nm",
"-ar",
"ranlib",
"clang", # clang++ matches g++ -> clan[g++]
]
if any(x in basename for x in substring_to_be_filtered):
continue
# Filter out links in favor of real executables on
# all systems but Cray
host_platform = str(spack.platforms.host())
if os.path.islink(exe) and host_platform != "cray":
continue
result.append(exe)
return result
@classmethod
def determine_variants(cls, exes, version_str):
compilers = cls.determine_compiler_paths(exes=exes)
languages = set()
translation = {"cxx": "c++"}
for lang, compiler in compilers.items():
languages.add(translation.get(lang, lang))
variant_str = "languages={0}".format(",".join(languages))
return variant_str, {"compilers": compilers}
@classmethod
def validate_detected_spec(cls, spec, extra_attributes):
# For GCC 'compilers' is a mandatory attribute
msg = 'the extra attribute "compilers" must be set for ' 'the detected spec "{0}"'.format(
spec
)
assert "compilers" in extra_attributes, msg
compilers = extra_attributes["compilers"]
for constraint, key in {
"languages=c": "c",
"languages=c++": "cxx",
"languages=d": "d",
"languages=fortran": "fortran",
}.items():
if spec.satisfies(constraint):
msg = "{0} not in {1}"
assert key in compilers, msg.format(key, spec)
@property
def cc(self):
msg = "cannot retrieve C compiler [spec is not concrete]"
assert self.spec.concrete, msg
if self.spec.external:
return self.spec.extra_attributes["compilers"].get("c", None)
result = None
if "languages=c" in self.spec:
result = str(self.spec.prefix.bin.gcc)
return result
@property
def cxx(self):
msg = "cannot retrieve C++ compiler [spec is not concrete]"
assert self.spec.concrete, msg
if self.spec.external:
return self.spec.extra_attributes["compilers"].get("cxx", None)
result = None
if "languages=c++" in self.spec:
result = os.path.join(self.spec.prefix.bin, "g++")
return result
@property
def fortran(self):
msg = "cannot retrieve Fortran compiler [spec is not concrete]"
assert self.spec.concrete, msg
if self.spec.external:
return self.spec.extra_attributes["compilers"].get("fortran", None)
result = None
if "languages=fortran" in self.spec:
result = str(self.spec.prefix.bin.gfortran)
return result
def url_for_version(self, version):
# This function will be called when trying to fetch from url, before
# mirrors are tried. It takes care of modifying the suffix of gnu
# mirror path so that Spack will also look for the correct file in
# the mirrors
if (version < Version("6.4.0") and version != Version("5.5.0")) or version == Version(
"7.1.0"
):
self.gnu_mirror_path = self.gnu_mirror_path.replace("xz", "bz2")
return super().url_for_version(version)
def patch(self):
spec = self.spec
prefix = self.spec.prefix
# Fix a standard header file for OS X Yosemite that
# is GCC incompatible by replacing non-GCC compliant macros
if "yosemite" in spec.architecture:
if os.path.isfile("/usr/include/dispatch/object.h"):
new_dispatch_dir = join_path(prefix, "include", "dispatch")
mkdirp(new_dispatch_dir)
new_header = join_path(new_dispatch_dir, "object.h")
install("/usr/include/dispatch/object.h", new_header)
filter_file(
r"typedef void \(\^dispatch_block_t\)\(void\)",
"typedef void* dispatch_block_t",
new_header,
)
# Use installed libz
if self.version >= Version("6"):
filter_file(
"@zlibdir@", "-L{0}".format(spec["zlib-api"].prefix.lib), "gcc/Makefile.in"
)
filter_file(
"@zlibinc@", "-I{0}".format(spec["zlib-api"].prefix.include), "gcc/Makefile.in"
)
if spec.satisfies("+nvptx"):
# backport of 383400a6078d upstream to allow support of cuda@11:
filter_file(
'#define ASM_SPEC "%{misa=*:-m %*}"',
'#define ASM_SPEC "%{misa=*:-m %*; :-m sm_35}"',
"gcc/config/nvptx/nvptx.h",
string=True,
)
filter_file(
"Target RejectNegative ToLower Joined "
"Enum(ptx_isa) Var(ptx_isa_option) Init(PTX_ISA_SM30)",
"Target RejectNegative ToLower Joined "
"Enum(ptx_isa) Var(ptx_isa_option) Init(PTX_ISA_SM35)",
"gcc/config/nvptx/nvptx.opt",
string=True,
)
self.build_optimization_config()
def get_common_target_flags(self, spec):
"""Get the right (but pessimistic) architecture specific flags supported by
both host gcc and to-be-built gcc. For example: gcc@7 %gcc@12 target=znver3
should pick -march=znver1, since that's what gcc@7 supports."""
microarchitectures = [spec.target] + spec.target.ancestors
for uarch in microarchitectures:
try:
return uarch.optimization_flags("gcc", str(spec.version))
except UnsupportedMicroarchitecture:
pass
# no arch specific flags in common, unlikely to happen.
return ""
def build_optimization_config(self):
"""Write a config/spack.mk file with sensible optimization flags, taking into
account bootstrapping subtleties."""
build_type_flags = {
"Debug": "-O0 -g",
"Release": "-O3",
"RelWithDebInfo": "-O2 -g",
"MinSizeRel": "-Os",
}
# Generic optimization flags.
flags = build_type_flags[self.spec.variants["build_type"].value]
# Pessimistic target specific flags. For example, when building
# gcc@11 %gcc@7 on znver3, Spack will fix the target to znver1 during
# concretization, so we'll stick to that. The other way around however can
# result in compilation errors, when gcc@7 is built with gcc@11, and znver3
# is taken as a the target, which gcc@7 doesn't support.
# Note we're not adding this for aarch64 because of
# https://github.com/spack/spack/issues/31184
if "+bootstrap %gcc" in self.spec and self.spec.target.family != "aarch64":
flags += " " + self.get_common_target_flags(self.spec)
if "+bootstrap" in self.spec:
variables = ["BOOT_CFLAGS", "CFLAGS_FOR_TARGET", "CXXFLAGS_FOR_TARGET"]
else:
variables = ["CFLAGS", "CXXFLAGS"]
# Redefine a few variables without losing other defaults:
# BOOT_CFLAGS = $(filter-out -O% -g%, $(BOOT_CFLAGS)) -O3
# This makes sure that build_type=Release is really -O3, not -O3 -g.
fmt_string = "{} := $(filter-out -O% -g%, $({})) {}\n"
with open("config/spack.mk", "w") as f:
for var in variables:
f.write(fmt_string.format(var, var, flags))
# Improve the build time for stage 2 a bit by enabling -O1 in stage 1.
# Note: this is ignored under ~bootstrap.
f.write("STAGE1_CFLAGS += -O1\n")
# https://gcc.gnu.org/install/configure.html
def configure_args(self):
spec = self.spec
# Generic options to compile GCC
options = [
# Distributor options
"--with-pkgversion=Spack GCC",
"--with-bugurl=https://github.com/spack/spack/issues",
# Xcode 10 dropped 32-bit support
"--disable-multilib",
"--enable-languages={0}".format(",".join(spec.variants["languages"].value)),
# Drop gettext dependency
"--disable-nls",
]
# Avoid excessive realpath/stat calls for every system header
# by making -fno-canonical-system-headers the default.
if self.version >= Version("4.8.0"):
options.append("--disable-canonical-system-headers")
# Use installed libz
if self.version >= Version("6"):
options.append("--with-system-zlib")
if "zstd" in spec:
options.append("--with-zstd-include={0}".format(spec["zstd"].headers.directories[0]))
options.append("--with-zstd-lib={0}".format(spec["zstd"].libs.directories[0]))
# Enabling language "jit" requires --enable-host-shared.
if "languages=jit" in spec:
options.append("--enable-host-shared")
# Binutils
if spec.satisfies("+binutils"):
binutils = spec["binutils"].prefix.bin
options.extend(
[
"--with-gnu-ld",
"--with-ld=" + binutils.ld,
"--with-gnu-as",
"--with-as=" + binutils.join("as"),
]
)
elif spec.satisfies("%apple-clang@15:"):
# https://github.com/iains/gcc-darwin-arm64/issues/117
# https://github.com/iains/gcc-12-branch/issues/22
# https://github.com/iains/gcc-13-branch/issues/8
options.append("--with-ld=/Library/Developer/CommandLineTools/usr/bin/ld-classic")
# enable_bootstrap
if spec.satisfies("+bootstrap"):
options.extend(["--enable-bootstrap"])
else:
options.extend(["--disable-bootstrap"])
# Configure include and lib directories explicitly for these
# dependencies since the short GCC option assumes that libraries
# are installed in "/lib" which might not be true on all OS
# (see #10842)
#
# More info at: https://gcc.gnu.org/install/configure.html
for dep_str in ("mpfr", "gmp", "mpc", "isl"):
if dep_str not in spec:
options.append("--without-{0}".format(dep_str))
continue
dep_spec = spec[dep_str]
include_dir = dep_spec.headers.directories[0]
lib_dir = dep_spec.libs.directories[0]
options.extend(
[
"--with-{0}-include={1}".format(dep_str, include_dir),
"--with-{0}-lib={1}".format(dep_str, lib_dir),
]
)
# nvptx-none offloading for host compiler
if spec.satisfies("+nvptx"):
options.extend(
[
"--enable-offload-targets=nvptx-none",
"--with-cuda-driver-include={0}".format(spec["cuda"].prefix.include),
"--with-cuda-driver-lib={0}".format(spec["cuda"].libs.directories[0]),
"--disable-bootstrap",
"--disable-multilib",
]
)
if sys.platform == "darwin":
options.extend(
[
"--with-native-system-header-dir=/usr/include",
"--with-sysroot={0}".format(macos_sdk_path()),
"--with-libiconv-prefix={0}".format(spec["iconv"].prefix),
]
)
# enable appropriate bootstrapping flags
stage1_ldflags = str(self.rpath_args)
boot_ldflags = stage1_ldflags + " -static-libstdc++ -static-libgcc"
options.append("--with-stage1-ldflags=" + stage1_ldflags)
options.append("--with-boot-ldflags=" + boot_ldflags)
options.append("--with-build-config=spack")
if "languages=d" in spec:
# Phobos is the standard library for the D Programming Language. The documentation says
# that on some targets, 'libphobos' is not enabled by default, but compiles and works
# if '--enable-libphobos' is used. Specifics are documented for affected targets.
# See https://gcc.gnu.org/install/prerequisites.html#GDC-prerequisite
# Unfortunately, it is unclear where exactly the aforementioned specifics are
# documented but GDC seems to be unusable without the library, therefore we enable it
# explicitly:
options.append("--enable-libphobos")
if spec.satisfies("@12:"):
options.append("GDC={0}".format(self.detect_gdc()))
return options
# Copy nvptx-tools into the GCC install prefix
def copy_nvptx_tools(self):
nvptx_tools_bin_path = self.spec["nvptx-tools"].prefix.bin
gcc_bin_path = self.prefix.bin
mkdirp(gcc_bin_path)
copy_list = ["as", "ld", "nm", "run", "run-single"]
for file in copy_list:
fullname = f"nvptx-none-{file}"
copy(join_path(nvptx_tools_bin_path, fullname), join_path(gcc_bin_path, fullname))
link_list = ["ar", "ranlib"]
for file in link_list:
fullname = f"nvptx-none-{file}"
orig_target = readlink(join_path(nvptx_tools_bin_path, fullname))
symlink(orig_target, join_path(gcc_bin_path, fullname))
util_dir_path = join_path(self.prefix, "nvptx-none", "bin")
mkdirp(util_dir_path)
util_list = ["ar", "as", "ld", "nm", "ranlib"]
for file in util_list:
rel_target = join_path("..", "..", "bin", f"nvptx-none-{file}")
dest_link = join_path(util_dir_path, file)
symlink(rel_target, dest_link)
# run configure/make/make(install) for the nvptx-none target
# before running the host compiler phases
@run_before("configure")
def nvptx_install(self):
spec = self.spec
prefix = self.prefix
if not spec.satisfies("+nvptx"):
return
# config.guess returns the host triple, e.g. "x86_64-pc-linux-gnu"
guess = Executable("./config.guess")
targetguess = guess(output=str).rstrip("\n")
options = getattr(self, "configure_flag_args", [])
options += ["--prefix={0}".format(prefix)]
options += [
"--with-cuda-driver-include={0}".format(spec["cuda"].prefix.include),
"--with-cuda-driver-lib={0}".format(spec["cuda"].libs.directories[0]),
]
self.copy_nvptx_tools()
pattern = join_path(self.stage.source_path, "newlibsource", "*")
files = glob.glob(pattern)
if files:
symlink(join_path(files[0], "newlib"), "newlib")
# self.build_directory = 'spack-build-nvptx'
with working_dir("spack-build-nvptx", create=True):
options = [
"--prefix={0}".format(prefix),
"--enable-languages={0}".format(",".join(spec.variants["languages"].value)),
"--with-mpfr={0}".format(spec["mpfr"].prefix),
"--with-gmp={0}".format(spec["gmp"].prefix),
"--target=nvptx-none",
"--with-build-time-tools={0}".format(join_path(prefix, "nvptx-none", "bin")),
"--enable-as-accelerator-for={0}".format(targetguess),
"--disable-sjlj-exceptions",
"--enable-newlib-io-long-long",
]
configure = Executable("../configure")
configure(*options)
make()
make("install")
@property
def build_targets(self):
if "+profiled" in self.spec:
return ["profiledbootstrap"]
return []
@property
def install_targets(self):
if "+strip" in self.spec:
return ["install-strip"]
return ["install"]
@property
def spec_dir(self):
# e.g. lib/gcc/x86_64-unknown-linux-gnu/4.9.2
spec_dir = glob.glob(f"{self.prefix.lib}/gcc/*/*")
return spec_dir[0] if spec_dir else None
@run_after("install")
def write_specs_file(self):
"""(1) inject an rpath to its runtime library dir, (2) add a default programs search path
to <binutils>/bin."""
if not self.spec_dir:
tty.warn(f"Could not install specs for {self.spec.format('{name}{@version}')}.")
return
# Find which directories have shared libraries
for dir in ["lib64", "lib"]:
libdir = join_path(self.prefix, dir)
if glob.glob(join_path(libdir, "libgcc_s.*")):
rpath_dir = libdir
break
else:
tty.warn("No dynamic libraries found in lib/lib64")
rpath_dir = None
specs_file = join_path(self.spec_dir, "specs")
with open(specs_file, "w") as f:
# can't extend the builtins without dumping them first
f.write(self.spec["gcc"].command("-dumpspecs", output=str, error=os.devnull).strip())
f.write("\n\n# Generated by Spack\n\n")
# rpath
if rpath_dir:
f.write(f"*link_libgcc:\n+ -rpath={rpath_dir}\n\n")
# programs search path
if self.spec.satisfies("+binutils"):
f.write(f"*self_spec:\n+ -B{self.spec['binutils'].prefix.bin}\n\n")
set_install_permissions(specs_file)
tty.info(f"Wrote new spec file to {specs_file}")
def setup_run_environment(self, env):
# Search prefix directory for possibly modified compiler names
from spack.compilers.gcc import Gcc as Compiler
# Get the contents of the installed binary directory
bin_path = self.spec.prefix.bin
if not os.path.isdir(bin_path):
return
bin_contents = os.listdir(bin_path)
# Find the first non-symlink compiler binary present for each language
for lang in ["cc", "cxx", "fc", "f77"]:
for filename, regexp in itertools.product(bin_contents, Compiler.search_regexps(lang)):
if not regexp.match(filename):
continue
abspath = os.path.join(bin_path, filename)
# Skip broken symlinks (https://github.com/spack/spack/issues/41327)
if not os.path.exists(abspath):
continue
# Set the proper environment variable
env.set(lang.upper(), abspath)
# Stop searching filename/regex combos for this language
break
def detect_gdc(self):
"""Detect and return the path to GDC that belongs to the same instance of GCC that is used
by self.compiler.
If the path cannot be detected, raise InstallError with recommendations for the users on
how to circumvent the problem.
Should be use only if self.spec.satisfies("@12: languages=d")
"""
# Detect GCC package in the directory of the GCC compiler
# or in the $PATH if self.compiler.cc is not an absolute path:
from spack.detection import by_path
compiler_dir = os.path.dirname(self.compiler.cc)
detected_packages = by_path(
[self.name], path_hints=([compiler_dir] if os.path.isdir(compiler_dir) else None)
)
# We consider only packages that satisfy the following constraint:
required_spec = Spec("languages=c,c++,d")
candidate_specs = [
p.spec
for p in filter(
lambda p: p.spec.satisfies(required_spec), detected_packages.get(self.name, ())
)
]
if candidate_specs:
# We now need to filter specs that match the compiler version:
compiler_spec = Spec(repr(self.compiler.spec))
# First, try to filter specs that satisfy the compiler spec:
new_candidate_specs = list(
filter(lambda s: s.satisfies(compiler_spec), candidate_specs)
)
# The compiler version might be more specific than what we can detect. For example, the
# user might have "gcc@10.2.1-sys" as the compiler spec in compilers.yaml. In that
# case, we end up with an empty list of candidates. To circumvent the problem, we try
# to filter specs that are satisfied by the compiler spec:
if not new_candidate_specs:
new_candidate_specs = list(
filter(lambda s: compiler_spec.satisfies(s), candidate_specs)
)
candidate_specs = new_candidate_specs
error_nl = "\n " # see SpackError.__str__()
if not candidate_specs:
raise InstallError(
"Cannot detect GDC",
long_msg="Starting version 12, the D frontend requires a working GDC."
"{0}You can install it with Spack by running:"
"{0}{0}spack install gcc@9:11 languages=c,c++,d"
"{0}{0}Once that has finished, you will need to add it to your compilers.yaml file"
"{0}and use it to install this spec (i.e. {1} ...).".format(
error_nl, self.spec.format("{name}{@version} {variants.languages}")
),
)
elif len(candidate_specs) == 0:
return candidate_specs[0].extra_attributes["compilers"]["d"]
else:
# It is rather unlikely to end up here but let us try to resolve the ambiguity:
candidate_gdc = candidate_specs[0].extra_attributes["compilers"]["d"]
if all(
candidate_gdc == s.extra_attributes["compilers"]["d"] for s in candidate_specs[1:]
):
# It does not matter which one we take if they are all the same:
return candidate_gdc
else:
raise InstallError(
"Cannot resolve ambiguity when detecting GDC that belongs to "
"%{0}".format(self.compiler.spec),
long_msg="The candidates are:{0}{0}{1}{0}".format(
error_nl,
error_nl.join(
"{0} (cc: {1})".format(
s.extra_attributes["compilers"]["d"],
s.extra_attributes["compilers"]["c"],
)
for s in candidate_specs
),
),
)
@classmethod
def runtime_constraints(cls, *, spec, pkg):
"""Callback function to inject runtime-related rules into the solver.
Rule-injection is obtained through method calls of the ``pkg`` argument.
Documentation for this function is temporary. When the API will be in its final state,
we'll document the behavior at https://spack.readthedocs.io/en/latest/
Args:
spec: spec that will inject runtime dependencies
pkg: object used to forward information to the solver
"""
pkg("*").depends_on(
"gcc-runtime",
when="%gcc",
type="link",
description="If any package uses %gcc, it depends on gcc-runtime",
)
pkg("*").depends_on(
f"gcc-runtime@{str(spec.version)}:",
when=f"%{str(spec)}",
type="link",
description=f"If any package uses %{str(spec)}, "
f"it depends on gcc-runtime@{str(spec.version)}:",
)
gfortran_str = "libgfortran@5"
if spec.satisfies("gcc@:6"):
gfortran_str = "libgfortran@3"
elif spec.satisfies("gcc@7"):
gfortran_str = "libgfortran@4"
for fortran_virtual in ("fortran-rt", gfortran_str):
pkg("*").depends_on(
fortran_virtual,
when=f"%{str(spec)}",
languages=["fortran"],
type="link",
description=f"Add a dependency on '{gfortran_str}' for nodes compiled with "
f"{str(spec)} and using the 'fortran' language",
)
# The version of gcc-runtime is the same as the %gcc used to "compile" it
pkg("gcc-runtime").requires(f"@={str(spec.version)}", when=f"%{str(spec)}")
def _post_buildcache_install_hook(self):
if not self.spec.satisfies("platform=linux"):
return
# Setting up the runtime environment shouldn't be necessary here.
relocation_args = []
gcc = self.spec["gcc"].command
specs_file = os.path.join(self.spec_dir, "specs")
dryrun = gcc("test.c", "-###", output=os.devnull, error=str).strip()
if not dryrun:
tty.warn(f"Cannot relocate {specs_file}, compiler might not be working properly")
return
dynamic_linker = spack.util.libc.parse_dynamic_linker(dryrun)
if not dynamic_linker:
tty.warn(f"Cannot relocate {specs_file}, compiler might not be working properly")
return
libc = spack.util.libc.libc_from_dynamic_linker(dynamic_linker)
# We search for crt1.o ourselves because `gcc -print-prile-name=crt1.o` can give a rather
# convoluted relative path from a different prefix.
startfile_prefix = spack.util.libc.startfile_prefix(libc.external_path, dynamic_linker)
gcc_can_locate = lambda p: os.path.isabs(
gcc(f"-print-file-name={p}", output=str, error=os.devnull).strip()
)
if not gcc_can_locate("crt1.o"):
relocation_args.append(f"-B{startfile_prefix}")
# libc headers may also be in a multiarch subdir.
header_dir = spack.util.libc.libc_include_dir_from_startfile_prefix(
libc.external_path, startfile_prefix
)
if header_dir and all(
os.path.exists(os.path.join(header_dir, h))
for h in libc.package_class.representative_headers
):
relocation_args.append(f"-idirafter {header_dir}")
else:
tty.warn(
f"Cannot relocate {specs_file} include directories, "
f"compiler might not be working properly"
)
# Delete current spec files.
try:
os.unlink(specs_file)
except OSError:
pass
# Write a new one and append flags for libc
self.write_specs_file()
if relocation_args:
with open(specs_file, "a") as f:
f.write(f"*self_spec:\n+ {' '.join(relocation_args)}\n\n")