Skip to content
  1. Feb 16, 2022
    • Christopher Di Bella's avatar
      [llvm-libgcc] initial commit · c5a20b51
      Christopher Di Bella authored
      Note: the term "libgcc" refers to the all of `libgcc.a`, `libgcc_eh.a`,
      and `libgcc_s.so`.
      
      Enabling libunwind as a replacement for libgcc on Linux has proven to be
      challenging since libgcc_s.so is a required dependency in the [Linux
      standard base][5]. Some software is transitively dependent on libgcc
      because glibc makes hardcoded calls to functions in libgcc_s. For example,
      the function `__GI___backtrace` eventually makes its way to a [hardcoded
      dlopen to libgcc_s' _Unwind_Backtrace][1]. Since libgcc_{eh.a,s.so} and
      libunwind have the same ABI, but different implementations, the two
      libraries end up [cross-talking, which ultimately results in a
      segfault][2].
      
      To solve this problem, libunwind needs to build a “libgcc”. That is, link
      the necessary functions from compiler-rt and libunwind into an archive
      and shared object that advertise themselves as `libgcc.a`, `libgcc_eh.a`,
      and `libgcc_s.so`, so that glibc’s baked calls are diverted to the
      correct objects in memory. Fortunately for us, compiler-rt and libunwind
      use the same ABI as the libgcc family, so the problem is solvable at the
      llvm-project configuration level: no program source needs to be edited.
      Thus, the end result is for a user to configure their LLVM build with a
      flag that indicates they want to archive compiler-rt/unwind as libgcc.
      We achieve this by compiling libunwind with all the symbols necessary
      for compiler-rt to emulate the libgcc family, and then generate symlinks
      named for our "libgcc" that point to their corresponding libunwind
      counterparts.
      
      We alternatively considered patching glibc so that the source doesn't
      directly refer to libgcc, but rather _defaults_ to libgcc, so that a
      system preferring compiler-rt/libunwind can point to these libraries
      at the config stage instead. Even if we modified the Linux standard
      base, this alternative won't work because binaries that are built using
      libgcc will still end up having crosstalk between the differing
      implementations.
      
      This problem has been solved in this manner for [FreeBSD][3], and this
      CL has been tested against [Chrome OS][4].
      
      [1]: https://github.com/bminor/glibc/blob/master/sysdeps/arm/backtrace.c#L68
      [2]: https://bugs.chromium.org/p/chromium/issues/detail?id=1162190#c16
      [3]: https://github.com/freebsd/freebsd-src/tree/main/lib/libgcc_s
      [4]: https://chromium-review.googlesource.com/c/chromiumos/overlays/chromiumos-overlay/+/2945947
      [5]: https://refspecs.linuxbase.org/LSB_5.0.0/LSB-Core-generic/LSB-Core-generic/libgcc-s.html
      
      Differential Revision: https://reviews.llvm.org/D108416
      c5a20b51
Loading