Blame SOURCES/0004-PATCH-clang-Prefer-gcc-toolchains-with-libgcc_s.so-w.patch

b99d8d
From d8af49687765744efaae7ba0f0c4c0fcd58a0e31 Mon Sep 17 00:00:00 2001
78ae70
From: serge-sans-paille <sguelton@redhat.com>
78ae70
Date: Wed, 23 Sep 2020 12:47:30 +0000
b99d8d
Subject: [PATCH 4/6] [PATCH][clang] Prefer gcc toolchains with libgcc_s.so
b99d8d
 when not static linking libgcc
78ae70
78ae70
Fedora ships cross-compilers on all platforms, so a user could end up
78ae70
with a gcc x86_64 cross-compiler installed on an x86_64 system. clang
78ae70
maintains a list of supported triples for each target and when all
78ae70
else is equal will prefer toolchains with triples that appear earlier
78ae70
in the list.
78ae70
78ae70
The cross-compiler triple on Fedora is x86_64-linux-gnu and this comes
78ae70
before the Fedora system compiler's triple: x86_64-redhat-linux in
78ae70
the triples list, so the cross compiler is always preferred. This
78ae70
is a problem, because the cross compiler is missing libraries, like
78ae70
libgcc_s.so, that clang expects to be there so linker invocations
78ae70
will fail.
78ae70
78ae70
This patch fixes this by checking for the existence of libgcc_s.so
78ae70
when it is required and taking that into account when selecting a
78ae70
toolchain.
78ae70
---
b99d8d
 clang/lib/Driver/ToolChains/Gnu.cpp                      | 16 ++++++++++++++--
b99d8d
 clang/lib/Driver/ToolChains/Gnu.h                        |  4 +++-
b99d8d
 .../usr/lib/gcc/x86_64-linux-gnu/7/crtbegin.o            |  0
b99d8d
 .../usr/lib/gcc/x86_64-redhat-linux/7/crtbegin.o         |  0
b99d8d
 .../usr/lib/gcc/x86_64-redhat-linux/7/libgcc_s.so        |  0
b99d8d
 clang/test/Driver/linux-ld.c                             | 12 ++++++++++++
78ae70
 6 files changed, 29 insertions(+), 3 deletions(-)
b99d8d
 create mode 100644 clang/test/Driver/Inputs/fedora_28_tree/usr/lib/gcc/x86_64-linux-gnu/7/crtbegin.o
b99d8d
 create mode 100644 clang/test/Driver/Inputs/fedora_28_tree/usr/lib/gcc/x86_64-redhat-linux/7/crtbegin.o
b99d8d
 create mode 100644 clang/test/Driver/Inputs/fedora_28_tree/usr/lib/gcc/x86_64-redhat-linux/7/libgcc_s.so
78ae70
b99d8d
diff --git a/clang/lib/Driver/ToolChains/Gnu.cpp b/clang/lib/Driver/ToolChains/Gnu.cpp
b99d8d
index 5deeb10..5d51517 100644
b99d8d
--- a/clang/lib/Driver/ToolChains/Gnu.cpp
b99d8d
+++ b/clang/lib/Driver/ToolChains/Gnu.cpp
b99d8d
@@ -2539,6 +2539,8 @@ void Generic_GCC::GCCInstallationDetector::ScanLibDirForGCCTriple(
78ae70
        (TargetArch == llvm::Triple::x86 &&
78ae70
         TargetTriple.getOS() != llvm::Triple::Solaris)}};
78ae70
 
78ae70
+  bool NeedLibgccShared = !Args.hasArg(options::OPT_static_libgcc) &&
78ae70
+                          !Args.hasArg(options::OPT_static);
78ae70
   for (auto &Suffix : Suffixes) {
78ae70
     if (!Suffix.Active)
78ae70
       continue;
b99d8d
@@ -2556,8 +2558,17 @@ void Generic_GCC::GCCInstallationDetector::ScanLibDirForGCCTriple(
78ae70
           continue; // Saw this path before; no need to look at it again.
78ae70
       if (CandidateVersion.isOlderThan(4, 1, 1))
78ae70
         continue;
78ae70
-      if (CandidateVersion <= Version)
78ae70
-        continue;
78ae70
+
78ae70
+      bool CandidateHasLibGccShared = false;
78ae70
+      if (CandidateVersion <= Version) {
78ae70
+        if (NeedLibgccShared && !HasLibGccShared) {
78ae70
+          CandidateHasLibGccShared =
78ae70
+                D.getVFS().exists(LI->path() + "/libgcc_s.so");
78ae70
+
78ae70
+        }
78ae70
+        if (HasLibGccShared || !CandidateHasLibGccShared)
78ae70
+          continue;
78ae70
+      }
78ae70
 
78ae70
       if (!ScanGCCForMultilibs(TargetTriple, Args, LI->path(),
78ae70
                                NeedsBiarchSuffix))
b99d8d
@@ -2571,6 +2582,7 @@ void Generic_GCC::GCCInstallationDetector::ScanLibDirForGCCTriple(
78ae70
       GCCInstallPath = (LibDir + "/" + LibSuffix + "/" + VersionText).str();
78ae70
       GCCParentLibPath = (GCCInstallPath + "/../" + Suffix.ReversePath).str();
78ae70
       IsValid = true;
78ae70
+      HasLibGccShared = CandidateHasLibGccShared;
78ae70
     }
78ae70
   }
78ae70
 }
b99d8d
diff --git a/clang/lib/Driver/ToolChains/Gnu.h b/clang/lib/Driver/ToolChains/Gnu.h
b99d8d
index 90d3baf..9d0cea2 100644
b99d8d
--- a/clang/lib/Driver/ToolChains/Gnu.h
b99d8d
+++ b/clang/lib/Driver/ToolChains/Gnu.h
78ae70
@@ -190,6 +190,7 @@ public:
78ae70
   /// Driver, and has logic for fuzzing that where appropriate.
78ae70
   class GCCInstallationDetector {
78ae70
     bool IsValid;
78ae70
+    bool HasLibGccShared;
78ae70
     llvm::Triple GCCTriple;
78ae70
     const Driver &D;
78ae70
 
b99d8d
@@ -216,7 +217,8 @@ public:
b99d8d
     const std::string GentooConfigDir = "/etc/env.d/gcc";
78ae70
 
78ae70
   public:
78ae70
-    explicit GCCInstallationDetector(const Driver &D) : IsValid(false), D(D) {}
78ae70
+    explicit GCCInstallationDetector(const Driver &D)
78ae70
+        : IsValid(false), HasLibGccShared(false), D(D) {}
78ae70
     void init(const llvm::Triple &TargetTriple, const llvm::opt::ArgList &Args,
78ae70
               ArrayRef<std::string> ExtraTripleAliases = None);
78ae70
 
b99d8d
diff --git a/clang/test/Driver/Inputs/fedora_28_tree/usr/lib/gcc/x86_64-linux-gnu/7/crtbegin.o b/clang/test/Driver/Inputs/fedora_28_tree/usr/lib/gcc/x86_64-linux-gnu/7/crtbegin.o
78ae70
new file mode 100644
b99d8d
index 0000000..e69de29
b99d8d
diff --git a/clang/test/Driver/Inputs/fedora_28_tree/usr/lib/gcc/x86_64-redhat-linux/7/crtbegin.o b/clang/test/Driver/Inputs/fedora_28_tree/usr/lib/gcc/x86_64-redhat-linux/7/crtbegin.o
78ae70
new file mode 100644
b99d8d
index 0000000..e69de29
b99d8d
diff --git a/clang/test/Driver/Inputs/fedora_28_tree/usr/lib/gcc/x86_64-redhat-linux/7/libgcc_s.so b/clang/test/Driver/Inputs/fedora_28_tree/usr/lib/gcc/x86_64-redhat-linux/7/libgcc_s.so
78ae70
new file mode 100644
b99d8d
index 0000000..e69de29
b99d8d
diff --git a/clang/test/Driver/linux-ld.c b/clang/test/Driver/linux-ld.c
b99d8d
index 24d3c78..071bb9b 100644
b99d8d
--- a/clang/test/Driver/linux-ld.c
b99d8d
+++ b/clang/test/Driver/linux-ld.c
78ae70
@@ -784,6 +784,18 @@
78ae70
 // CHECK-FEDORA-31-RISCV64: "{{.*}}/usr/lib/gcc/riscv64-redhat-linux/9{{/|\\\\}}crtend.o"
78ae70
 // CHECK-FEDORA-31-RISCV64: "{{.*}}/usr/lib/gcc/riscv64-redhat-linux/9{{/|\\\\}}crtn.o"
78ae70
 //
78ae70
+// Check that clang does not select the cross compiler by default on Fedora 28.
78ae70
+//
78ae70
+// RUN: %clang -no-canonical-prefixes %s -### -o %t.o 2>&1 \
78ae70
+// RUN:     --target=x86_64-unknown-linux-gnu \
78ae70
+// RUN:     --gcc-toolchain="" \
78ae70
+// RUN:     --sysroot=%S/Inputs/fedora_28_tree \
78ae70
+// RUN:   | FileCheck --check-prefix=CHECK-FEDORA-28-X86_64 %s
78ae70
+//
78ae70
+// CHECK-FEDORA-28-X86_64: "{{.*}}ld{{(.exe)?}}" "--sysroot=[[SYSROOT:[^"]+]]"
78ae70
+// CHECK-FEDORA-28-X86_64: "[[SYSROOT]]/usr/lib/gcc/x86_64-redhat-linux/7/crtbegin.o"
78ae70
+// CHECK-FEDORA-28-X86_64: "-L[[SYSROOT]]/usr/lib/gcc/x86_64-redhat-linux/7"
78ae70
+//
78ae70
 // RUN: %clang -no-canonical-prefixes %s -### -o %t.o 2>&1 \
78ae70
 // RUN:     --target=arm-unknown-linux-gnueabi -rtlib=platform \
78ae70
 // RUN:     --gcc-toolchain="" \
78ae70
-- 
b99d8d
1.8.3.1
78ae70