Skip to content

Commit 940a109

Browse files
committed
add test that tracks that functions defined in extern blocks are not exported
this maybe should change, but now at least it won't be changed unintentionally
1 parent 9061915 commit 940a109

File tree

2 files changed

+62
-32
lines changed

2 files changed

+62
-32
lines changed

tests/run-make/naked-symbol-visibility/a_rust_dylib.rs

+13
Original file line numberDiff line numberDiff line change
@@ -72,3 +72,16 @@ extern "C" fn naked_weak_linkage() -> u32 {
7272
extern "C" fn naked_external_linkage() -> u32 {
7373
unsafe { asm!("mov rax, 42", "ret", options(noreturn)) }
7474
}
75+
76+
// functions that are declared in an `extern "C"` block are currently not exported
77+
// this maybe should change in the future, this is just tracking the current behavior
78+
// reported in https://github.com/rust-lang/rust/issues/128071
79+
std::arch::global_asm! {
80+
".globl function_defined_in_global_asm",
81+
"function_defined_in_global_asm:",
82+
"ret",
83+
}
84+
85+
extern "C" {
86+
pub fn function_defined_in_global_asm();
87+
}
Original file line numberDiff line numberDiff line change
@@ -1,63 +1,80 @@
11
// @only-x86_64
2-
use run_make_support::{dynamic_lib_name, llvm_readobj, regex, rustc};
2+
use run_make_support::object::read::{File, Object, Symbol};
3+
use run_make_support::object::ObjectSymbol;
4+
use run_make_support::{dynamic_lib_name, rfs, rustc};
35

46
fn main() {
57
let rdylib_name = dynamic_lib_name("a_rust_dylib");
68
rustc().arg("-Zshare-generics=no").input("a_rust_dylib.rs").run();
79

10+
let binary_data = rfs::read(&rdylib_name);
11+
let rdylib = File::parse(&*binary_data).unwrap();
12+
813
// check vanilla symbols
9-
not_exported(&rdylib_name, "private_vanilla_rust_function_from_rust_dylib");
10-
global_function(&rdylib_name, "public_vanilla_rust_function_from_rust_dylib");
11-
not_exported(&rdylib_name, "public_vanilla_generic_function_from_rust_dylib");
14+
not_exported(&rdylib, "private_vanilla_rust_function_from_rust_dylib");
15+
global_function(&rdylib, "public_vanilla_rust_function_from_rust_dylib");
16+
not_exported(&rdylib, "public_vanilla_generic_function_from_rust_dylib");
1217

13-
weak_function(&rdylib_name, "vanilla_weak_linkage");
14-
global_function(&rdylib_name, "vanilla_external_linkage");
18+
weak_function(&rdylib, "vanilla_weak_linkage");
19+
global_function(&rdylib, "vanilla_external_linkage");
1520

1621
// naked should mirror vanilla
17-
not_exported(&rdylib_name, "private_naked_rust_function_from_rust_dylib");
18-
global_function(&rdylib_name, "public_naked_rust_function_from_rust_dylib");
19-
not_exported(&rdylib_name, "public_naked_generic_function_from_rust_dylib");
22+
not_exported(&rdylib, "private_naked_rust_function_from_rust_dylib");
23+
global_function(&rdylib, "public_naked_rust_function_from_rust_dylib");
24+
not_exported(&rdylib, "public_naked_generic_function_from_rust_dylib");
25+
26+
weak_function(&rdylib, "naked_weak_linkage");
27+
global_function(&rdylib, "naked_external_linkage");
2028

21-
weak_function(&rdylib_name, "naked_weak_linkage");
22-
global_function(&rdylib_name, "naked_external_linkage");
29+
// functions that are declared in an `extern "C"` block are currently not exported
30+
// this maybe should change in the future, this is just tracking the current behavior
31+
// reported in https://github.com/rust-lang/rust/issues/128071
32+
not_exported(&rdylib, "function_defined_in_global_asm");
2333

2434
// share generics should expose the generic functions
2535
rustc().arg("-Zshare-generics=yes").input("a_rust_dylib.rs").run();
26-
global_function(&rdylib_name, "public_vanilla_generic_function_from_rust_dylib");
27-
global_function(&rdylib_name, "public_naked_generic_function_from_rust_dylib");
36+
let binary_data = rfs::read(&rdylib_name);
37+
let rdylib = File::parse(&*binary_data).unwrap();
38+
39+
global_function(&rdylib, "public_vanilla_generic_function_from_rust_dylib");
40+
global_function(&rdylib, "public_naked_generic_function_from_rust_dylib");
2841
}
2942

3043
#[track_caller]
31-
fn global_function(path: &str, symbol_name: &str) {
32-
let lines = find_dynamic_symbol(path, symbol_name);
33-
let [line] = lines.as_slice() else {
34-
panic!("symbol {symbol_name} occurs {} times", lines.len())
44+
fn global_function(file: &File, symbol_name: &str) {
45+
let symbols = find_dynamic_symbol(file, symbol_name);
46+
let [symbol] = symbols.as_slice() else {
47+
panic!("symbol {symbol_name} occurs {} times", symbols.len())
3548
};
3649

37-
assert!(line.contains("FUNC"), "`{symbol_name}` is not a function");
38-
assert!(line.contains("GLOBAL"), "`{symbol_name}` is not marked as global");
50+
assert!(symbol.is_definition(), "`{symbol_name}` is not a function");
51+
assert!(symbol.is_global(), "`{symbol_name}` is not marked as global");
3952
}
4053

4154
#[track_caller]
42-
fn weak_function(path: &str, symbol_name: &str) {
43-
let lines = find_dynamic_symbol(path, symbol_name);
44-
let [line] = lines.as_slice() else {
45-
panic!("symbol {symbol_name} occurs {} times", lines.len())
55+
fn weak_function(file: &File, symbol_name: &str) {
56+
let symbols = find_dynamic_symbol(file, symbol_name);
57+
let [symbol] = symbols.as_slice() else {
58+
panic!("symbol {symbol_name} occurs {} times", symbols.len())
4659
};
4760

48-
assert!(line.contains("FUNC"), "`{symbol_name}` is not a function");
49-
assert!(line.contains("WEAK"), "`{symbol_name}` is not marked as weak");
61+
assert!(symbol.is_definition(), "`{symbol_name}` is not a function");
62+
assert!(symbol.is_weak(), "`{symbol_name}` is not marked as weak");
5063
}
5164

5265
#[track_caller]
53-
fn not_exported(path: &str, symbol_name: &str) {
54-
assert_eq!(find_dynamic_symbol(path, symbol_name).len(), 0)
66+
fn not_exported(file: &File, symbol_name: &str) {
67+
assert_eq!(find_dynamic_symbol(file, symbol_name).len(), 0)
5568
}
5669

57-
fn find_dynamic_symbol<'a>(path: &str, symbol_name: &str) -> Vec<String> {
58-
let out = llvm_readobj().arg("--dyn-symbols").input(path).run().stdout_utf8();
59-
out.lines()
60-
.filter(|&line| !line.contains("__imp_") && line.contains(symbol_name))
61-
.map(|line| line.to_string())
70+
fn find_dynamic_symbol<'file, 'data>(
71+
file: &'file File<'data>,
72+
expected: &str,
73+
) -> Vec<Symbol<'data, 'file>> {
74+
file.dynamic_symbols()
75+
.filter(|symbol| {
76+
let name = symbol.name().unwrap();
77+
!name.contains("__imp_") && name.contains(expected)
78+
})
6279
.collect()
6380
}

0 commit comments

Comments
 (0)