elf_sections.rs 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374
  1. use header::Tag;
  2. /// This tag contains section header table from an ELF kernel.
  3. ///
  4. /// The sections iterator is provided via the `sections` method.
  5. #[derive(Debug)]
  6. pub struct ElfSectionsTag {
  7. inner: *const ElfSectionsTagInner,
  8. offset: usize,
  9. }
  10. pub unsafe fn elf_sections_tag(tag: &Tag, offset: usize) -> ElfSectionsTag {
  11. assert_eq!(9, tag.typ);
  12. let es = ElfSectionsTag {
  13. inner: (tag as *const Tag).offset(1) as *const ElfSectionsTagInner,
  14. offset,
  15. };
  16. assert!((es.get().entry_size * es.get().shndx) <= tag.size);
  17. es
  18. }
  19. #[derive(Clone, Copy, Debug)]
  20. #[repr(C, packed)] // only repr(C) would add unwanted padding at the end
  21. struct ElfSectionsTagInner {
  22. number_of_sections: u32,
  23. entry_size: u32,
  24. shndx: u32, // string table
  25. }
  26. impl ElfSectionsTag {
  27. /// Get an iterator of loaded ELF sections.
  28. ///
  29. /// # Examples
  30. ///
  31. /// ```ignore
  32. /// if let Some(elf_tag) = boot_info.elf_sections_tag() {
  33. /// let mut total = 0;
  34. /// for section in elf_tag.sections() {
  35. /// println!("Section: {:?}", section);
  36. /// total += 1;
  37. /// }
  38. /// }
  39. /// ```
  40. pub fn sections(&self) -> impl Iterator<Item = ElfSection> {
  41. let string_section_offset = (self.get().shndx * self.get().entry_size) as isize;
  42. let string_section_ptr =
  43. unsafe { self.first_section().offset(string_section_offset) as *const _ };
  44. ElfSectionIter {
  45. current_section: self.first_section(),
  46. remaining_sections: self.get().number_of_sections,
  47. entry_size: self.get().entry_size,
  48. string_section: string_section_ptr,
  49. offset: self.offset,
  50. }
  51. }
  52. fn first_section(&self) -> *const u8 {
  53. (unsafe { self.inner.offset(1) }) as *const _
  54. }
  55. fn get(&self) -> &ElfSectionsTagInner {
  56. unsafe { &*self.inner }
  57. }
  58. }
  59. /// An iterator over some ELF sections.
  60. #[derive(Clone, Debug)]
  61. pub struct ElfSectionIter {
  62. current_section: *const u8,
  63. remaining_sections: u32,
  64. entry_size: u32,
  65. string_section: *const u8,
  66. offset: usize,
  67. }
  68. impl Iterator for ElfSectionIter {
  69. type Item = ElfSection;
  70. fn next(&mut self) -> Option<ElfSection> {
  71. while self.remaining_sections != 0 {
  72. let section = ElfSection {
  73. inner: self.current_section,
  74. string_section: self.string_section,
  75. entry_size: self.entry_size,
  76. offset: self.offset,
  77. };
  78. self.current_section = unsafe { self.current_section.offset(self.entry_size as isize) };
  79. self.remaining_sections -= 1;
  80. if section.section_type() != ElfSectionType::Unused {
  81. return Some(section);
  82. }
  83. }
  84. None
  85. }
  86. }
  87. /// A single generic ELF Section.
  88. #[derive(Debug)]
  89. pub struct ElfSection {
  90. inner: *const u8,
  91. string_section: *const u8,
  92. entry_size: u32,
  93. offset: usize,
  94. }
  95. #[derive(Clone, Copy, Debug)]
  96. #[repr(C, packed)]
  97. struct ElfSectionInner32 {
  98. name_index: u32,
  99. typ: u32,
  100. flags: u32,
  101. addr: u32,
  102. offset: u32,
  103. size: u32,
  104. link: u32,
  105. info: u32,
  106. addralign: u32,
  107. entry_size: u32,
  108. }
  109. #[derive(Clone, Copy, Debug)]
  110. #[repr(C, packed)]
  111. struct ElfSectionInner64 {
  112. name_index: u32,
  113. typ: u32,
  114. flags: u64,
  115. addr: u64,
  116. offset: u64,
  117. size: u64,
  118. link: u32,
  119. info: u32,
  120. addralign: u64,
  121. entry_size: u64,
  122. }
  123. impl ElfSection {
  124. /// Get the section type as a `ElfSectionType` enum variant.
  125. pub fn section_type(&self) -> ElfSectionType {
  126. match self.get().typ() {
  127. 0 => ElfSectionType::Unused,
  128. 1 => ElfSectionType::ProgramSection,
  129. 2 => ElfSectionType::LinkerSymbolTable,
  130. 3 => ElfSectionType::StringTable,
  131. 4 => ElfSectionType::RelaRelocation,
  132. 5 => ElfSectionType::SymbolHashTable,
  133. 6 => ElfSectionType::DynamicLinkingTable,
  134. 7 => ElfSectionType::Note,
  135. 8 => ElfSectionType::Uninitialized,
  136. 9 => ElfSectionType::RelRelocation,
  137. 10 => ElfSectionType::Reserved,
  138. 11 => ElfSectionType::DynamicLoaderSymbolTable,
  139. 0x6000_0000..=0x6FFF_FFFF => ElfSectionType::EnvironmentSpecific,
  140. 0x7000_0000..=0x7FFF_FFFF => ElfSectionType::ProcessorSpecific,
  141. _ => panic!(),
  142. }
  143. }
  144. /// Get the "raw" section type as a `u32`
  145. pub fn section_type_raw(&self) -> u32 {
  146. self.get().typ()
  147. }
  148. /// Read the name of the section.
  149. pub fn name(&self) -> &str {
  150. use core::{slice, str};
  151. let name_ptr = unsafe { self.string_table().offset(self.get().name_index() as isize) };
  152. let strlen = {
  153. let mut len = 0;
  154. while unsafe { *name_ptr.offset(len) } != 0 {
  155. len += 1;
  156. }
  157. len as usize
  158. };
  159. str::from_utf8(unsafe { slice::from_raw_parts(name_ptr, strlen) }).unwrap()
  160. }
  161. /// Get the physical start address of the section.
  162. pub fn start_address(&self) -> u64 {
  163. self.get().addr()
  164. }
  165. /// Get the physical end address of the section.
  166. ///
  167. /// This is the same as doing `section.start_address() + section.size()`
  168. pub fn end_address(&self) -> u64 {
  169. self.get().addr() + self.get().size()
  170. }
  171. /// Get the section's size in bytes.
  172. pub fn size(&self) -> u64 {
  173. self.get().size()
  174. }
  175. /// Get the section's address alignment constraints.
  176. ///
  177. /// That is, the value of `start_address` must be congruent to 0,
  178. /// modulo the value of `addrlign`. Currently, only 0 and positive
  179. /// integral powers of two are allowed. Values 0 and 1 mean the section has no
  180. /// alignment constraints.
  181. pub fn addralign(&self) -> u64 {
  182. self.get().addralign()
  183. }
  184. /// Get the section's flags.
  185. pub fn flags(&self) -> ElfSectionFlags {
  186. ElfSectionFlags::from_bits_truncate(self.get().flags())
  187. }
  188. /// Check if the `ALLOCATED` flag is set in the section flags.
  189. pub fn is_allocated(&self) -> bool {
  190. self.flags().contains(ElfSectionFlags::ALLOCATED)
  191. }
  192. fn get(&self) -> &dyn ElfSectionInner {
  193. match self.entry_size {
  194. 40 => unsafe { &*(self.inner as *const ElfSectionInner32) },
  195. 64 => unsafe { &*(self.inner as *const ElfSectionInner64) },
  196. _ => panic!(),
  197. }
  198. }
  199. unsafe fn string_table(&self) -> *const u8 {
  200. let addr = match self.entry_size {
  201. 40 => (*(self.string_section as *const ElfSectionInner32)).addr as usize,
  202. 64 => (*(self.string_section as *const ElfSectionInner64)).addr as usize,
  203. _ => panic!(),
  204. };
  205. (addr + self.offset) as *const _
  206. }
  207. }
  208. trait ElfSectionInner {
  209. fn name_index(&self) -> u32;
  210. fn typ(&self) -> u32;
  211. fn flags(&self) -> u64;
  212. fn addr(&self) -> u64;
  213. fn size(&self) -> u64;
  214. fn addralign(&self) -> u64;
  215. }
  216. impl ElfSectionInner for ElfSectionInner32 {
  217. fn name_index(&self) -> u32 {
  218. self.name_index
  219. }
  220. fn typ(&self) -> u32 {
  221. self.typ
  222. }
  223. fn flags(&self) -> u64 {
  224. self.flags.into()
  225. }
  226. fn addr(&self) -> u64 {
  227. self.addr.into()
  228. }
  229. fn size(&self) -> u64 {
  230. self.size.into()
  231. }
  232. fn addralign(&self) -> u64 {
  233. self.addralign.into()
  234. }
  235. }
  236. impl ElfSectionInner for ElfSectionInner64 {
  237. fn name_index(&self) -> u32 {
  238. self.name_index
  239. }
  240. fn typ(&self) -> u32 {
  241. self.typ
  242. }
  243. fn flags(&self) -> u64 {
  244. self.flags
  245. }
  246. fn addr(&self) -> u64 {
  247. self.addr
  248. }
  249. fn size(&self) -> u64 {
  250. self.size
  251. }
  252. fn addralign(&self) -> u64 {
  253. self.addralign.into()
  254. }
  255. }
  256. /// An enum abstraction over raw ELF section types.
  257. #[derive(PartialEq, Eq, Debug, Copy, Clone)]
  258. #[repr(u32)]
  259. pub enum ElfSectionType {
  260. /// This value marks the section header as inactive; it does not have an
  261. /// associated section. Other members of the section header have undefined
  262. /// values.
  263. Unused = 0,
  264. /// The section holds information defined by the program, whose format and
  265. /// meaning are determined solely by the program.
  266. ProgramSection = 1,
  267. /// This section holds a linker symbol table.
  268. LinkerSymbolTable = 2,
  269. /// The section holds a string table.
  270. StringTable = 3,
  271. /// The section holds relocation entries with explicit addends, such as type
  272. /// Elf32_Rela for the 32-bit class of object files. An object file may have
  273. /// multiple relocation sections.
  274. RelaRelocation = 4,
  275. /// The section holds a symbol hash table.
  276. SymbolHashTable = 5,
  277. /// The section holds dynamic linking tables.
  278. DynamicLinkingTable = 6,
  279. /// This section holds information that marks the file in some way.
  280. Note = 7,
  281. /// A section of this type occupies no space in the file but otherwise resembles
  282. /// `ProgramSection`. Although this section contains no bytes, the
  283. /// sh_offset member contains the conceptual file offset.
  284. Uninitialized = 8,
  285. /// The section holds relocation entries without explicit addends, such as type
  286. /// Elf32_Rel for the 32-bit class of object files. An object file may have
  287. /// multiple relocation sections.
  288. RelRelocation = 9,
  289. /// This section type is reserved but has unspecified semantics.
  290. Reserved = 10,
  291. /// This section holds a dynamic loader symbol table.
  292. DynamicLoaderSymbolTable = 11,
  293. /// Values in this inclusive range (`[0x6000_0000, 0x6FFF_FFFF)`) are
  294. /// reserved for environment-specific semantics.
  295. EnvironmentSpecific = 0x6000_0000,
  296. /// Values in this inclusive range (`[0x7000_0000, 0x7FFF_FFFF)`) are
  297. /// reserved for processor-specific semantics.
  298. ProcessorSpecific = 0x7000_0000,
  299. }
  300. bitflags! {
  301. /// ELF Section bitflags.
  302. pub struct ElfSectionFlags: u64 {
  303. /// The section contains data that should be writable during program execution.
  304. const WRITABLE = 0x1;
  305. /// The section occupies memory during the process execution.
  306. const ALLOCATED = 0x2;
  307. /// The section contains executable machine instructions.
  308. const EXECUTABLE = 0x4;
  309. // plus environment-specific use at 0x0F000000
  310. // plus processor-specific use at 0xF0000000
  311. }
  312. }