|
| 1 | +use run_make_support::rfs::{create_dir, remove_file, rename}; |
| 2 | +use run_make_support::{Rustc, dynamic_lib_name, run_in_tmpdir, rust_lib_name, rustc}; |
| 3 | + |
| 4 | +#[derive(Debug, Copy, Clone)] |
| 5 | +enum LibraryKind { |
| 6 | + Rlib, |
| 7 | + Dylib, |
| 8 | +} |
| 9 | + |
| 10 | +impl LibraryKind { |
| 11 | + fn crate_type(&self) -> &str { |
| 12 | + match self { |
| 13 | + LibraryKind::Rlib => "rlib", |
| 14 | + LibraryKind::Dylib => "dylib", |
| 15 | + } |
| 16 | + } |
| 17 | + |
| 18 | + fn add_extern(&self, rustc: &mut Rustc, dep_name: &str, dep_path: &str) { |
| 19 | + let dep_path = match self { |
| 20 | + LibraryKind::Dylib => format!("{dep_path}/{}", dynamic_lib_name(dep_name)), |
| 21 | + LibraryKind::Rlib => format!("{dep_path}/{}", rust_lib_name(dep_name)), |
| 22 | + }; |
| 23 | + rustc.extern_(dep_name, dep_path); |
| 24 | + } |
| 25 | +} |
| 26 | + |
| 27 | +fn main() { |
| 28 | + // The compiler takes different paths based on if --extern is passed or not, so we test all |
| 29 | + // combinations (`rlib`/`dylib` x `--extern`/`no --extern`). |
| 30 | + for kind in [LibraryKind::Rlib, LibraryKind::Dylib] { |
| 31 | + eprintln!("Testing library kind {kind:?}"); |
| 32 | + lookup_rmeta_in_lib_dir(kind); |
| 33 | + lookup_rmeta_in_lib_search_path(kind); |
| 34 | + lookup_rmeta_in_lib_dir_extern(kind); |
| 35 | + lookup_rmeta_in_lib_search_path_extern(kind); |
| 36 | + } |
| 37 | +} |
| 38 | + |
| 39 | +// Lookup .rmeta file in the same directory as a rlib/dylib with stub metadata. |
| 40 | +fn lookup_rmeta_in_lib_dir(kind: LibraryKind) { |
| 41 | + run_in_tmpdir(|| { |
| 42 | + build_dep_rustc(kind).run(); |
| 43 | + rustc().input("foo.rs").run(); |
| 44 | + }); |
| 45 | +} |
| 46 | + |
| 47 | +// Lookup .rmeta file in a directory passed to -L. |
| 48 | +fn lookup_rmeta_in_lib_search_path(kind: LibraryKind) { |
| 49 | + run_in_tmpdir(|| { |
| 50 | + build_dep_rustc(kind).run(); |
| 51 | + |
| 52 | + // Move the rmeta directory to a different path |
| 53 | + create_dir("rmeta-dir"); |
| 54 | + rename("libdep1.rmeta", "rmeta-dir/libdep1.rmeta"); |
| 55 | + |
| 56 | + rustc().input("foo.rs").library_search_path("rmeta-dir").run(); |
| 57 | + }); |
| 58 | +} |
| 59 | + |
| 60 | +// Lookup .rmeta file in the same directory as a rlib/dylib with stub metadata when specifying the |
| 61 | +// dependency using --extern. |
| 62 | +fn lookup_rmeta_in_lib_dir_extern(kind: LibraryKind) { |
| 63 | + run_in_tmpdir(|| { |
| 64 | + // Generate libdep1.rlib and libdep1.rmeta in deps |
| 65 | + create_dir("deps"); |
| 66 | + build_dep_rustc(kind).out_dir("deps").run(); |
| 67 | + |
| 68 | + let mut rustc = rustc(); |
| 69 | + kind.add_extern(&mut rustc, "dep1", "deps"); |
| 70 | + rustc.input("foo.rs").run(); |
| 71 | + }); |
| 72 | +} |
| 73 | + |
| 74 | +// Lookup .rmeta file in a directory passed to -L when specifying the dependency using --extern. |
| 75 | +fn lookup_rmeta_in_lib_search_path_extern(kind: LibraryKind) { |
| 76 | + run_in_tmpdir(|| { |
| 77 | + create_dir("deps"); |
| 78 | + build_dep_rustc(kind).out_dir("deps").run(); |
| 79 | + |
| 80 | + // Move the rmeta directory to a different dir |
| 81 | + create_dir("rmeta-dir"); |
| 82 | + rename("deps/libdep1.rmeta", "rmeta-dir/libdep1.rmeta"); |
| 83 | + |
| 84 | + let mut rustc = rustc(); |
| 85 | + kind.add_extern(&mut rustc, "dep1", "deps"); |
| 86 | + rustc.library_search_path("rmeta-dir"); |
| 87 | + rustc.input("foo.rs").run(); |
| 88 | + }); |
| 89 | +} |
| 90 | + |
| 91 | +fn build_dep_rustc(kind: LibraryKind) -> Rustc { |
| 92 | + let mut dep_rustc = rustc(); |
| 93 | + dep_rustc |
| 94 | + .arg("-Zsplit-metadata") |
| 95 | + .crate_type(kind.crate_type()) |
| 96 | + .input("dep1.rs") |
| 97 | + .emit("metadata,link"); |
| 98 | + if matches!(kind, LibraryKind::Dylib) { |
| 99 | + dep_rustc.arg("-Cprefer-dynamic"); |
| 100 | + } |
| 101 | + dep_rustc |
| 102 | +} |
0 commit comments