class GccAT14 < Formula desc "GNU compiler collection" homepage "https://gcc.gnu.org/" url "https://ftpmirror.gnu.org/gnu/gcc/gcc-14.3.0/gcc-14.3.0.tar.xz" mirror "https://ftp.gnu.org/gnu/gcc/gcc-14.3.0/gcc-14.3.0.tar.xz" sha256 "e0dc77297625631ac8e50fa92fffefe899a4eb702592da5c32ef04e2293aca3a" license "GPL-3.0-or-later" => { with: "GCC-exception-3.1" } livecheck do url :stable regex(%r{href=["']?gcc[._-]v?(14(?:\.\d+)+)(?:/?["' >]|\.t)}i) end no_autobump! because: :requires_manual_review bottle do rebuild 2 sha256 arm64_tahoe: "6a6f4479e5f7f14995756fc8c1845c299680ea0c5788c534a9f625d48b757687" sha256 arm64_sequoia: "12a5bfe365a9bc4d26752a0169517ecbb3bcc168b98dfc2c1b7f034e5878dea9" sha256 arm64_sonoma: "36963df72a5f78d963aa6e2704d0d676880fbdade137f6b8887bbb7a9e6cac42" sha256 tahoe: "1f349e397413a3ea759324791f048254d1a7a557543b07525694665bc4e84554" sha256 sequoia: "d94a51af9d807077bfda895e34a09eeaefc71ecc1386a28e6497dd4457589bfb" sha256 sonoma: "07fbaa38d4dc9f30456e6c09a2fdedd590dcf5829b1849a5b2c57fb13e0d65c1" sha256 cellar: :any_skip_relocation, arm64_linux: "97fd978bc8307b20e06e35fcc5151d5ca9dc93c0c6043d56098a92c6b0b354f7" sha256 cellar: :any_skip_relocation, x86_64_linux: "14982763e0d0e18a57595baff2fa3c0df6fc5b16ae204df02cb7095038ea111c" 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 "flex" => :build uses_from_macos "m4" => :build uses_from_macos "zlib" on_macos do # macOS make is too old, has intermittent parallel build issue depends_on "make" => :build end on_linux do depends_on "binutils" end # Branch from the Darwin maintainer of GCC, with a few generic fixes and # Apple Silicon support, located at https://github.com/iains/gcc-14-branch patch do on_macos do url "https://raw.githubusercontent.com/Homebrew/homebrew-core/1cf441a0/Patches/gcc/gcc-14.3.0.diff" sha256 "b8611362ae43a5644ab908d6e4d9bfc90346a914c3ba851197086d54148b1289" end end def install # GCC will suffer build errors if forced to use a particular linker. ENV.delete "LD" # We avoiding building: # - Ada and D, which require a pre-existing GCC to bootstrap # - Go, currently not supported on macOS # - BRIG languages = %w[c c++ objc obj-c++ fortran] # Modula-2 has problems with macOS 15 for now # https://github.com/Homebrew/homebrew-core/pull/221029 languages << "m2" if !OS.mac? || MacOS.version < :sequoia pkgversion = "Homebrew GCC #{pkg_version} #{build.used_options*" "}".strip args = %W[ --prefix=#{opt_prefix} --libdir=#{opt_lib}/gcc/#{version.major} --disable-nls --enable-checking=release --with-gcc-major-version-only --enable-languages=#{languages.join(",")} --program-suffix=-#{version.major} --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} --with-system-zlib ] if OS.mac? cpu = Hardware::CPU.arm? ? "aarch64" : "x86_64" args << "--build=#{cpu}-apple-darwin#{OS.kernel_version.major}" # System headers may not be in /usr/include sdk = MacOS.sdk_path_if_needed args << "--with-sysroot=#{sdk}" if sdk # Avoid this semi-random failure: # "Error: Failed changing install name" # "Updated load commands do not fit in the header" make_args = %w[BOOT_LDFLAGS=-Wl,-headerpad_max_install_names] else # Fix Linux error: gnu/stubs-32.h: No such file or directory. args << "--disable-multilib" # Enable to PIE by default to match what the host GCC uses args << "--enable-default-pie" # 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=" inreplace "gcc/config/aarch64/t-aarch64-linux", "lp64=../lib64", "lp64=" ENV.append_path "CPATH", Formula["zlib"].opt_include ENV.append_path "LIBRARY_PATH", Formula["zlib"].opt_lib end mkdir "build" do system "../configure", *args system "gmake", *make_args # Do not strip the binaries on macOS, it makes them unsuitable # for loading plugins install_target = OS.mac? ? "install" : "install-strip" # To make sure GCC does not record cellar paths, we configure it with # opt_prefix as the prefix. Then we use DESTDIR to install into a # temporary location, then move into the cellar path. system "gmake", install_target, "DESTDIR=#{Pathname.pwd}/../instdir" mv Dir[Pathname.pwd/"../instdir/#{opt_prefix}/*"], prefix end # Handle conflicts between GCC formulae and avoid interfering # with system compilers. # Rename man7. man7.glob("*.7") { |file| add_suffix file, version.major } # Even when we disable building info pages some are still installed. rm_r(info) # Work around GCC install bug # https://gcc.gnu.org/bugzilla/show_bug.cgi?id=105664 rm_r(bin.glob("*-gcc-tmp")) 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 if OS.linux? gcc = bin/"gcc-#{version.major}" 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([specs_orig, specs].select(&:exist?)) 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} -L#{glibc.opt_lib}` 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. # Note: *link will silently add #{libdir} first to the RPATH libdir = HOMEBREW_PREFIX/"lib/gcc/#{version.major}" 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#{glibc.opt_lib}" : "+"} -L#{libdir} -L#{HOMEBREW_PREFIX}/lib *link: + --dynamic-linker #{HOMEBREW_PREFIX}/lib/ld.so -rpath #{libdir} *homebrew_rpath: -rpath #{HOMEBREW_PREFIX}/lib EOS inreplace(specs, " %o ", "\\0%(homebrew_rpath) ") end end test do (testpath/"hello-c.c").write <<~C #include int main() { puts("Hello, world!"); return 0; } C system bin/"gcc-#{version.major}", "-o", "hello-c", "hello-c.c" assert_equal "Hello, world!\n", shell_output("./hello-c") (testpath/"hello-cc.cc").write <<~CPP #include struct exception { }; int main() { std::cout << "Hello, world!" << std::endl; try { throw exception{}; } catch (exception) { } catch (...) { } return 0; } CPP system bin/"g++-#{version.major}", "-o", "hello-cc", "hello-cc.cc" assert_equal "Hello, world!\n", shell_output("./hello-cc") (testpath/"test.f90").write <<~FORTRAN 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 FORTRAN system bin/"gfortran-#{version.major}", "-o", "test", "test.f90" assert_equal "Done\n", shell_output("./test") # Modula-2 is temporarily disabled on macOS 15 return if OS.mac? && MacOS.version >= :sequoia (testpath/"hello.mod").write <<~EOS MODULE hello; FROM InOut IMPORT WriteString, WriteLn; BEGIN WriteString("Hello, world!"); WriteLn; END hello. EOS system bin/"gm2-#{version.major}", "-o", "hello-m2", "hello.mod" assert_equal "Hello, world!\n", shell_output("./hello-m2") end end