elf_sections.rs 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280
  1. use header::Tag;
  2. #[derive(Debug)]
  3. pub struct ElfSectionsTag {
  4. inner: *const ElfSectionsTagInner,
  5. }
  6. pub fn elf_sections_tag(tag: &Tag) -> ElfSectionsTag {
  7. assert_eq!(9, tag.typ);
  8. let es = ElfSectionsTag {
  9. inner: unsafe { (tag as *const _).offset(1) } as *const _,
  10. };
  11. assert!((es.get().entry_size * es.get().shndx) <= tag.size);
  12. es
  13. }
  14. #[derive(Debug)]
  15. #[repr(C, packed)] // only repr(C) would add unwanted padding at the end
  16. struct ElfSectionsTagInner {
  17. number_of_sections: u32,
  18. entry_size: u32,
  19. shndx: u32, // string table
  20. }
  21. impl ElfSectionsTag {
  22. pub fn sections(&self) -> ElfSectionIter {
  23. let string_section_offset = (self.get().shndx * self.get().entry_size) as isize;
  24. let string_section_ptr = unsafe {
  25. self.first_section().offset(string_section_offset) as *const _
  26. };
  27. ElfSectionIter {
  28. current_section: self.first_section(),
  29. remaining_sections: self.get().number_of_sections - 1,
  30. entry_size: self.get().entry_size,
  31. string_section: string_section_ptr,
  32. }
  33. }
  34. fn first_section(&self) -> *const u8 {
  35. (unsafe { self.inner.offset(1) }) as *const _
  36. }
  37. fn get(&self) -> &ElfSectionsTagInner {
  38. unsafe { &*self.inner }
  39. }
  40. }
  41. #[derive(Clone, Debug)]
  42. pub struct ElfSectionIter {
  43. current_section: *const u8,
  44. remaining_sections: u32,
  45. entry_size: u32,
  46. string_section: *const u8,
  47. }
  48. impl Iterator for ElfSectionIter {
  49. type Item = ElfSection;
  50. fn next(&mut self) -> Option<ElfSection> {
  51. if self.remaining_sections == 0 {
  52. return None;
  53. }
  54. loop {
  55. let section = ElfSection {
  56. inner: self.current_section,
  57. string_section: self.string_section,
  58. entry_size: self.entry_size,
  59. };
  60. self.current_section = unsafe { self.current_section.offset(self.entry_size as isize) };
  61. self.remaining_sections -= 1;
  62. if section.section_type() != ElfSectionType::Unused {
  63. return Some(section);
  64. }
  65. }
  66. }
  67. }
  68. #[derive(Debug)]
  69. pub struct ElfSection {
  70. inner: *const u8,
  71. string_section: *const u8,
  72. entry_size: u32,
  73. }
  74. #[derive(Debug)]
  75. #[repr(C, packed)]
  76. struct ElfSectionInner32 {
  77. name_index: u32,
  78. typ: u32,
  79. flags: u32,
  80. addr: u32,
  81. offset: u32,
  82. size: u32,
  83. link: u32,
  84. info: u32,
  85. addralign: u32,
  86. entry_size: u32,
  87. }
  88. #[derive(Debug)]
  89. #[repr(C, packed)]
  90. struct ElfSectionInner64 {
  91. name_index: u32,
  92. typ: u32,
  93. flags: u64,
  94. addr: u64,
  95. offset: u64,
  96. size: u64,
  97. link: u32,
  98. info: u32,
  99. addralign: u64,
  100. entry_size: u64,
  101. }
  102. impl ElfSection {
  103. pub fn section_type(&self) -> ElfSectionType {
  104. match self.get().typ() {
  105. 0 => ElfSectionType::Unused,
  106. 1 => ElfSectionType::ProgramSection,
  107. 2 => ElfSectionType::LinkerSymbolTable,
  108. 3 => ElfSectionType::StringTable,
  109. 4 => ElfSectionType::RelaRelocation,
  110. 5 => ElfSectionType::SymbolHashTable,
  111. 6 => ElfSectionType::DynamicLinkingTable,
  112. 7 => ElfSectionType::Note,
  113. 8 => ElfSectionType::Uninitialized,
  114. 9 => ElfSectionType::RelRelocation,
  115. 10 => ElfSectionType::Reserved,
  116. 11 => ElfSectionType::DynamicLoaderSymbolTable,
  117. 0x6000_0000...0x6FFF_FFFF => ElfSectionType::EnvironmentSpecific,
  118. 0x7000_0000...0x7FFF_FFFF => ElfSectionType::ProcessorSpecific,
  119. _ => panic!(),
  120. }
  121. }
  122. pub fn section_type_raw(&self) -> u32 {
  123. self.get().typ()
  124. }
  125. pub fn name(&self) -> &str {
  126. use core::{str, slice};
  127. let name_ptr = unsafe {
  128. self.string_table().offset(self.get().name_index() as isize)
  129. };
  130. let strlen = {
  131. let mut len = 0;
  132. while unsafe { *name_ptr.offset(len) } != 0 {
  133. len += 1;
  134. }
  135. len as usize
  136. };
  137. str::from_utf8(unsafe { slice::from_raw_parts(name_ptr, strlen) }).unwrap()
  138. }
  139. pub fn start_address(&self) -> u64 {
  140. self.get().addr()
  141. }
  142. pub fn end_address(&self) -> u64 {
  143. self.get().addr() + self.get().size()
  144. }
  145. pub fn size(&self) -> u64 {
  146. self.get().size()
  147. }
  148. pub fn flags(&self) -> ElfSectionFlags {
  149. ElfSectionFlags::from_bits_truncate(self.get().flags())
  150. }
  151. pub fn is_allocated(&self) -> bool {
  152. self.flags().contains(ElfSectionFlags::ALLOCATED)
  153. }
  154. fn get(&self) -> &ElfSectionInner {
  155. match self.entry_size {
  156. 40 => unsafe { &*(self.inner as *const ElfSectionInner32) },
  157. 64 => unsafe { &*(self.inner as *const ElfSectionInner64) },
  158. _ => panic!(),
  159. }
  160. }
  161. unsafe fn string_table(&self) -> *const u8 {
  162. match self.entry_size {
  163. 40 => (*(self.string_section as *const ElfSectionInner32)).addr as *const _,
  164. 64 => (*(self.string_section as *const ElfSectionInner64)).addr as *const _,
  165. _ => panic!(),
  166. }
  167. }
  168. }
  169. trait ElfSectionInner {
  170. fn name_index(&self) -> u32;
  171. fn typ(&self) -> u32;
  172. fn flags(&self) -> u64;
  173. fn addr(&self) -> u64;
  174. fn size(&self) -> u64;
  175. }
  176. impl ElfSectionInner for ElfSectionInner32 {
  177. fn name_index(&self) -> u32 {
  178. self.name_index
  179. }
  180. fn typ(&self) -> u32 {
  181. self.typ
  182. }
  183. fn flags(&self) -> u64 {
  184. self.flags.into()
  185. }
  186. fn addr(&self) -> u64 {
  187. self.addr.into()
  188. }
  189. fn size(&self) -> u64 {
  190. self.size.into()
  191. }
  192. }
  193. impl ElfSectionInner for ElfSectionInner64 {
  194. fn name_index(&self) -> u32 {
  195. self.name_index
  196. }
  197. fn typ(&self) -> u32 {
  198. self.typ
  199. }
  200. fn flags(&self) -> u64 {
  201. self.flags
  202. }
  203. fn addr(&self) -> u64 {
  204. self.addr
  205. }
  206. fn size(&self) -> u64 {
  207. self.size
  208. }
  209. }
  210. #[derive(PartialEq, Eq, Debug, Copy, Clone)]
  211. #[repr(u32)]
  212. pub enum ElfSectionType {
  213. Unused = 0,
  214. ProgramSection = 1,
  215. LinkerSymbolTable = 2,
  216. StringTable = 3,
  217. RelaRelocation = 4,
  218. SymbolHashTable = 5,
  219. DynamicLinkingTable = 6,
  220. Note = 7,
  221. Uninitialized = 8,
  222. RelRelocation = 9,
  223. Reserved = 10,
  224. DynamicLoaderSymbolTable = 11,
  225. EnvironmentSpecific = 0x6000_0000,
  226. ProcessorSpecific = 0x7000_0000,
  227. }
  228. bitflags! {
  229. pub struct ElfSectionFlags: u64 {
  230. const WRITABLE = 0x1;
  231. const ALLOCATED = 0x2;
  232. const EXECUTABLE = 0x4;
  233. // plus environment-specific use at 0x0F000000
  234. // plus processor-specific use at 0xF0000000
  235. }
  236. }