class Gcc < Formula desc "GNU compiler collection" homepage "https://gcc.gnu.org/" if Hardware::CPU.arm? # Branch from the Darwin maintainer of GCC with Apple Silicon support, # located at https://github.com/iains/gcc-darwin-arm64 and # backported with his help to gcc-11 branch. Too big for a patch. url "https://github.com/fxcoudert/gcc/archive/refs/tags/gcc-11.1.0-arm-20210504.tar.gz" sha256 "ce862b4a4bdc8f36c9240736d23cd625a48af82c2332d2915df0e16e1609a74c" version "11.1.0" else url "https://ftp.gnu.org/gnu/gcc/gcc-11.1.0/gcc-11.1.0.tar.xz" mirror "https://ftpmirror.gnu.org/gcc/gcc-11.1.0/gcc-11.1.0.tar.xz" sha256 "4c4a6fb8a8396059241c2e674b85b351c26a5d678274007f076957afa1cc9ddf" end license "GPL-3.0-or-later" => { with: "GCC-exception-3.1" } revision 1 head "https://gcc.gnu.org/git/gcc.git" livecheck do # Should be # url :stable # but that does not work with the ARM-specific branch above url "https://ftp.gnu.org/gnu/gcc/gcc-11.1.0" regex(%r{href=.*?gcc[._-]v?(\d+(?:\.\d+)+)(?:/?["' >]|\.t)}i) end bottle do sha256 cellar: :any, arm64_big_sur: "5ad4c157cf19f01c6acfed380db28ff15276f02f4b5d6a20f5a7034583b174aa" sha256 cellar: :any, big_sur: "4ec68e83ce46f4c686a4c9a7f90a748705543826da81e4c74c78d210b6c66c81" sha256 cellar: :any, catalina: "c8405807d9bdab853432100e8d85bf3b4c7d4a4123067f099699a492d40a430b" sha256 cellar: :any, mojave: "cac0a37271b71e40b3df7b9fa83190c11dfcd9640d8b3d02bc2ba2bae5b964ac" end # The bottles are built on systems with the CLT installed, and do not work # out of the box on Xcode-only systems due to an incorrect sysroot. pour_bottle? only_if: :clt_installed depends_on "gmp" depends_on "isl" depends_on "libmpc" depends_on "mpfr" depends_on "zstd" uses_from_macos "zlib" on_linux do depends_on "binutils" end # GCC bootstraps itself, so it is OK to have an incompatible C++ stdlib cxxstdlib_check :skip def version_suffix if build.head? "HEAD" else version.major.to_s end end def install # GCC will suffer build errors if forced to use a particular linker. ENV.delete "LD" # We avoiding building: # - Ada, which requires a pre-existing GCC Ada compiler to bootstrap # - Go, currently not supported on macOS # - BRIG languages = %w[c c++ objc obj-c++ fortran] languages << "d" if Hardware::CPU.intel? pkgversion = "Homebrew GCC #{pkg_version} #{build.used_options*" "}".strip cpu = Hardware::CPU.arm? ? "aarch64" : "x86_64" args = %W[ --prefix=#{prefix} --libdir=#{lib}/gcc/#{version_suffix} --disable-nls --enable-checking=release --enable-languages=#{languages.join(",")} --program-suffix=-#{version_suffix} --with-gmp=#{Formula["gmp"].opt_prefix} --with-mpfr=#{Formula["mpfr"].opt_prefix} --with-mpc=#{Formula["libmpc"].opt_prefix} --with-isl=#{Formula["isl"].opt_prefix} --with-zstd=#{Formula["zstd"].opt_prefix} --with-pkgversion=#{pkgversion} --with-bugurl=#{tap.issues_url} ] # libphobos is part of gdc args << "--enable-libphobos" if Hardware::CPU.intel? on_macos do args << "--build=#{cpu}-apple-darwin#{OS.kernel_version.major}" args << "--with-system-zlib" # Xcode 10 dropped 32-bit support args << "--disable-multilib" if DevelopmentTools.clang_build_version >= 1000 # Workaround for Xcode 12.5 bug on Intel # https://gcc.gnu.org/bugzilla/show_bug.cgi?id=100340 args << "--without-build-config" if Hardware::CPU.intel? && DevelopmentTools.clang_build_version >= 1205 # System headers may not be in /usr/include sdk = MacOS.sdk_path_if_needed if sdk args << "--with-native-system-header-dir=/usr/include" args << "--with-sysroot=#{sdk}" end # Ensure correct install names when linking against libgcc_s; # see discussion in https://github.com/Homebrew/legacy-homebrew/pull/34303 inreplace "libgcc/config/t-slibgcc-darwin", "@shlib_slibdir@", "#{HOMEBREW_PREFIX}/lib/gcc/#{version_suffix}" end on_linux do # Fix cc1: error while loading shared libraries: libisl.so.15 args << "--with-boot-ldflags=-static-libstdc++ -static-libgcc #{ENV["LDFLAGS"]}" # Fix Linux error: gnu/stubs-32.h: No such file or directory. args << "--disable-multilib" # Change the default directory name for 64-bit libraries to `lib` # https://stackoverflow.com/a/54038769 inreplace "gcc/config/i386/t-linux64", "m64=../lib64", "m64=" end mkdir "build" do system "../configure", *args on_macos do # 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. system "make", "BOOT_LDFLAGS=-Wl,-headerpad_max_install_names" system "make", "install" end on_linux do system "make" system "make", "install-strip" end bin.install_symlink bin/"gfortran-#{version_suffix}" => "gfortran" bin.install_symlink bin/"gdc-#{version_suffix}" => "gdc" if Hardware::CPU.intel? on_linux do # Only the newest brewed gcc should install gfortan libs as we can only have one. lib.install_symlink Dir[lib/"gcc/#{version_suffix}/libgfortran.*"] end end # Handle conflicts between GCC formulae and avoid interfering # with system compilers. # Rename man7. Dir.glob(man7/"*.7") { |file| add_suffix file, version_suffix } # Even when we disable building info pages some are still installed. info.rmtree end def add_suffix(file, suffix) dir = File.dirname(file) ext = File.extname(file) base = File.basename(file, ext) File.rename file, "#{dir}/#{base}-#{suffix}#{ext}" end def post_install on_linux do gcc = bin/"gcc-#{version_suffix}" libgcc = Pathname.new(Utils.safe_popen_read(gcc, "-print-libgcc-file-name")).parent raise "command failed: #{gcc} -print-libgcc-file-name" if $CHILD_STATUS.exitstatus.nonzero? glibc = Formula["glibc"] glibc_installed = glibc.any_version_installed? # Symlink system crt1.o and friends where gcc can find it. crtdir = if glibc_installed glibc.opt_lib else Pathname.new(Utils.safe_popen_read("/usr/bin/cc", "-print-file-name=crti.o")).parent end ln_sf Dir[crtdir/"*crt?.o"], libgcc # Create the GCC specs file # See https://gcc.gnu.org/onlinedocs/gcc/Spec-Files.html # Locate the specs file specs = libgcc/"specs" ohai "Creating the GCC specs file: #{specs}" specs_orig = Pathname.new("#{specs}.orig") rm_f [specs_orig, specs] system_header_dirs = ["#{HOMEBREW_PREFIX}/include"] if glibc_installed # https://github.com/Linuxbrew/brew/issues/724 system_header_dirs << glibc.opt_include else # Locate the native system header dirs if user uses system glibc target = Utils.safe_popen_read(gcc, "-print-multiarch").chomp raise "command failed: #{gcc} -print-multiarch" if $CHILD_STATUS.exitstatus.nonzero? system_header_dirs += ["/usr/include/#{target}", "/usr/include"] end # Save a backup of the default specs file specs_string = Utils.safe_popen_read(gcc, "-dumpspecs") raise "command failed: #{gcc} -dumpspecs" if $CHILD_STATUS.exitstatus.nonzero? specs_orig.write specs_string # Set the library search path # For include path: # * `-isysroot #{HOMEBREW_PREFIX}/nonexistent` prevents gcc searching built-in # system header files. # * `-idirafter ` instructs gcc to search system header # files after gcc internal header files. # For libraries: # * `-nostdlib -L#{libgcc}` instructs gcc to use brewed glibc # if applied. # * `-L#{libdir}` instructs gcc to find the corresponding gcc # libraries. It is essential if there are multiple brewed gcc # with different versions installed. # Noted that it should only be passed for the `gcc@*` formulae. # * `-L#{HOMEBREW_PREFIX}/lib` instructs gcc to find the rest # brew libraries. libdir = HOMEBREW_PREFIX/"lib/gcc/#{version_suffix}" specs.write specs_string + <<~EOS *cpp_unique_options: + -isysroot #{HOMEBREW_PREFIX}/nonexistent #{system_header_dirs.map { |p| "-idirafter #{p}" }.join(" ")} *link_libgcc: #{glibc_installed ? "-nostdlib -L#{libgcc}" : "+"} -L#{libdir} -L#{HOMEBREW_PREFIX}/lib *link: + --dynamic-linker #{HOMEBREW_PREFIX}/lib/ld.so -rpath #{libdir} -rpath #{HOMEBREW_PREFIX}/lib EOS end end test do (testpath/"hello-c.c").write <<~EOS #include int main() { puts("Hello, world!"); return 0; } EOS system "#{bin}/gcc-#{version_suffix}", "-o", "hello-c", "hello-c.c" assert_equal "Hello, world!\n", `./hello-c` (testpath/"hello-cc.cc").write <<~EOS #include struct exception { }; int main() { std::cout << "Hello, world!" << std::endl; try { throw exception{}; } catch (exception) { } catch (...) { } return 0; } EOS system "#{bin}/g++-#{version_suffix}", "-o", "hello-cc", "hello-cc.cc" assert_equal "Hello, world!\n", `./hello-cc` (testpath/"test.f90").write <<~EOS integer,parameter::m=10000 real::a(m), b(m) real::fact=0.5 do concurrent (i=1:m) a(i) = a(i) + fact*b(i) end do write(*,"(A)") "Done" end EOS system "#{bin}/gfortran", "-o", "test", "test.f90" assert_equal "Done\n", `./test` if Hardware::CPU.intel? (testpath/"hello_d.d").write <<~EOS import std.stdio; int main() { writeln("Hello, world!"); return 0; } EOS system "#{bin}/gdc-#{version_suffix}", "-o", "hello-d", "hello_d.d" assert_equal "Hello, world!\n", `./hello-d` end end end