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

linker: Move native library search from linker to rustc #138753

Open
wants to merge 3 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
43 changes: 25 additions & 18 deletions compiler/rustc_codegen_ssa/src/back/linker.rs
Original file line number Diff line number Diff line change
Expand Up @@ -6,9 +6,7 @@ use std::{env, io, iter, mem, str};

use cc::windows_registry;
use rustc_hir::def_id::{CrateNum, LOCAL_CRATE};
use rustc_metadata::{
find_native_static_library, try_find_native_dynamic_library, try_find_native_static_library,
};
use rustc_metadata::{try_find_native_dynamic_library, try_find_native_static_library};
use rustc_middle::bug;
use rustc_middle::middle::dependency_format::Linkage;
use rustc_middle::middle::exported_symbols;
Expand Down Expand Up @@ -614,15 +612,15 @@ impl<'a> Linker for GccLinker<'a> {
}

fn link_staticlib_by_name(&mut self, name: &str, verbatim: bool, whole_archive: bool) {
if let Some(path) = try_find_native_static_library(self.sess, name, verbatim) {
return self.link_staticlib_by_path(&path, whole_archive);
}
self.hint_static();
let colon = if verbatim && self.is_gnu { ":" } else { "" };
if !whole_archive {
self.link_or_cc_arg(format!("-l{colon}{name}"));
} else if self.sess.target.is_like_osx {
// -force_load is the macOS equivalent of --whole-archive, but it
// involves passing the full path to the library to link.
self.link_arg("-force_load");
self.link_arg(find_native_static_library(name, verbatim, self.sess));
self.link_args(&["-force_load", name]);
} else {
self.link_arg("--whole-archive")
.link_or_cc_arg(format!("-l{colon}{name}"))
Expand Down Expand Up @@ -953,15 +951,12 @@ impl<'a> Linker for MsvcLinker<'a> {
}

fn link_staticlib_by_name(&mut self, name: &str, verbatim: bool, whole_archive: bool) {
// On MSVC-like targets rustc supports static libraries using alternative naming
// scheme (`libfoo.a`) unsupported by linker, search for such libraries manually.
if let Some(path) = try_find_native_static_library(self.sess, name, verbatim) {
self.link_staticlib_by_path(&path, whole_archive);
} else {
let prefix = if whole_archive { "/WHOLEARCHIVE:" } else { "" };
let suffix = if verbatim { "" } else { ".lib" };
self.link_arg(format!("{prefix}{name}{suffix}"));
return self.link_staticlib_by_path(&path, whole_archive);
}
let prefix = if whole_archive { "/WHOLEARCHIVE:" } else { "" };
let suffix = if verbatim { "" } else { ".lib" };
self.link_arg(format!("{prefix}{name}{suffix}"));
}

fn link_staticlib_by_path(&mut self, path: &Path, whole_archive: bool) {
Expand Down Expand Up @@ -1190,7 +1185,10 @@ impl<'a> Linker for EmLinker<'a> {
self.link_or_cc_arg(path);
}

fn link_staticlib_by_name(&mut self, name: &str, _verbatim: bool, _whole_archive: bool) {
fn link_staticlib_by_name(&mut self, name: &str, verbatim: bool, whole_archive: bool) {
if let Some(path) = try_find_native_static_library(self.sess, name, verbatim) {
return self.link_staticlib_by_path(&path, whole_archive);
}
self.link_or_cc_args(&["-l", name]);
}

Expand Down Expand Up @@ -1356,7 +1354,10 @@ impl<'a> Linker for WasmLd<'a> {
self.link_or_cc_arg(path);
}

fn link_staticlib_by_name(&mut self, name: &str, _verbatim: bool, whole_archive: bool) {
fn link_staticlib_by_name(&mut self, name: &str, verbatim: bool, whole_archive: bool) {
if let Some(path) = try_find_native_static_library(self.sess, name, verbatim) {
return self.link_staticlib_by_path(&path, whole_archive);
}
if !whole_archive {
self.link_or_cc_args(&["-l", name]);
} else {
Expand Down Expand Up @@ -1490,7 +1491,10 @@ impl<'a> Linker for L4Bender<'a> {
) {
}

fn link_staticlib_by_name(&mut self, name: &str, _verbatim: bool, whole_archive: bool) {
fn link_staticlib_by_name(&mut self, name: &str, verbatim: bool, whole_archive: bool) {
if let Some(path) = try_find_native_static_library(self.sess, name, verbatim) {
return self.link_staticlib_by_path(&path, whole_archive);
}
self.hint_static();
if !whole_archive {
self.link_arg(format!("-PC{name}"));
Expand Down Expand Up @@ -1664,12 +1668,15 @@ impl<'a> Linker for AixLinker<'a> {
}

fn link_staticlib_by_name(&mut self, name: &str, verbatim: bool, whole_archive: bool) {
if let Some(path) = try_find_native_static_library(self.sess, name, verbatim) {
return self.link_staticlib_by_path(&path, whole_archive);
}
self.hint_static();
if !whole_archive {
self.link_or_cc_arg(if verbatim { String::from(name) } else { format!("-l{name}") });
} else {
let mut arg = OsString::from("-bkeepfile:");
arg.push(find_native_static_library(name, verbatim, self.sess));
arg.push(name);
self.link_or_cc_arg(arg);
}
}
Expand Down
13 changes: 7 additions & 6 deletions compiler/rustc_metadata/src/native_libs.rs
Original file line number Diff line number Diff line change
Expand Up @@ -95,14 +95,15 @@ pub fn try_find_native_static_library(
name: &str,
verbatim: bool,
) -> Option<PathBuf> {
let target = (&*sess.target.staticlib_prefix, &*sess.target.staticlib_suffix);
let unix = ("lib", ".a");
let formats = if verbatim {
vec![("".into(), "".into())]
vec![("", "")]
} else if target != unix && sess.target.is_like_msvc {
// On Windows MSVC naming scheme `libfoo.a` is used as a fallback from default `foo.lib`.
Comment on lines +102 to +103
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 sure this is correct. To my knowledge, MSVC does not use .a at all for static libs (unless we/the Rust ecosystem have started doing this somewhere). What probably does use .a though would be *-windows-gnu.

This code originated a long time ago, far before MSVC was supported: cb823b0. My hunch is that this probably needs to either be changed to target Windows generally or *-windows-gnu more specifically.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

All built-in targets except windows-msvc already have the libname.a naming scheme in their target specs, so this fallback doesn't apply to them (due to the target != unix condition). So on windows-gnu in particular the fallback is also not necessary.

unless we/the Rust ecosystem have started doing this somewhere

Meson is using the libname.a naming scheme on MSVC and CMake now also supports it due to Meson (#123436, https://mesonbuild.com/FAQ.html#why-does-building-my-project-with-msvc-output-static-libraries-called-libfooa, https://gitlab.kitware.com/cmake/cmake/-/issues/23975).

Copy link
Member

Choose a reason for hiding this comment

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

Ok, can you mention that Meson and CMake do this in the comment?

vec![target, unix]
} else {
let os = (sess.target.staticlib_prefix.clone(), sess.target.staticlib_suffix.clone());
// On Windows, static libraries sometimes show up as libfoo.a and other
// times show up as foo.lib
let unix = ("lib".into(), ".a".into());
if os == unix { vec![os] } else { vec![os, unix] }
vec![target]
};

walk_native_lib_search_dirs(sess, None, |dir, is_framework| {
Expand Down
6 changes: 6 additions & 0 deletions src/doc/rustc/src/command-line-arguments.md
Original file line number Diff line number Diff line change
Expand Up @@ -82,6 +82,12 @@ The name used in a `link` attribute may be overridden using the form `-l
ATTR_NAME:LINK_NAME` where `ATTR_NAME` is the name in the `link` attribute,
and `LINK_NAME` is the name of the actual library that will be linked.

The compiler may attempt to search for the library in native library search directories
(controlled by `-L`), and pass it to linker by full path if the search is successful.
Otherwise the library will be passed to linker by name, so it can perform its own search.
In some cases this enables use of alternative library naming schemes or `+verbatim` modifier
even if they are not natively supported by linker.

[link-attribute]: ../reference/items/external-blocks.html#the-link-attribute

### Linking modifiers: `whole-archive`
Expand Down
4 changes: 2 additions & 2 deletions tests/run-make/native-link-modifier-bundle/rmake.rs
Original file line number Diff line number Diff line change
Expand Up @@ -68,7 +68,7 @@ fn main() {
.crate_type("cdylib")
.print("link-args")
.run()
.assert_stdout_not_contains(r#"-l[" ]*native-staticlib"#);
.assert_stdout_not_contains("libnative-staticlib.a");
llvm_nm()
.input(dynamic_lib_name("cdylib_bundled"))
.run()
Expand All @@ -81,7 +81,7 @@ fn main() {
.crate_type("cdylib")
.print("link-args")
.run()
.assert_stdout_contains_regex(r#"-l[" ]*native-staticlib"#);
.assert_stdout_contains_regex(r"libnative-staticlib.a");
llvm_nm()
.input(dynamic_lib_name("cdylib_non_bundled"))
.run()
Expand Down
3 changes: 0 additions & 3 deletions tests/run-make/native-link-modifier-verbatim-linker/rmake.rs
Original file line number Diff line number Diff line change
Expand Up @@ -3,9 +3,6 @@
// This test is the same as native-link-modifier-rustc, but without rlibs.
// See https://github.com/rust-lang/rust/issues/99425

//@ ignore-apple
// Reason: linking fails due to the unusual ".ext" staticlib name.

use run_make_support::rustc;

fn main() {
Expand Down
7 changes: 5 additions & 2 deletions tests/run-make/rlib-format-packed-bundled-libs-3/rmake.rs
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@ use run_make_support::{
//@ only-linux
// Reason: differences in the native lib compilation process causes differences
// in the --print link-args output
// FIXME: The test actually passes on windows-gnu, enable it there.

fn main() {
build_native_static_lib("native_dep_1");
Expand Down Expand Up @@ -77,8 +78,10 @@ fn main() {
.stdout_utf8();

let re = regex::Regex::new(
"--whole-archive.*native_dep_1.*--whole-archive.*lnative_dep_2.*no-whole-archive.*lnative_dep_4"
).unwrap();
"--whole-archive.*native_dep_1.*--whole-archive.*libnative_dep_2.a\
.*no-whole-archive.*libnative_dep_4.a",
)
.unwrap();

assert!(re.is_match(&out));
}
Loading