about summary refs log tree commit diff
path: root/src/test
diff options
context:
space:
mode:
authorMark Rousskov <mark.simulacrum@gmail.com>2021-08-30 16:10:19 -0400
committerMark Rousskov <mark.simulacrum@gmail.com>2021-08-30 16:22:53 -0400
commit4c7c97a2087fc66e6a6d105b01877b5407a8a7cd (patch)
tree9865fc2c2660883b93b32d13a0514a0f5af5c882 /src/test
parent4a6547cca6d2b2f465f01331927855734687b527 (diff)
downloadrust-4c7c97a2087fc66e6a6d105b01877b5407a8a7cd.tar.gz
rust-4c7c97a2087fc66e6a6d105b01877b5407a8a7cd.zip
Fix loading large rlibs
Bumps object crate to permit parsing archives with 64-bit table entries. These
are primarily encountered when there's more than 4GB of archive data.
Diffstat (limited to 'src/test')
-rw-r--r--src/test/run-make/issue-88351-large-rlib/Makefile7
-rw-r--r--src/test/run-make/issue-88351-large-rlib/bar.rs5
-rw-r--r--src/test/run-make/issue-88351-large-rlib/main.rs64
3 files changed, 76 insertions, 0 deletions
diff --git a/src/test/run-make/issue-88351-large-rlib/Makefile b/src/test/run-make/issue-88351-large-rlib/Makefile
new file mode 100644
index 00000000000..98938e5ebc4
--- /dev/null
+++ b/src/test/run-make/issue-88351-large-rlib/Makefile
@@ -0,0 +1,7 @@
+-include ../../run-make-fulldeps/tools.mk
+
+all:
+	$(RUSTC) main.rs
+	$(TMPDIR)/main $(TMPDIR)
+	$(RUSTC) $(TMPDIR)/foo.rs --crate-type=rlib -l static=foo -L$(TMPDIR)
+	RUSTC_LOG=rustc_metadata=debug $(RUSTC) bar.rs --extern foo=$(TMPDIR)/libfoo.rlib --edition=2018
diff --git a/src/test/run-make/issue-88351-large-rlib/bar.rs b/src/test/run-make/issue-88351-large-rlib/bar.rs
new file mode 100644
index 00000000000..1b600036490
--- /dev/null
+++ b/src/test/run-make/issue-88351-large-rlib/bar.rs
@@ -0,0 +1,5 @@
+fn main() {
+    unsafe {
+        println!("{}", foo::FOO_11_49[0]);
+    }
+}
diff --git a/src/test/run-make/issue-88351-large-rlib/main.rs b/src/test/run-make/issue-88351-large-rlib/main.rs
new file mode 100644
index 00000000000..f97830f3d86
--- /dev/null
+++ b/src/test/run-make/issue-88351-large-rlib/main.rs
@@ -0,0 +1,64 @@
+//! Large archive example.
+//!
+//! This creates several C files with a bunch of global arrays. The goal is to
+//! create an rlib that is over 4GB in size so that the LLVM archiver creates
+//! a /SYM64/ entry instead of /.
+//!
+//! It compiles the C files to .o files, and then uses `ar` to collect them
+//! into a static library. It creates `foo.rs` with references to all the C
+//! arrays, and then uses `rustc` to build an rlib with that static
+//! information. It then creates `bar.rs` which links the giant libfoo.rlib,
+//! which should fail since it can't read the large libfoo.rlib file.
+
+use std::env;
+use std::fs::File;
+use std::io::{BufWriter, Write};
+use std::process::Command;
+
+// Number of object files to create.
+const NOBJ: u32 = 12;
+// Make the filename longer than 16 characters to force names to be placed in //
+const PREFIX: &str = "abcdefghijklmnopqrstuvwxyz";
+
+fn main() {
+    let tmpdir = std::path::PathBuf::from(env::args_os().nth(1).unwrap());
+    let mut foo_rs = File::create(tmpdir.join("foo.rs")).unwrap();
+    write!(foo_rs, "extern \"C\" {{\n").unwrap();
+    for obj in 0..NOBJ {
+        let filename = tmpdir.join(&format!("{}{}.c", PREFIX, obj));
+        let f = File::create(&filename).unwrap();
+        let mut buf = BufWriter::new(f);
+        write!(buf, "#include<stdint.h>\n").unwrap();
+        for n in 0..50 {
+            write!(buf, "int64_t FOO_{}_{}[] = {{\n", obj, n).unwrap();
+            for x in 0..1024 {
+                for y in 0..1024 {
+                    write!(buf, "{},", (obj + n + x + y) % 10).unwrap();
+                }
+                write!(buf, "\n").unwrap();
+            }
+            write!(buf, "}};\n").unwrap();
+            write!(foo_rs, "    pub static FOO_{}_{}: [i64; 1024*1024];\n", obj, n).unwrap();
+        }
+        drop(buf);
+        println!("compile {:?}", filename);
+        let status =
+            Command::new("cc").current_dir(&tmpdir).arg("-c").arg(&filename).status().unwrap();
+        if !status.success() {
+            panic!("failed: {:?}", status);
+        }
+    }
+    write!(foo_rs, "}}\n").unwrap();
+    drop(foo_rs);
+    let mut cmd = Command::new("ar");
+    cmd.arg("-crs");
+    cmd.arg(tmpdir.join("libfoo.a"));
+    for obj in 0..NOBJ {
+        cmd.arg(tmpdir.join(&format!("{}{}.o", PREFIX, obj)));
+    }
+    println!("archiving: {:?}", cmd);
+    let status = cmd.status().unwrap();
+    if !status.success() {
+        panic!("failed: {:?}", status);
+    }
+}