Skip to content

Commit be04b07

Browse files
committed
Support raw-dylib link kind on ELF
raw-dylib is a link kind that allows rustc to link against a library without having any library files present. This currently only exists on Windows. rustc will take all the symbols from raw-dylib link blocks and put them in an import library, where they can then be resolved by the linker. While import libraries don't exist on ELF, it would still be convenient to have this same functionality. Not having the libraries present at build-time can be convenient for several reasons, especially cross-compilation. With raw-dylib, code linking against a library can be cross-compiled without needing to have these libraries available on the build machine. If the libc crate makes use of this, it would allow cross-compilation without having any libc available on the build machine. This is not yet possible with this implementation, at least against libc's like glibc that use symbol versioning. The raw-dylib kind could be extended with support for symbol versioning in the future. This implementation is very experimental and I have not tested it very well. I have tested it for a toy example and the lz4-sys crate, where it was able to successfully link a binary despite not having a corresponding library at build-time.
1 parent 8e59cf9 commit be04b07

File tree

62 files changed

+609
-127
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

62 files changed

+609
-127
lines changed

compiler/rustc_codegen_ssa/src/back/link.rs

+270-41
Large diffs are not rendered by default.

compiler/rustc_codegen_ssa/src/back/metadata.rs

+27-69
Original file line numberDiff line numberDiff line change
@@ -9,8 +9,7 @@ use itertools::Itertools;
99
use object::write::{self, StandardSegment, Symbol, SymbolSection};
1010
use object::{
1111
Architecture, BinaryFormat, Endianness, FileFlags, Object, ObjectSection, ObjectSymbol,
12-
SectionFlags, SectionKind, SubArchitecture, SymbolFlags, SymbolKind, SymbolScope, elf, pe,
13-
xcoff,
12+
SectionFlags, SectionKind, SymbolFlags, SymbolKind, SymbolScope, elf, pe, xcoff,
1413
};
1514
use rustc_abi::Endian;
1615
use rustc_data_structures::memmap::Mmap;
@@ -206,61 +205,12 @@ pub(crate) fn create_object_file(sess: &Session) -> Option<write::Object<'static
206205
Endian::Little => Endianness::Little,
207206
Endian::Big => Endianness::Big,
208207
};
209-
let (architecture, sub_architecture) = match &sess.target.arch[..] {
210-
"arm" => (Architecture::Arm, None),
211-
"aarch64" => (
212-
if sess.target.pointer_width == 32 {
213-
Architecture::Aarch64_Ilp32
214-
} else {
215-
Architecture::Aarch64
216-
},
217-
None,
218-
),
219-
"x86" => (Architecture::I386, None),
220-
"s390x" => (Architecture::S390x, None),
221-
"mips" | "mips32r6" => (Architecture::Mips, None),
222-
"mips64" | "mips64r6" => (Architecture::Mips64, None),
223-
"x86_64" => (
224-
if sess.target.pointer_width == 32 {
225-
Architecture::X86_64_X32
226-
} else {
227-
Architecture::X86_64
228-
},
229-
None,
230-
),
231-
"powerpc" => (Architecture::PowerPc, None),
232-
"powerpc64" => (Architecture::PowerPc64, None),
233-
"riscv32" => (Architecture::Riscv32, None),
234-
"riscv64" => (Architecture::Riscv64, None),
235-
"sparc" => {
236-
if sess.unstable_target_features.contains(&sym::v8plus) {
237-
// Target uses V8+, aka EM_SPARC32PLUS, aka 64-bit V9 but in 32-bit mode
238-
(Architecture::Sparc32Plus, None)
239-
} else {
240-
// Target uses V7 or V8, aka EM_SPARC
241-
(Architecture::Sparc, None)
242-
}
243-
}
244-
"sparc64" => (Architecture::Sparc64, None),
245-
"avr" => (Architecture::Avr, None),
246-
"msp430" => (Architecture::Msp430, None),
247-
"hexagon" => (Architecture::Hexagon, None),
248-
"bpf" => (Architecture::Bpf, None),
249-
"loongarch64" => (Architecture::LoongArch64, None),
250-
"csky" => (Architecture::Csky, None),
251-
"arm64ec" => (Architecture::Aarch64, Some(SubArchitecture::Arm64EC)),
252-
// Unsupported architecture.
253-
_ => return None,
254-
};
255-
let binary_format = if sess.target.is_like_osx {
256-
BinaryFormat::MachO
257-
} else if sess.target.is_like_windows {
258-
BinaryFormat::Coff
259-
} else if sess.target.is_like_aix {
260-
BinaryFormat::Xcoff
261-
} else {
262-
BinaryFormat::Elf
208+
let Some((architecture, sub_architecture)) =
209+
sess.target.object_architecture(&sess.unstable_target_features)
210+
else {
211+
return None;
263212
};
213+
let binary_format = sess.target.binary_format();
264214

265215
let mut file = write::Object::new(binary_format, architecture, endianness);
266216
file.set_sub_architecture(sub_architecture);
@@ -300,7 +250,26 @@ pub(crate) fn create_object_file(sess: &Session) -> Option<write::Object<'static
300250

301251
file.set_mangling(original_mangling);
302252
}
303-
let e_flags = match architecture {
253+
let e_flags = elf_e_flags(architecture, sess);
254+
// adapted from LLVM's `MCELFObjectTargetWriter::getOSABI`
255+
let os_abi = elf_os_abi(sess);
256+
let abi_version = 0;
257+
add_gnu_property_note(&mut file, architecture, binary_format, endianness);
258+
file.flags = FileFlags::Elf { os_abi, abi_version, e_flags };
259+
Some(file)
260+
}
261+
262+
pub(super) fn elf_os_abi(sess: &Session) -> u8 {
263+
match sess.target.options.os.as_ref() {
264+
"hermit" => elf::ELFOSABI_STANDALONE,
265+
"freebsd" => elf::ELFOSABI_FREEBSD,
266+
"solaris" => elf::ELFOSABI_SOLARIS,
267+
_ => elf::ELFOSABI_NONE,
268+
}
269+
}
270+
271+
pub(super) fn elf_e_flags(architecture: Architecture, sess: &Session) -> u32 {
272+
match architecture {
304273
Architecture::Mips => {
305274
let arch = match sess.target.options.cpu.as_ref() {
306275
"mips1" => elf::EF_MIPS_ARCH_1,
@@ -391,18 +360,7 @@ pub(crate) fn create_object_file(sess: &Session) -> Option<write::Object<'static
391360
e_flags
392361
}
393362
_ => 0,
394-
};
395-
// adapted from LLVM's `MCELFObjectTargetWriter::getOSABI`
396-
let os_abi = match sess.target.options.os.as_ref() {
397-
"hermit" => elf::ELFOSABI_STANDALONE,
398-
"freebsd" => elf::ELFOSABI_FREEBSD,
399-
"solaris" => elf::ELFOSABI_SOLARIS,
400-
_ => elf::ELFOSABI_NONE,
401-
};
402-
let abi_version = 0;
403-
add_gnu_property_note(&mut file, architecture, binary_format, endianness);
404-
file.flags = FileFlags::Elf { os_abi, abi_version, e_flags };
405-
Some(file)
363+
}
406364
}
407365

408366
/// Mach-O files contain information about:

compiler/rustc_feature/src/unstable.rs

+2
Original file line numberDiff line numberDiff line change
@@ -594,6 +594,8 @@ declare_features! (
594594
(unstable, precise_capturing_in_traits, "1.83.0", Some(130044)),
595595
/// Allows macro attributes on expressions, statements and non-inline modules.
596596
(unstable, proc_macro_hygiene, "1.30.0", Some(54727)),
597+
/// Allows the use of raw-dylibs on ELF platforms
598+
(incomplete, raw_dylib_elf, "CURRENT_RUSTC_VERSION", Some(135694)),
597599
/// Makes `&` and `&mut` patterns eat only one layer of references in Rust 2024.
598600
(incomplete, ref_pat_eat_one_layer_2024, "1.79.0", Some(123076)),
599601
/// Makes `&` and `&mut` patterns eat only one layer of references in Rust 2024—structural variant

compiler/rustc_metadata/messages.ftl

+3
Original file line numberDiff line numberDiff line change
@@ -233,6 +233,9 @@ metadata_prev_alloc_error_handler =
233233
metadata_prev_global_alloc =
234234
previous global allocator defined here
235235
236+
metadata_raw_dylib_elf_unstable =
237+
link kind `raw-dylib` is unstable on ELF platforms
238+
236239
metadata_raw_dylib_no_nul =
237240
link name must not contain NUL characters if link kind is `raw-dylib`
238241

compiler/rustc_metadata/src/native_libs.rs

+19-2
Original file line numberDiff line numberDiff line change
@@ -17,7 +17,7 @@ use rustc_session::search_paths::PathKind;
1717
use rustc_session::utils::NativeLibKind;
1818
use rustc_span::def_id::{DefId, LOCAL_CRATE};
1919
use rustc_span::{Symbol, sym};
20-
use rustc_target::spec::LinkSelfContainedComponents;
20+
use rustc_target::spec::{BinaryFormat, LinkSelfContainedComponents};
2121

2222
use crate::{errors, fluent_generated};
2323

@@ -263,9 +263,26 @@ impl<'tcx> Collector<'tcx> {
263263
NativeLibKind::Framework { as_needed: None }
264264
}
265265
"raw-dylib" => {
266-
if !sess.target.is_like_windows {
266+
if sess.target.is_like_windows {
267+
// raw-dylib is stable and working on Windows
268+
} else if sess.target.binary_format() == BinaryFormat::Elf
269+
&& features.raw_dylib_elf()
270+
{
271+
// raw-dylib is unstable on ELF, but the user opted in
272+
} else if sess.target.binary_format() == BinaryFormat::Elf
273+
&& sess.is_nightly_build()
274+
{
275+
feature_err(
276+
sess,
277+
sym::raw_dylib_elf,
278+
span,
279+
fluent_generated::metadata_raw_dylib_elf_unstable,
280+
)
281+
.emit();
282+
} else {
267283
sess.dcx().emit_err(errors::RawDylibOnlyWindows { span });
268284
}
285+
269286
NativeLibKind::RawDylib
270287
}
271288
"link-arg" => {

compiler/rustc_session/src/utils.rs

+1
Original file line numberDiff line numberDiff line change
@@ -34,6 +34,7 @@ pub enum NativeLibKind {
3434
as_needed: Option<bool>,
3535
},
3636
/// Dynamic library (e.g. `foo.dll` on Windows) without a corresponding import library.
37+
/// On Linux, it refers to a generated shared library stub.
3738
RawDylib,
3839
/// A macOS-specific kind of dynamic libraries.
3940
Framework {

compiler/rustc_span/src/symbol.rs

+1
Original file line numberDiff line numberDiff line change
@@ -1597,6 +1597,7 @@ symbols! {
15971597
quote,
15981598
range_inclusive_new,
15991599
raw_dylib,
1600+
raw_dylib_elf,
16001601
raw_eq,
16011602
raw_identifiers,
16021603
raw_ref_op,

compiler/rustc_target/src/spec/mod.rs

+67-1
Original file line numberDiff line numberDiff line change
@@ -42,7 +42,7 @@ use std::path::{Path, PathBuf};
4242
use std::str::FromStr;
4343
use std::{fmt, io};
4444

45-
use rustc_data_structures::fx::FxHashSet;
45+
use rustc_data_structures::fx::{FxHashSet, FxIndexSet};
4646
use rustc_fs_util::try_canonicalize;
4747
use rustc_macros::{Decodable, Encodable, HashStable_Generic};
4848
use rustc_serialize::{Decodable, Decoder, Encodable, Encoder};
@@ -69,6 +69,7 @@ mod base;
6969
mod json;
7070

7171
pub use base::avr_gnu::ef_avr_arch;
72+
pub use object::BinaryFormat;
7273

7374
/// Linker is called through a C/C++ compiler.
7475
#[derive(Clone, Copy, Debug, Eq, Ord, PartialEq, PartialOrd)]
@@ -3394,6 +3395,71 @@ impl Target {
33943395
s => s.clone(),
33953396
}
33963397
}
3398+
3399+
pub fn binary_format(&self) -> object::BinaryFormat {
3400+
if self.is_like_osx {
3401+
object::BinaryFormat::MachO
3402+
} else if self.is_like_windows {
3403+
object::BinaryFormat::Coff
3404+
} else if self.is_like_aix {
3405+
object::BinaryFormat::Xcoff
3406+
} else {
3407+
object::BinaryFormat::Elf
3408+
}
3409+
}
3410+
3411+
pub fn object_architecture(
3412+
&self,
3413+
unstable_target_features: &FxIndexSet<Symbol>,
3414+
) -> Option<(object::Architecture, Option<object::SubArchitecture>)> {
3415+
use object::Architecture;
3416+
Some(match self.arch.as_ref() {
3417+
"arm" => (Architecture::Arm, None),
3418+
"aarch64" => (
3419+
if self.pointer_width == 32 {
3420+
Architecture::Aarch64_Ilp32
3421+
} else {
3422+
Architecture::Aarch64
3423+
},
3424+
None,
3425+
),
3426+
"x86" => (Architecture::I386, None),
3427+
"s390x" => (Architecture::S390x, None),
3428+
"mips" | "mips32r6" => (Architecture::Mips, None),
3429+
"mips64" | "mips64r6" => (Architecture::Mips64, None),
3430+
"x86_64" => (
3431+
if self.pointer_width == 32 {
3432+
Architecture::X86_64_X32
3433+
} else {
3434+
Architecture::X86_64
3435+
},
3436+
None,
3437+
),
3438+
"powerpc" => (Architecture::PowerPc, None),
3439+
"powerpc64" => (Architecture::PowerPc64, None),
3440+
"riscv32" => (Architecture::Riscv32, None),
3441+
"riscv64" => (Architecture::Riscv64, None),
3442+
"sparc" => {
3443+
if unstable_target_features.contains(&sym::v8plus) {
3444+
// Target uses V8+, aka EM_SPARC32PLUS, aka 64-bit V9 but in 32-bit mode
3445+
(Architecture::Sparc32Plus, None)
3446+
} else {
3447+
// Target uses V7 or V8, aka EM_SPARC
3448+
(Architecture::Sparc, None)
3449+
}
3450+
}
3451+
"sparc64" => (Architecture::Sparc64, None),
3452+
"avr" => (Architecture::Avr, None),
3453+
"msp430" => (Architecture::Msp430, None),
3454+
"hexagon" => (Architecture::Hexagon, None),
3455+
"bpf" => (Architecture::Bpf, None),
3456+
"loongarch64" => (Architecture::LoongArch64, None),
3457+
"csky" => (Architecture::Csky, None),
3458+
"arm64ec" => (Architecture::Aarch64, Some(object::SubArchitecture::Arm64EC)),
3459+
// Unsupported architecture.
3460+
_ => return None,
3461+
})
3462+
}
33973463
}
33983464

33993465
/// Either a target tuple string or a path to a JSON file.

src/tools/compiletest/src/directive-list.rs

+2
Original file line numberDiff line numberDiff line change
@@ -48,6 +48,7 @@ const KNOWN_DIRECTIVE_NAMES: &[&str] = &[
4848
"ignore-coverage-run",
4949
"ignore-cross-compile",
5050
"ignore-eabi",
51+
"ignore-elf",
5152
"ignore-emscripten",
5253
"ignore-endian-big",
5354
"ignore-enzyme",
@@ -175,6 +176,7 @@ const KNOWN_DIRECTIVE_NAMES: &[&str] = &[
175176
"only-beta",
176177
"only-bpf",
177178
"only-cdb",
179+
"only-elf",
178180
"only-gnu",
179181
"only-i686-pc-windows-gnu",
180182
"only-i686-pc-windows-msvc",

src/tools/compiletest/src/header/cfg.rs

+9
Original file line numberDiff line numberDiff line change
@@ -166,6 +166,15 @@ fn parse_cfg_name_directive<'a>(
166166
message: "when the target vendor is Apple"
167167
}
168168

169+
condition! {
170+
name: "elf",
171+
condition: !config.target.contains("windows")
172+
&& !config.target.contains("apple")
173+
&& !config.target.contains("aix")
174+
&& !config.target.contains("uefi"),
175+
message: "when the target binary format is ELF"
176+
}
177+
169178
condition! {
170179
name: "enzyme",
171180
condition: config.has_enzyme,
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
int this_is_a_library_function() {
2+
return 42;
3+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
#![feature(raw_dylib_elf)]
2+
#![allow(incomplete_features)]
3+
4+
#[link(name = "liblibrary.so.1", kind = "raw-dylib", modifiers = "+verbatim")]
5+
unsafe extern "C" {
6+
safe fn this_is_a_library_function() -> core::ffi::c_int;
7+
}
8+
9+
fn main() {
10+
println!("{}", this_is_a_library_function())
11+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
42
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,26 @@
1+
//@ only-elf
2+
3+
//! Ensure ELF raw-dylib is able to link against the verbatim versioned library
4+
//! without it being present, and then be executed against this library.
5+
6+
use run_make_support::{build_native_dynamic_lib, cwd, diff, rfs, run, rustc};
7+
8+
fn main() {
9+
// We compile the binary without having the library present.
10+
// We also set the rpath to the current directory so we can pick up the library at runtime.
11+
rustc()
12+
.crate_type("bin")
13+
.input("main.rs")
14+
.arg(&format!("-Wl,-rpath={}", cwd().display()))
15+
.run();
16+
17+
// Now, *after* building the binary, we build the library...
18+
build_native_dynamic_lib("library");
19+
// ... rename it to have the versioned library name...
20+
rfs::rename("liblibrary.so", "liblibrary.so.1");
21+
22+
// ... and run with this library, ensuring it was linked correctly at runtime.
23+
let output = run("main").stdout_utf8();
24+
25+
diff().expected_file("output.txt").actual_text("actual", output).run();
26+
}
+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
int this_is_a_library_function() {
2+
return 42;
3+
}

tests/run-make/raw-dylib-elf/main.rs

+11
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
#![feature(raw_dylib_elf)]
2+
#![allow(incomplete_features)]
3+
4+
#[link(name = "library", kind = "raw-dylib")]
5+
unsafe extern "C" {
6+
safe fn this_is_a_library_function() -> core::ffi::c_int;
7+
}
8+
9+
fn main() {
10+
println!("{}", this_is_a_library_function())
11+
}
+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
42

0 commit comments

Comments
 (0)