elf_sections.rs 2.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108
  1. #[derive(Debug)]
  2. #[repr(packed)] // repr(C) would add unwanted padding before first_section
  3. pub struct ElfSectionsTag {
  4. typ: u32,
  5. size: u32,
  6. pub number_of_sections: u32,
  7. entry_size: u32,
  8. shndx: u32, // string table
  9. first_section: ElfSection,
  10. }
  11. impl ElfSectionsTag {
  12. pub fn sections(&'static self) -> ElfSectionIter {
  13. ElfSectionIter {
  14. current_section: &self.first_section,
  15. remaining_sections: self.number_of_sections - 1,
  16. entry_size: self.entry_size,
  17. }
  18. }
  19. }
  20. #[derive(Clone)]
  21. pub struct ElfSectionIter {
  22. current_section: &'static ElfSection,
  23. remaining_sections: u32,
  24. entry_size: u32,
  25. }
  26. impl Iterator for ElfSectionIter {
  27. type Item = &'static ElfSection;
  28. fn next(&mut self) -> Option<&'static ElfSection> {
  29. if self.remaining_sections == 0 {
  30. None
  31. } else {
  32. let section = self.current_section;
  33. let next_section_addr = (self.current_section as *const _ as u32) + self.entry_size;
  34. self.current_section = unsafe{ &*(next_section_addr as *const ElfSection) };
  35. self.remaining_sections -= 1;
  36. if section.typ == ElfSectionType::Unused as u32 {
  37. self.next()
  38. } else {
  39. Some(section)
  40. }
  41. }
  42. }
  43. }
  44. #[derive(Debug)]
  45. #[repr(C)]
  46. pub struct ElfSection {
  47. name: u32,
  48. typ: u32,
  49. pub flags: u64,
  50. pub addr: u64,
  51. offset: u64,
  52. pub size: u64,
  53. link: u32,
  54. info: u32,
  55. addralign: u64,
  56. entry_size: u64,
  57. }
  58. impl ElfSection {
  59. pub fn start_address(&self) -> usize {
  60. self.addr as usize
  61. }
  62. pub fn end_address(&self) -> usize {
  63. (self.addr + self.size) as usize
  64. }
  65. pub fn flags(&self) -> ElfSectionFlags {
  66. ElfSectionFlags::from_bits_truncate(self.flags)
  67. }
  68. pub fn is_allocated(&self) -> bool {
  69. self.flags().contains(ELF_SECTION_ALLOCATED)
  70. }
  71. }
  72. #[repr(u32)]
  73. pub enum ElfSectionType {
  74. Unused = 0,
  75. ProgramSection = 1,
  76. LinkerSymbolTable = 2,
  77. StringTable = 3,
  78. RelaRelocation = 4,
  79. SymbolHashTable = 5,
  80. DynamicLinkingTable = 6,
  81. Note = 7,
  82. Uninitialized = 8,
  83. RelRelocation = 9,
  84. Reserved = 10,
  85. DynamicLoaderSymbolTable = 11,
  86. // plus environment-specific use from 0x60000000 to 0x6FFFFFFF
  87. // plus processor-specific use from 0x70000000 to 0x7FFFFFFF
  88. }
  89. bitflags! {
  90. flags ElfSectionFlags: u64 {
  91. const ELF_SECTION_WRITABLE = 0x1,
  92. const ELF_SECTION_ALLOCATED = 0x2,
  93. const ELF_SECTION_EXECUTABLE = 0x4,
  94. // plus environment-specific use at 0x0F000000
  95. // plus processor-specific use at 0xF0000000
  96. }
  97. }