Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Link with ld.gold by default #29974

Merged
merged 1 commit into from
Dec 30, 2015
Merged

Link with ld.gold by default #29974

merged 1 commit into from
Dec 30, 2015

Conversation

brson
Copy link
Contributor

@brson brson commented Nov 21, 2015

When using cc for linking rustc will, if gold is available (by looking for /usr/bin/ld.gold), pass -fuse-ld=gold to cc.

In some scenarios gold links much faster than ld. Servo uses it to considerably speed up linking. gold behaves nearly identically to ld (though I think there are rare corner cases that don't work still). I've run this through crater and everything there continues to link.

To disable, pass -C disable-gold.

@rust-highfive
Copy link
Contributor

r? @alexcrichton

(rust_highfive has picked a reviewer for you, use r? to override)

@@ -199,6 +200,24 @@ impl<'a> Linker for GnuLinker<'a> {
fn export_symbols(&mut self, _: &Session, _: &CrateTranslation, _: &Path) {
// noop, visibility in object files takes care of this
}

fn try_gold_linker(&mut self) {
let gold_exists = Path::new("/usr/bin/ld.gold").exists();
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You should check for it in all pathes listed in $PATH, not hard-code /usr/bin.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is very important on, e.g. NixOS, where /usr/bin only contains env.
EDIT: Seems to have been brought up by @nagisa already.

@bluss bluss added the relnotes Marks issues that should be documented in the release notes of the next release. label Nov 21, 2015
@alexcrichton
Copy link
Member

r=me with probing all of PATH instead of just /usr/bin

@brson
Copy link
Contributor Author

brson commented Nov 23, 2015

I'm not sure I agree with probing $PATH. Are there known cases where gold appears somewhere else?

Edit: Ah, I'll just do it

@petrochenkov
Copy link
Contributor

In MinGW ld.gold is in $PATH, but it can't produce Windows executables AFAIK. Won't this patch break everything on Windows by default?

@brson
Copy link
Contributor Author

brson commented Nov 23, 2015

I'll conditionalize it on linux too then.

@alexcrichton
Copy link
Member

I figured that if you go out of your way to install ld.gold to a custom location it may as well be nice for us to pick it up by default without requiring a modification of /usr/bin

@nagisa
Copy link
Member

nagisa commented Nov 23, 2015

I'm not sure I agree with probing $PATH. Are there known cases where gold appears somewhere else?


I figured that if you go out of your way to install ld.gold to a custom location it may as well be nice for us to pick it up by default without requiring a modification of /usr/bin

Some distribution/OSes/package managers install to /usr/local/bin by default (and do not link /usr/bin/), then there’s NixOS based things which install to random-looking pathes and may link things into arbitrary pathes.

@@ -515,6 +515,8 @@ options! {CodegenOptions, CodegenSetter, basic_codegen_options,
"explicitly enable the cfg(debug_assertions) directive"),
inline_threshold: Option<usize> = (None, parse_opt_uint,
"set the inlining threshold for"),
disable_gold: bool = (false, parse_bool,
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is… a pretty confusing option (both its description and behaviour).

We already have -C linker which possibly swaps out gcc for something else and now we pass a special flag into linker assuming it is gcc-compatible (in absence of this option) without checking if specified linker is indeed compatible. That might make -C linker harder to use.

Maybe it would be better to have -C linker=gcc which overrides any default/dubious flags we might add?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We already have -C linker which possibly swaps out gcc for something else and now we pass a special flag into linker assuming it is gcc-compatible (in absence of this option) without checking if specified linker is indeed compatible. That might make -C linker harder to use

As far as I can tell that is an existing problem. Even if you override the linker, rustc assumes it takes either gcc or link.exe style arguments, depending on platform. IOW I tihnk -C linker is only good for specifying a different path to the expected linker, not for using a completely different 'style' of linker.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm not super worried about random codegen options being deprecated at some point, an option like this is pretty flavorful and likely to only be a one-time use case (if ever), so if we in the future just hide this and make it ignored by default then it's probably not too bad.

It may be the case that gcc -fuse-ld=gold -fuse-ld=not-gold (or some similar incantation) also works, so we could just be sure to allow that pattern, but I don't know if it exists. Either way this seems fine to me for now.

@nagisa
Copy link
Member

nagisa commented Nov 23, 2015

Other than my nits, I really want this to land. ld.gold is, and always has been, a superior linker compared to the default.

@michaelwoerister
Copy link
Member

🐤 👍

@brson
Copy link
Contributor Author

brson commented Nov 24, 2015

I've updated it to search $PATH and to only work on linux. I haven't addressed @nagisa's comment about the confusingness of the -C disable-gold option. It's true that there are current and future configurations where it will not do anything, and also that -C linker is limited in what it actually does. I don't have a strong opinion about what the command line args look like here.


let gold_exists = match env::var("PATH") {
Ok(env_path) => {
env_path.split(":").any(|p| {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Use split_paths instead.

@alexcrichton
Copy link
Member

@petrochenkov

In MinGW ld.gold is in $PATH, but it can't produce Windows executables AFAIK. Won't this patch break everything on Windows by default?

I wouldn't be too worried about this, personally. That seems super unlikely to ever happen (if it doesn't support windows) and even if it does exist it'll exist as ld.gold.exe most likely which this isn't even probing for. @nagisa's point about OSes like FreeBSD and such also lean me towards just not checking the OS at all and blindly assuming that if ld.gold exists it's the "right linker" to use.

@nagisa
Copy link
Member

nagisa commented Nov 25, 2015

A very relevant DragonflyBSD commit

@brson
Copy link
Contributor Author

brson commented Dec 1, 2015

Gold only works reliably on x86 / x86_64 so this should probably also check that the target is x86 to avoid asking for gold on cross-compiles.

Edit: maybe arm?

Actually, checking that gold exists anywhere then asking for it for a cross-compile is probably bad. Maybe the check should be that the target == host and is a supported arch.

@brson
Copy link
Contributor Author

brson commented Dec 1, 2015

@bstrie suggests this requires an RFC. cc @rust-lang/compiler

@brson brson added T-compiler Relevant to the compiler team, which will review and decide on the PR/issue. I-nominated labels Dec 1, 2015
@nagisa
Copy link
Member

nagisa commented Dec 1, 2015

@brson my ld.gold --help prints this:

ld.gold: supported targets: elf64-littleaarch64 elf64-bigaarch64 elf32-littleaarch64 elf32-bigaarch64 elf64-tradlittlemips elf32-tradlittlemips-nacl elf64-tradbigmips elf32-tradlittlemips-nacl elf32-tradlittlemips elf32-tradlittlemips-nacl elf32-tradbigmips elf32-tradlittlemips-nacl elf32-tilegx-be elf64-tilegx-be elf32-tilegx-le elf64-tilegx-le elf32-bigarm elf32-bigarm-nacl elf32-littlearm elf32-littlearm-nacl elf64-powerpcle elf64-powerpc elf32-powerpcle elf32-powerpc elf64-sparc elf32-sparc elf32-x86-64 elf32-x86-64-freebsd elf32-x86-64-nacl elf64-x86-64 elf64-x86-64-freebsd elf64-x86-64-nacl elf32-i386 elf32-i386-freebsd elf32-i386-nacl
ld.gold: supported emulations: aarch64_elf64_le_vec aarch64_elf64_be_vec aarch64_elf32_le_vec aarch64_elf32_be_vec elf64-tradlittlemips elf32-tradlittlemips-nacl elf64-tradbigmips elf32-tradlittlemips-nacl elf32-tradlittlemips elf32-tradlittlemips-nacl elf32-tradbigmips elf32-tradlittlemips-nacl elf32tilegx_be elf64tilegx_be elf32tilegx elf64tilegx armelfb armelfb_nacl armelf armelf_nacl elf64lppc elf64ppc elf32lppc elf32ppc elf64_sparc elf32_sparc elf32_x86_64 elf32_x86_64_nacl elf_x86_64 elf_x86_64_nacl elf_i386 elf_i386_nacl

Are you claiming most of these targets do not work?

@brson
Copy link
Contributor Author

brson commented Dec 1, 2015

@nagisa No, I'm claiming I have no evidence they do. I know that I read just today that gold's powerpc support is incomplete.

Edit: I see my previous statement was strongly saying gold only works on x86. That was a mistake.

@brson
Copy link
Contributor Author

brson commented Dec 1, 2015

I made this patch more defensive. Now it only uses gold when ld.gold exists, the host is linux, the host and the target are roughly the same, and the target is x86/x86_64. As @nagisa points out there are other platforms that gold support, but I have most confidence this won't break anything on x86 linux.

@xitep
Copy link

xitep commented Dec 31, 2015

Hello,

being bound to Debian Wheezy, I'm now forced to use -C disable-gold due to gcc not supporting -fuse-ld=gold on this platform:

> cc -fuse-ld=gold -v
cc: error: unrecognized command line option ‘-fuse-ld=gold’
Using built-in specs.
COLLECT_GCC=cc
COLLECT_LTO_WRAPPER=/usr/lib/gcc/x86_64-linux-gnu/4.7/lto-wrapper
Target: x86_64-linux-gnu
Configured with: ../src/configure -v --with-pkgversion='Debian 4.7.2-5' --with-bugurl=file:///usr/share/doc/gcc-4.7/README.Bugs --enable-languages=c,c++,go,fortran,objc,obj-c++ --prefix=/usr --program-suffix=-4.7 --enable-shared --enable-linker-build-id --with-system-zlib --libexecdir=/usr/lib --without-included-gettext --enable-threads=posix --with-gxx-include-dir=/usr/include/c++/4.7 --libdir=/usr/lib --enable-nls --with-sysroot=/ --enable-clocale=gnu --enable-libstdcxx-debug --enable-libstdcxx-time=yes --enable-gnu-unique-object --enable-plugin --enable-objc-gc --with-arch-32=i586 --with-tune=generic --enable-checking=release --build=x86_64-linux-gnu --host=x86_64-linux-gnu --target=x86_64-linux-gnu
Thread model: posix
gcc version 4.7.2 (Debian 4.7.2-5) 

Unfortunately, I'm stuck with cargo based projects containing a custom build.rs as I haven't found a way to pass disable-gold to the compilation of that script (either directly in my project or any of its dependencies.) Any ideas how I could help myself without having to upgrade gcc?

@eddyb
Copy link
Member

eddyb commented Dec 31, 2015

@xitep You can write a wrapper shell script for rustc that passes -C disable-gold and the rest of the arguments to the real binary.

@xitep
Copy link

xitep commented Dec 31, 2015

@eddyb thank you so much for the idea! it works nicely.

@nagisa
Copy link
Member

nagisa commented Jan 1, 2016

This might have killed play.rust-lang.org. see e.g. https://play.rust-lang.org/?gist=5dc29f8c79c267056ca2&version=nightly cc @alexcrichton @brson

@birkenfeld
Copy link
Contributor

Another bit of incompatibility: gold does not have the same default library search path; in particular, it does not search in /usr/local/lib. This has broken building bindings for me, especially since there doesn't seem to be an easy way to add a libdir in Cargo (without build script)?

@eddyb
Copy link
Member

eddyb commented Jan 5, 2016

@birkenfeld That sounds like a distro bug rather than rustc's fault, unless it's intentional, in which case... what's the reasoning?

@birkenfeld
Copy link
Contributor

You mean, the library being in /usr/local? That's a reasonable location for self-built or site-distributed binaries, and the libs I'm writing bindings are not distributed by the system package manager.

@eddyb
Copy link
Member

eddyb commented Jan 5, 2016

@birkenfeld I was referring to "gold does not have the same default library search path".

@birkenfeld
Copy link
Contributor

Ah! That, as far I could find out from quick Googling, is "normal" behavior for gold. Apparently ld.bfd includes the runtime linker search path (i.e. ld.so.conf and LD_LIBRARY_PATH) when looking for libs, while gold doesn't (but it respects LIBRARY_PATH).

Possibly gold's behavior is cleaner here, I don't know. But it's definitely annoying to trip over these kinds of problems without a good way to disable gold from Cargo.

@eddyb
Copy link
Member

eddyb commented Jan 5, 2016

@birkenfeld if gold repsects LIBRARY_PATH, can't you just put /usr/local/lib in there?

@birkenfeld
Copy link
Contributor

Yes, I can (and I will).

Just noting this as a data point to consider before stabilizing this change (one which likely doesn't show up in a crater run).

@nagisa
Copy link
Member

nagisa commented Jan 5, 2016

ld.gold not respecting ld.so.conf would be pretty crazy. That being said, ldconfig doesn’t search /usr/local/lib by default, so the behaviour might be as expected.

@brson
Copy link
Contributor Author

brson commented Jan 8, 2016

This broke playpen on nightly.

@brson
Copy link
Contributor Author

brson commented Jan 8, 2016

being bound to Debian Wheezy, I'm now forced to use -C disable-gold due to gcc not supporting -fuse-ld=gold on this platform:

This seems really bad. So when you use the Rust downloaded from official sources, it doesn't work on Debian? I think rustc needs to be able to figure out this scenario.

@brson
Copy link
Contributor Author

brson commented Jan 8, 2016

@xitep Do you know how we could identify configurations like yours and automatically disable gold?

@brson
Copy link
Contributor Author

brson commented Jan 8, 2016

@xitep @birkenfeld I filed issues for both the regression you identified.

@birkenfeld
Copy link
Contributor

Thanks!

@xitep
Copy link

xitep commented Jan 8, 2016

@brson many thanks! so far i haven't searched intensively, but it looks like something along the lines of cc -fuse-ld=gold -v 2>&1 | grep 'unrecognized command line option' would be necessary; this would be agnostic to the host environment merely relying on cc itself. but it would require an extra invocation of the compiler. btw: relying on the exit code of cc is not possible really in this case as it ways results in exit status zero when being passed -v. however, -v is used to make it happy with no input files specified.

you can see the full output on my debian wheezy (stable) of the cc -fuse-ld=gold -v command in my comment above.

lhecker referenced this pull request in zonyitoo/coio-rs Jan 10, 2016
@lhecker
Copy link

lhecker commented Jan 10, 2016

I think I should mention, @brson, that ld.gold is also a bit more error prone on non-standard filesystems... It fails to link everytime I use it on a shared folder (the "hgfs" filesystem) inside a VMware instance. While I don't think that this is a huge issue I would recommend to note somewhere that ld.gold is at fault here (and not Rust). Maybe even directly in the output of rustc...

@brson
Copy link
Contributor Author

brson commented Jan 12, 2016

Thanks for the interesting note @lhecker.

@brson
Copy link
Contributor Author

brson commented Jan 12, 2016

@xitep Thanks for the investigation on detecting -fuse-ld. I'm noting it in the bug.

@brson brson removed the relnotes Marks issues that should be documented in the release notes of the next release. label Jan 14, 2016
bors added a commit that referenced this pull request Jan 14, 2016
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.