elf/lib.rs
1//! The `elf` crate provides a pure-safe-rust interface for reading ELF object files.
2//!
3//! # Capabilities
4//!
5//! ### ✨ Works in `no_std` environments ✨
6//! This crate provides an elf parsing interface which does not allocate or use any std
7//! features, so it can be used in `no_std` environments such as kernels and bootloaders.
8//! The no_std variant merely disables the additional stream-oriented `std:: Read + Seek` interface.
9//! All core parsing functionality is the same!
10//!
11//! ### ✨ Endian-aware ✨
12//! This crate handles translating between file and host endianness when
13//! parsing the ELF contents and provides four endian parsing implementations
14//! optimized to support the different common use-cases for an ELF parsing library.
15//! Parsing is generic across the specifications and each trait impl represents a
16//! specification that encapsulates an interface for parsing integers from some
17//! set of allowed byte orderings.
18//!
19//! * [AnyEndian](endian::AnyEndian): Dynamically parsing either byte order at runtime based on the type of ELF object being parsed.
20//! * [BigEndian](endian::BigEndian)/[LittleEndian](endian::LittleEndian): For tools that know they only want to parse a single given byte order known at compile time.
21//! * [NativeEndian](type@endian::NativeEndian): For tools that know they want to parse the same byte order as the compilation target's byte order.
22//!
23//! When the limited specifications are used, errors are properly returned when asked to parse an ELF file
24//! with an unexpected byte ordering.
25//!
26//! ### ✨ Zero-alloc parser ✨
27//! This crate implements parsing in a way that avoids heap allocations. ELF structures
28//! are parsed and stored on the stack and provided by patterns such as lazily parsed iterators
29//! that yield stack allocated rust types, or lazily parsing tables that only parse out a particular
30//! entry on table.get(index). The structures are copy-converted as needed from the underlying file
31//! data into Rust's native struct representation.
32//!
33//! ### ✨ Fuzz Tested ✨
34//! Various parts of the library are fuzz tested for panics and crashes (see `fuzz/`).
35//!
36//! Memory safety is a core goal, as is providing a safe interface that errors on bad data
37//! over crashing or panicking. Checked integer math is used where appropriate, and ParseErrors are
38//! returned when bad or corrupted ELF structures are encountered.
39//!
40//! ### ✨ Uses only safe interfaces ✨
41//! With memory safety a core goal, this crate contains zero unsafe code blocks
42//! of its own and only uses safe interface methods from core and std, so you can
43//! trust in rust's memory safety guarantees without also having to trust this
44//! library developer as having truly been "right" in why some unsafe block was
45//! safe. 💃
46//!
47//! Note: I'd love to see this crate be enhanced further once rust provides safe transmutes.
48//!
49//! See: <https://github.com/rust-lang/project-safe-transmute>
50//!
51//! ### ✨ Some zero-copy interfaces ✨
52//! The StringTable, for instance, yields `&[u8]` and `&str` backed by the raw string table bytes.
53//!
54//! The [ElfBytes] parser type also does not make raw copies of the underlying file data to back
55//! the parser lazy parser interfaces `ParsingIterator` and `ParsingTable`. They merely wrap byte slices
56//! internally, and yield rust repr values on demand, which does entail copying of the bytes into the
57//! parsed rust-native format.
58//!
59//! Depending on the use-case, it can be more efficient to restructure the raw ELF into different layouts
60//! for more efficient interpretation, say, by re-indexing a flat table into a HashMap. `ParsingIterator`s
61//! make that easy and rustily-intuitive.
62//!
63//! The `ParsingIterator`s are also nice in that you can easily zip/enumerate/filter/collect them
64//! how you wish. Do you know that you want to do multiple passes over pairs from different tables? Just
65//! zip/collect them into another type so you only parse/endian-flip each entry once!
66//!
67//! ### ✨ Stream-based lazy i/o interface ✨
68//! The [ElfStream] parser type takes a `std:: Read + Seek` (such as `std::fs::File`) where ranges of
69//! file contents are read lazily on-demand based on what the user wants to parse.
70//!
71//! This, alongside the bytes-oriented interface, allow you to decide which tradeoffs
72//! you want to make. If you're going to be working with the whole file contents,
73//! then the byte slice approach is probably worthwhile to minimize i/o overhead by
74//! streaming the whole file into memory at once. If you're only going to be
75//! inspecting part of the file, then the [ElfStream] approach would help avoid the
76//! overhead of reading a bunch of unused file data just to parse out a few things, (like
77//! grabbing the `.gnu.note.build-id`)
78//!
79//! ### ✨ Tiny library with no dependencies and fast compilation times ✨
80//! Release-target compilation times on this developer's 2021 m1 macbook are sub-second.
81//!
82//! Example using [ElfBytes]:
83//! ```
84//! use elf::ElfBytes;
85//! use elf::endian::AnyEndian;
86//! use elf::note::Note;
87//! use elf::note::NoteGnuBuildId;
88//! use elf::section::SectionHeader;
89//!
90//! let path = std::path::PathBuf::from("sample-objects/symver.x86_64.so");
91//! let file_data = std::fs::read(path).expect("Could not read file.");
92//! let slice = file_data.as_slice();
93//! let file = ElfBytes::<AnyEndian>::minimal_parse(slice).expect("Open test1");
94//!
95//! // Get the ELF file's build-id
96//! let abi_shdr: SectionHeader = file
97//! .section_header_by_name(".note.gnu.build-id")
98//! .expect("section table should be parseable")
99//! .expect("file should have a .note.ABI-tag section");
100//!
101//! let notes: Vec<Note> = file
102//! .section_data_as_notes(&abi_shdr)
103//! .expect("Should be able to get note section data")
104//! .collect();
105//! assert_eq!(
106//! notes[0],
107//! Note::GnuBuildId(NoteGnuBuildId(
108//! &[140, 51, 19, 23, 221, 90, 215, 131, 169, 13,
109//! 210, 183, 215, 77, 216, 175, 167, 110, 3, 209]))
110//! );
111//!
112//! // Find lazy-parsing types for the common ELF sections (we want .dynsym, .dynstr, .hash)
113//! let common = file.find_common_data().expect("shdrs should parse");
114//! let (dynsyms, strtab) = (common.dynsyms.unwrap(), common.dynsyms_strs.unwrap());
115//! let hash_table = common.sysv_hash.unwrap();
116//!
117//! // Use the hash table to find a given symbol in it.
118//! let name = b"memset";
119//! let (sym_idx, sym) = hash_table.find(name, &dynsyms, &strtab)
120//! .expect("hash table and symbols should parse").unwrap();
121//!
122//! // Verify that we got the same symbol from the hash table we expected
123//! assert_eq!(sym_idx, 2);
124//! assert_eq!(strtab.get(sym.st_name as usize).unwrap(), "memset");
125//! assert_eq!(sym, dynsyms.get(sym_idx).unwrap());
126//! ```
127
128#![cfg_attr(not(feature = "std"), no_std)]
129#![cfg_attr(all(feature = "nightly", not(feature = "std")), feature(error_in_core))]
130#![warn(rust_2018_idioms)]
131#![deny(missing_debug_implementations)]
132#![forbid(unsafe_code)]
133
134pub mod abi;
135
136pub mod compression;
137pub mod dynamic;
138pub mod file;
139pub mod gnu_symver;
140pub mod hash;
141pub mod note;
142pub mod relocation;
143pub mod section;
144pub mod segment;
145pub mod string_table;
146pub mod symbol;
147
148#[cfg(feature = "to_str")]
149pub mod to_str;
150
151pub mod endian;
152pub mod parse;
153
154mod elf_bytes;
155pub use elf_bytes::CommonElfData;
156pub use elf_bytes::ElfBytes;
157
158#[cfg(feature = "std")]
159mod elf_stream;
160#[cfg(feature = "std")]
161pub use elf_stream::ElfStream;
162
163pub use parse::ParseError;