summaryrefslogtreecommitdiffstats
path: root/third_party/rust/object/tests/round_trip/elf.rs
diff options
context:
space:
mode:
authorDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-07 09:22:09 +0000
committerDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-07 09:22:09 +0000
commit43a97878ce14b72f0981164f87f2e35e14151312 (patch)
tree620249daf56c0258faa40cbdcf9cfba06de2a846 /third_party/rust/object/tests/round_trip/elf.rs
parentInitial commit. (diff)
downloadfirefox-43a97878ce14b72f0981164f87f2e35e14151312.tar.xz
firefox-43a97878ce14b72f0981164f87f2e35e14151312.zip
Adding upstream version 110.0.1.upstream/110.0.1upstream
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'third_party/rust/object/tests/round_trip/elf.rs')
-rw-r--r--third_party/rust/object/tests/round_trip/elf.rs218
1 files changed, 218 insertions, 0 deletions
diff --git a/third_party/rust/object/tests/round_trip/elf.rs b/third_party/rust/object/tests/round_trip/elf.rs
new file mode 100644
index 0000000000..048db5795a
--- /dev/null
+++ b/third_party/rust/object/tests/round_trip/elf.rs
@@ -0,0 +1,218 @@
+use object::read::elf::{FileHeader, SectionHeader};
+use object::read::{Object, ObjectSymbol};
+use object::{
+ elf, read, write, Architecture, BinaryFormat, Endianness, LittleEndian, SectionIndex,
+ SectionKind, SymbolFlags, SymbolKind, SymbolScope, SymbolSection, U32,
+};
+use std::io::Write;
+
+#[test]
+fn symtab_shndx() {
+ let mut object =
+ write::Object::new(BinaryFormat::Elf, Architecture::X86_64, Endianness::Little);
+
+ for i in 0..0x10000 {
+ let name = format!("func{}", i).into_bytes();
+ let (section, offset) =
+ object.add_subsection(write::StandardSection::Text, &name, &[0xcc], 1);
+ object.add_symbol(write::Symbol {
+ name,
+ value: offset,
+ size: 1,
+ kind: SymbolKind::Text,
+ scope: SymbolScope::Linkage,
+ weak: false,
+ section: write::SymbolSection::Section(section),
+ flags: SymbolFlags::None,
+ });
+ }
+ let bytes = object.write().unwrap();
+
+ //std::fs::write(&"symtab_shndx.o", &bytes).unwrap();
+
+ let object = read::File::parse(&*bytes).unwrap();
+ assert_eq!(object.format(), BinaryFormat::Elf);
+ assert_eq!(object.architecture(), Architecture::X86_64);
+
+ for symbol in object.symbols().skip(1) {
+ assert_eq!(
+ symbol.section(),
+ SymbolSection::Section(SectionIndex(symbol.index().0))
+ );
+ }
+}
+
+#[cfg(feature = "compression")]
+#[test]
+fn compression_zlib() {
+ use object::read::ObjectSection;
+ use object::LittleEndian as LE;
+
+ let data = b"test data data data";
+ let len = data.len() as u64;
+
+ let mut ch = object::elf::CompressionHeader64::<LE>::default();
+ ch.ch_type.set(LE, object::elf::ELFCOMPRESS_ZLIB);
+ ch.ch_size.set(LE, len);
+ ch.ch_addralign.set(LE, 1);
+
+ let mut buf = Vec::new();
+ buf.write(object::bytes_of(&ch)).unwrap();
+ let mut encoder = flate2::write::ZlibEncoder::new(buf, flate2::Compression::default());
+ encoder.write_all(data).unwrap();
+ let compressed = encoder.finish().unwrap();
+
+ let mut object =
+ write::Object::new(BinaryFormat::Elf, Architecture::X86_64, Endianness::Little);
+ let section = object.add_section(
+ Vec::new(),
+ b".debug_info".to_vec(),
+ object::SectionKind::Other,
+ );
+ object.section_mut(section).set_data(compressed, 1);
+ object.section_mut(section).flags = object::SectionFlags::Elf {
+ sh_flags: object::elf::SHF_COMPRESSED.into(),
+ };
+ let bytes = object.write().unwrap();
+
+ //std::fs::write(&"compression.o", &bytes).unwrap();
+
+ let object = read::File::parse(&*bytes).unwrap();
+ assert_eq!(object.format(), BinaryFormat::Elf);
+ assert_eq!(object.architecture(), Architecture::X86_64);
+
+ let section = object.section_by_name(".debug_info").unwrap();
+ let uncompressed = section.uncompressed_data().unwrap();
+ assert_eq!(data, &*uncompressed);
+}
+
+#[cfg(feature = "compression")]
+#[test]
+fn compression_gnu() {
+ use object::read::ObjectSection;
+ use std::io::Write;
+
+ let data = b"test data data data";
+ let len = data.len() as u32;
+
+ let mut buf = Vec::new();
+ buf.write_all(b"ZLIB\0\0\0\0").unwrap();
+ buf.write_all(&len.to_be_bytes()).unwrap();
+ let mut encoder = flate2::write::ZlibEncoder::new(buf, flate2::Compression::default());
+ encoder.write_all(data).unwrap();
+ let compressed = encoder.finish().unwrap();
+
+ let mut object =
+ write::Object::new(BinaryFormat::Elf, Architecture::X86_64, Endianness::Little);
+ let section = object.add_section(
+ Vec::new(),
+ b".zdebug_info".to_vec(),
+ object::SectionKind::Other,
+ );
+ object.section_mut(section).set_data(compressed, 1);
+ let bytes = object.write().unwrap();
+
+ //std::fs::write(&"compression.o", &bytes).unwrap();
+
+ let object = read::File::parse(&*bytes).unwrap();
+ assert_eq!(object.format(), BinaryFormat::Elf);
+ assert_eq!(object.architecture(), Architecture::X86_64);
+
+ let section = object.section_by_name(".zdebug_info").unwrap();
+ let uncompressed = section.uncompressed_data().unwrap();
+ assert_eq!(data, &*uncompressed);
+}
+
+#[test]
+fn note() {
+ let endian = Endianness::Little;
+ let mut object = write::Object::new(BinaryFormat::Elf, Architecture::X86_64, endian);
+
+ // Add note section with align = 4.
+ let mut buffer = Vec::new();
+
+ buffer
+ .write(object::bytes_of(&elf::NoteHeader32 {
+ n_namesz: U32::new(endian, 6),
+ n_descsz: U32::new(endian, 11),
+ n_type: U32::new(endian, 1),
+ }))
+ .unwrap();
+ buffer.write(b"name1\0\0\0").unwrap();
+ buffer.write(b"descriptor\0\0").unwrap();
+
+ buffer
+ .write(object::bytes_of(&elf::NoteHeader32 {
+ n_namesz: U32::new(endian, 6),
+ n_descsz: U32::new(endian, 11),
+ n_type: U32::new(endian, 2),
+ }))
+ .unwrap();
+ buffer.write(b"name2\0\0\0").unwrap();
+ buffer.write(b"descriptor\0\0").unwrap();
+
+ let section = object.add_section(Vec::new(), b".note4".to_vec(), SectionKind::Note);
+ object.section_mut(section).set_data(buffer, 4);
+
+ // Add note section with align = 8.
+ let mut buffer = Vec::new();
+
+ buffer
+ .write(object::bytes_of(&elf::NoteHeader32 {
+ n_namesz: U32::new(endian, 6),
+ n_descsz: U32::new(endian, 11),
+ n_type: U32::new(endian, 1),
+ }))
+ .unwrap();
+ buffer.write(b"name1\0\0\0\0\0\0\0").unwrap();
+ buffer.write(b"descriptor\0\0\0\0\0\0").unwrap();
+
+ buffer
+ .write(object::bytes_of(&elf::NoteHeader32 {
+ n_namesz: U32::new(endian, 4),
+ n_descsz: U32::new(endian, 11),
+ n_type: U32::new(endian, 2),
+ }))
+ .unwrap();
+ buffer.write(b"abc\0").unwrap();
+ buffer.write(b"descriptor\0\0\0\0\0\0").unwrap();
+
+ let section = object.add_section(Vec::new(), b".note8".to_vec(), SectionKind::Note);
+ object.section_mut(section).set_data(buffer, 8);
+
+ let bytes = &*object.write().unwrap();
+
+ //std::fs::write(&"note.o", &bytes).unwrap();
+
+ let header = elf::FileHeader64::parse(bytes).unwrap();
+ let endian: LittleEndian = header.endian().unwrap();
+ let sections = header.sections(endian, bytes).unwrap();
+
+ let section = sections.section(SectionIndex(1)).unwrap();
+ assert_eq!(sections.section_name(endian, section).unwrap(), b".note4");
+ assert_eq!(section.sh_addralign(endian), 4);
+ let mut notes = section.notes(endian, bytes).unwrap().unwrap();
+ let note = notes.next().unwrap().unwrap();
+ assert_eq!(note.name(), b"name1");
+ assert_eq!(note.desc(), b"descriptor\0");
+ assert_eq!(note.n_type(endian), 1);
+ let note = notes.next().unwrap().unwrap();
+ assert_eq!(note.name(), b"name2");
+ assert_eq!(note.desc(), b"descriptor\0");
+ assert_eq!(note.n_type(endian), 2);
+ assert!(notes.next().unwrap().is_none());
+
+ let section = sections.section(SectionIndex(2)).unwrap();
+ assert_eq!(sections.section_name(endian, section).unwrap(), b".note8");
+ assert_eq!(section.sh_addralign(endian), 8);
+ let mut notes = section.notes(endian, bytes).unwrap().unwrap();
+ let note = notes.next().unwrap().unwrap();
+ assert_eq!(note.name(), b"name1");
+ assert_eq!(note.desc(), b"descriptor\0");
+ assert_eq!(note.n_type(endian), 1);
+ let note = notes.next().unwrap().unwrap();
+ assert_eq!(note.name(), b"abc");
+ assert_eq!(note.desc(), b"descriptor\0");
+ assert_eq!(note.n_type(endian), 2);
+ assert!(notes.next().unwrap().is_none());
+}