From df739a4e471d2e8df80598ed06a7f642eb934ee4 Mon Sep 17 00:00:00 2001 From: "L. E. Segovia" Date: Wed, 3 Apr 2024 20:08:56 -0300 Subject: [PATCH] MsvcLinker: allow linking dynamically to Meson and MinGW-style named libraries Fixes #122455 --- compiler/rustc_codegen_ssa/src/back/linker.rs | 50 ++++++++++++++++++- 1 file changed, 48 insertions(+), 2 deletions(-) diff --git a/compiler/rustc_codegen_ssa/src/back/linker.rs b/compiler/rustc_codegen_ssa/src/back/linker.rs index 174acb7fb277..f91517144064 100644 --- a/compiler/rustc_codegen_ssa/src/back/linker.rs +++ b/compiler/rustc_codegen_ssa/src/back/linker.rs @@ -796,6 +796,43 @@ pub struct MsvcLinker<'a> { sess: &'a Session, } +impl MsvcLinker<'_> { + // FIXME this duplicates rustc_metadata::find_native_static_library, + // as the Meson/MinGW suffix for import libraries can differ + fn find_native_dynamic_library( + name: &str, + verbatim: bool, + search_paths: &[PathBuf], + sess: &Session, + ) -> OsString { + let formats = if verbatim { + vec![("".into(), "".into())] + } else { + // While the official naming convention for MSVC import libraries + // is foo.lib... + let os = (sess.target.staticlib_prefix.clone(), sess.target.staticlib_suffix.clone()); + // ... Meson follows the libfoo.dll.a convention to + // disambiguate .a for static libraries + let meson = ("lib".into(), ".dll.a".into()); + // and MinGW uses .a altogether + let mingw = ("lib".into(), ".a".into()); + vec![os, meson, mingw] + }; + + for path in search_paths { + for (prefix, suffix) in &formats { + let test = path.join(format!("{prefix}{name}{suffix}")); + if test.exists() { + return OsString::from(test); + } + } + } + + // Allow the linker to find CRT libs itself + OsString::from(format!("{}{}", name, if verbatim { "" } else { ".lib" })) + } +} + impl<'a> Linker for MsvcLinker<'a> { fn cmd(&mut self) -> &mut Command { &mut self.cmd @@ -819,8 +856,17 @@ impl<'a> Linker for MsvcLinker<'a> { } } - fn link_dylib_by_name(&mut self, name: &str, verbatim: bool, _search_paths: &SearchPaths, _as_needed: bool) { - self.cmd.arg(format!("{}{}", name, if verbatim { "" } else { ".lib" })); + fn link_dylib_by_name( + &mut self, + name: &str, + verbatim: bool, + search_paths: &SearchPaths, + _as_needed: bool, + ) { + let search_paths = search_paths.get(self.sess); + let path = + MsvcLinker::<'a>::find_native_dynamic_library(name, verbatim, search_paths, self.sess); + self.cmd.arg(path); } fn link_staticlib_by_name(