lib.rs 36 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818
  1. #![no_std]
  2. #![deny(missing_debug_implementations)]
  3. use core::fmt;
  4. use header::{Tag, TagIter};
  5. pub use boot_loader_name::BootLoaderNameTag;
  6. pub use elf_sections::{ElfSectionsTag, ElfSection, ElfSectionIter, ElfSectionType, ElfSectionFlags};
  7. pub use framebuffer::{FramebufferTag, FramebufferType, FramebufferField, FramebufferColor};
  8. pub use memory_map::{MemoryMapTag, MemoryArea, MemoryAreaIter};
  9. pub use module::{ModuleTag, ModuleIter};
  10. pub use command_line::CommandLineTag;
  11. pub use rsdp::{RsdpV1Tag, RsdpV2Tag};
  12. #[macro_use]
  13. extern crate bitflags;
  14. mod header;
  15. mod boot_loader_name;
  16. mod elf_sections;
  17. mod memory_map;
  18. mod module;
  19. mod command_line;
  20. mod rsdp;
  21. mod framebuffer;
  22. pub unsafe fn load(address: usize) -> BootInformation {
  23. if !cfg!(test) {
  24. assert_eq!(0, address & 0b111);
  25. }
  26. let multiboot = &*(address as *const BootInformationInner);
  27. assert_eq!(0, multiboot.total_size & 0b111);
  28. assert!(multiboot.has_valid_end_tag());
  29. BootInformation { inner: multiboot }
  30. }
  31. pub struct BootInformation {
  32. inner: *const BootInformationInner,
  33. }
  34. #[repr(C, packed)]
  35. struct BootInformationInner {
  36. total_size: u32,
  37. _reserved: u32,
  38. }
  39. impl BootInformation {
  40. pub fn start_address(&self) -> usize {
  41. self.inner as usize
  42. }
  43. pub fn end_address(&self) -> usize {
  44. self.start_address() + self.total_size()
  45. }
  46. pub fn total_size(&self) -> usize {
  47. self.get().total_size as usize
  48. }
  49. pub fn elf_sections_tag(&self) -> Option<ElfSectionsTag> {
  50. self.get_tag(9).map(|tag| elf_sections::elf_sections_tag(tag))
  51. }
  52. pub fn memory_map_tag(&self) -> Option<&'static MemoryMapTag> {
  53. self.get_tag(6).map(|tag| unsafe { &*(tag as *const Tag as *const MemoryMapTag) })
  54. }
  55. pub fn module_tags(&self) -> ModuleIter {
  56. module::module_iter(self.tags())
  57. }
  58. pub fn boot_loader_name_tag(&self) -> Option<&'static BootLoaderNameTag> {
  59. self.get_tag(2).map(|tag| unsafe { &*(tag as *const Tag as *const BootLoaderNameTag) })
  60. }
  61. pub fn command_line_tag(&self) -> Option<&'static CommandLineTag> {
  62. self.get_tag(1).map(|tag| unsafe { &*(tag as *const Tag as *const CommandLineTag) })
  63. }
  64. pub fn framebuffer_tag(&self) -> Option<FramebufferTag<'static>> {
  65. self.get_tag(8).map(|tag| framebuffer::framebuffer_tag(tag))
  66. }
  67. pub fn rsdp_v1_tag(&self) -> Option<&'static RsdpV1Tag> {
  68. self.get_tag(14).map(|tag| unsafe { &*(tag as *const Tag as *const RsdpV1Tag) })
  69. }
  70. pub fn rsdp_v2_tag(&self) -> Option<&'static RsdpV2Tag> {
  71. self.get_tag(15).map(|tag| unsafe { &*(tag as *const Tag as *const RsdpV2Tag) })
  72. }
  73. fn get(&self) -> &BootInformationInner {
  74. unsafe { &*self.inner }
  75. }
  76. fn get_tag(&self, typ: u32) -> Option<&'static Tag> {
  77. self.tags().find(|tag| tag.typ == typ)
  78. }
  79. fn tags(&self) -> TagIter {
  80. TagIter { current: unsafe { self.inner.offset(1) } as *const _ }
  81. }
  82. }
  83. impl BootInformationInner {
  84. fn has_valid_end_tag(&self) -> bool {
  85. const END_TAG: Tag = Tag { typ: 0, size: 8 };
  86. let self_ptr = self as *const _;
  87. let end_tag_addr = self_ptr as usize + (self.total_size - END_TAG.size) as usize;
  88. let end_tag = unsafe { &*(end_tag_addr as *const Tag) };
  89. end_tag.typ == END_TAG.typ && end_tag.size == END_TAG.size
  90. }
  91. }
  92. impl fmt::Debug for BootInformation {
  93. fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
  94. writeln!(f, "multiboot information")?;
  95. writeln!(f, "S: {:#010X}, E: {:#010X}, L: {:#010X}",
  96. self.start_address(), self.end_address(), self.total_size())?;
  97. if let Some(boot_loader_name_tag) = self.boot_loader_name_tag() {
  98. writeln!(f, "boot loader name: {}", boot_loader_name_tag.name())?;
  99. }
  100. if let Some(command_line_tag) = self.command_line_tag() {
  101. writeln!(f, "command line: {}", command_line_tag.command_line())?;
  102. }
  103. if let Some(memory_map_tag) = self.memory_map_tag() {
  104. writeln!(f, "memory areas:")?;
  105. for area in memory_map_tag.memory_areas() {
  106. writeln!(f, " S: {:#010X}, E: {:#010X}, L: {:#010X}",
  107. area.start_address(), area.end_address(), area.size())?;
  108. }
  109. }
  110. if let Some(elf_sections_tag) = self.elf_sections_tag() {
  111. writeln!(f, "kernel sections:")?;
  112. for s in elf_sections_tag.sections() {
  113. writeln!(f, " name: {:15}, S: {:#08X}, E: {:#08X}, L: {:#08X}, F: {:#04X}",
  114. s.name(), s.start_address(),
  115. s.start_address() + s.size(), s.size(), s.flags().bits())?;
  116. }
  117. }
  118. writeln!(f, "module tags:")?;
  119. for mt in self.module_tags() {
  120. writeln!(f, " name: {:15}, S: {:#010X}, E: {:#010X}",
  121. mt.name(), mt.start_address(), mt.end_address())?;
  122. }
  123. Ok(())
  124. }
  125. }
  126. pub(crate) struct Reader {
  127. pub(crate) ptr: *const u8,
  128. pub(crate) off: usize
  129. }
  130. impl Reader {
  131. pub(crate) fn new<T>(ptr: *const T) -> Reader {
  132. Reader {
  133. ptr: ptr as *const u8,
  134. off: 0
  135. }
  136. }
  137. pub(crate) fn read_u8(&mut self) -> u8 {
  138. self.off += 1;
  139. unsafe {
  140. core::ptr::read(self.ptr.offset((self.off - 1) as isize))
  141. }
  142. }
  143. pub(crate) fn read_u16(&mut self) -> u16 {
  144. self.read_u8() as u16 | (self.read_u8() as u16) << 8
  145. }
  146. pub(crate) fn read_u32(&mut self) -> u32 {
  147. self.read_u16() as u32 | (self.read_u16() as u32) << 16
  148. }
  149. pub(crate) fn read_u64(&mut self) -> u64 {
  150. self.read_u32() as u64 | (self.read_u32() as u64) << 32
  151. }
  152. pub(crate) fn skip(&mut self, n: usize) {
  153. self.off += n;
  154. }
  155. }
  156. #[cfg(test)]
  157. mod tests {
  158. use super::load;
  159. use super::{ElfSectionFlags, ElfSectionType};
  160. use super::FramebufferType;
  161. #[test]
  162. fn no_tags() {
  163. let bytes: [u8; 16] = [
  164. 16, 0, 0, 0, // total_size
  165. 0, 0, 0, 0, // reserved
  166. 0, 0, 0, 0, // end tag type
  167. 8, 0, 0, 0, // end tag size
  168. ];
  169. let addr = bytes.as_ptr() as usize;
  170. let bi = unsafe { load(addr) };
  171. assert_eq!(addr, bi.start_address());
  172. assert_eq!(addr + bytes.len(), bi.end_address());
  173. assert_eq!(bytes.len(), bi.total_size());
  174. assert!(bi.elf_sections_tag().is_none());
  175. assert!(bi.memory_map_tag().is_none());
  176. assert!(bi.module_tags().next().is_none());
  177. assert!(bi.boot_loader_name_tag().is_none());
  178. assert!(bi.command_line_tag().is_none());
  179. }
  180. #[test]
  181. #[should_panic]
  182. fn invalid_total_size() {
  183. let bytes: [u8; 15] = [
  184. 15, 0, 0, 0, // total_size
  185. 0, 0, 0, 0, // reserved
  186. 0, 0, 0, 0, // end tag type
  187. 8, 0, 0, // end tag size
  188. ];
  189. let addr = bytes.as_ptr() as usize;
  190. let bi = unsafe { load(addr) };
  191. assert_eq!(addr, bi.start_address());
  192. assert_eq!(addr + bytes.len(), bi.end_address());
  193. assert_eq!(bytes.len(), bi.total_size());
  194. assert!(bi.elf_sections_tag().is_none());
  195. assert!(bi.memory_map_tag().is_none());
  196. assert!(bi.module_tags().next().is_none());
  197. assert!(bi.boot_loader_name_tag().is_none());
  198. assert!(bi.command_line_tag().is_none());
  199. }
  200. #[test]
  201. #[should_panic]
  202. fn invalid_end_tag() {
  203. let bytes: [u8; 16] = [
  204. 16, 0, 0, 0, // total_size
  205. 0, 0, 0, 0, // reserved
  206. 0, 0, 0, 0, // end tag type
  207. 9, 0, 0, 0, // end tag size
  208. ];
  209. let addr = bytes.as_ptr() as usize;
  210. let bi = unsafe { load(addr) };
  211. assert_eq!(addr, bi.start_address());
  212. assert_eq!(addr + bytes.len(), bi.end_address());
  213. assert_eq!(bytes.len(), bi.total_size());
  214. assert!(bi.elf_sections_tag().is_none());
  215. assert!(bi.memory_map_tag().is_none());
  216. assert!(bi.module_tags().next().is_none());
  217. assert!(bi.boot_loader_name_tag().is_none());
  218. assert!(bi.command_line_tag().is_none());
  219. }
  220. #[test]
  221. fn name_tag() {
  222. let bytes: [u8; 32] = [
  223. 32, 0, 0, 0, // total_size
  224. 0, 0, 0, 0, // reserved
  225. 2, 0, 0, 0, // boot loader name tag type
  226. 13, 0, 0, 0, // boot loader name tag size
  227. 110, 97, 109, 101, // boot loader name 'name'
  228. 0, 0, 0, 0, // boot loader name null + padding
  229. 0, 0, 0, 0, // end tag type
  230. 8, 0, 0, 0, // end tag size
  231. ];
  232. let addr = bytes.as_ptr() as usize;
  233. let bi = unsafe { load(addr) };
  234. assert_eq!(addr, bi.start_address());
  235. assert_eq!(addr + bytes.len(), bi.end_address());
  236. assert_eq!(bytes.len(), bi.total_size());
  237. assert!(bi.elf_sections_tag().is_none());
  238. assert!(bi.memory_map_tag().is_none());
  239. assert!(bi.module_tags().next().is_none());
  240. assert_eq!("name", bi.boot_loader_name_tag().unwrap().name());
  241. assert!(bi.command_line_tag().is_none());
  242. }
  243. #[test]
  244. fn framebuffer_tag_rgb() {
  245. // direct RGB mode test:
  246. // taken from GRUB2 running in QEMU at
  247. // 1280x720 with 32bpp in BGRA format.
  248. let bytes: [u8; 56] = [
  249. 56, 0, 0, 0, // total size
  250. 0, 0, 0, 0, // reserved
  251. 8, 0, 0, 0, // framebuffer tag type
  252. 40, 0, 0, 0, // framebuffer tag size
  253. 0, 0, 0, 253, // framebuffer low dword of address
  254. 0, 0, 0, 0, // framebuffer high dword of address
  255. 0, 20, 0, 0, // framebuffer pitch
  256. 0, 5, 0, 0, // framebuffer width
  257. 208, 2, 0, 0, // framebuffer height
  258. 32, 1, 0, 0, // framebuffer bpp, type, reserved word
  259. 16, 8, 8, 8, // framebuffer red pos/size, green pos/size
  260. 0, 8, 0, 0, // framebuffer blue pos/size, padding word
  261. 0, 0, 0, 0, // end tag type
  262. 8, 0, 0, 0 // end tag size
  263. ];
  264. let addr = bytes.as_ptr() as usize;
  265. let bi = unsafe { load(addr) };
  266. assert_eq!(addr, bi.start_address());
  267. assert_eq!(addr + bytes.len(), bi.end_address());
  268. assert_eq!(bytes.len(), bi.total_size());
  269. use framebuffer::{FramebufferTag, FramebufferField, FramebufferType};
  270. assert_eq!(bi.framebuffer_tag(), Some(FramebufferTag {
  271. address: 4244635648,
  272. pitch: 5120,
  273. width: 1280,
  274. height: 720,
  275. bpp: 32,
  276. buffer_type: FramebufferType::RGB {
  277. red: FramebufferField {
  278. position: 16, size: 8
  279. },
  280. green: FramebufferField {
  281. position: 8, size: 8
  282. },
  283. blue: FramebufferField {
  284. position: 0, size: 8
  285. }
  286. }
  287. }))
  288. }
  289. #[test]
  290. fn framebuffer_tag_indexed() {
  291. // indexed mode test:
  292. // this is synthetic, as I can't get QEMU
  293. // to run in indexed color mode.
  294. let bytes: [u8; 56] = [
  295. 56, 0, 0, 0, // total size
  296. 0, 0, 0, 0, // reserved
  297. 8, 0, 0, 0, // framebuffer tag type
  298. 40, 0, 0, 0, // framebuffer tag size
  299. 0, 0, 0, 253, // framebuffer low dword of address
  300. 0, 0, 0, 0, // framebuffer high dword of address
  301. 0, 20, 0, 0, // framebuffer pitch
  302. 0, 5, 0, 0, // framebuffer width
  303. 208, 2, 0, 0, // framebuffer height
  304. 32, 0, 0, 0, // framebuffer bpp, type, reserved word
  305. 0, 1, 0, 0, // framebuffer palette length
  306. 0, 24, 1, 0, // framebuffer palette address
  307. 0, 0, 0, 0, // end tag type
  308. 8, 0, 0, 0 // end tag size
  309. ];
  310. let addr = bytes.as_ptr() as usize;
  311. let bi = unsafe { load(addr) };
  312. assert_eq!(addr, bi.start_address());
  313. assert_eq!(addr + bytes.len(), bi.end_address());
  314. assert_eq!(bytes.len(), bi.total_size());
  315. use framebuffer::{FramebufferTag, FramebufferField, FramebufferType};
  316. assert!(bi.framebuffer_tag().is_some());
  317. let fbi = bi.framebuffer_tag().unwrap();
  318. assert_eq!(fbi.address, 4244635648);
  319. assert_eq!(fbi.pitch, 5120);
  320. assert_eq!(fbi.width, 1280);
  321. assert_eq!(fbi.height, 720);
  322. assert_eq!(fbi.bpp, 32);
  323. match fbi.buffer_type {
  324. FramebufferType::Indexed { palette } => {
  325. assert_eq!(palette.as_ptr() as usize, 71680);
  326. assert_eq!(palette.len(), 256);
  327. },
  328. _ => panic!("Expected indexed framebuffer type.")
  329. }
  330. }
  331. #[test]
  332. fn grub2() {
  333. let mut bytes: [u8; 960] = [
  334. 192, 3, 0, 0, // total_size
  335. 0, 0, 0, 0, // reserved
  336. 1, 0, 0, 0, // boot command tag type
  337. 9, 0, 0, 0, // boot command tag size
  338. 0, 0, 0, 0, // boot command null + padding
  339. 0, 0, 0, 0, // boot command padding
  340. 2, 0, 0, 0, // boot loader name tag type
  341. 26, 0, 0, 0, // boot loader name tag size
  342. 71, 82, 85, 66, // boot loader name
  343. 32, 50, 46, 48, // boot loader name
  344. 50, 126, 98, 101, // boot loader name
  345. 116, 97, 51, 45, // boot loader name
  346. 53, 0, 0, 0, // boot loader name null + padding
  347. 0, 0, 0, 0, // boot loader name padding
  348. 10, 0, 0, 0, // APM tag type
  349. 28, 0, 0, 0, // APM tag size
  350. 2, 1, 0, 240, // APM version, cseg
  351. 207, 212, 0, 0, // APM offset
  352. 0, 240, 0, 240, // APM cseg_16, dseg
  353. 3, 0, 240, 255, // APM flags, cseg_len
  354. 240, 255, 240, 255, // APM cseg_16_len, dseg_len
  355. 0, 0, 0, 0, // APM padding
  356. 6, 0, 0, 0, // memory map tag type
  357. 160, 0, 0, 0, // memory map tag size
  358. 24, 0, 0, 0, // memory map entry_size
  359. 0, 0, 0, 0, // memory map entry_version
  360. 0, 0, 0, 0, // memory map entry 0 base_addr
  361. 0, 0, 0, 0, // memory map entry 0 base_addr
  362. 0, 252, 9, 0, // memory map entry 0 length
  363. 0, 0, 0, 0, // memory map entry 0 length
  364. 1, 0, 0, 0, // memory map entry 0 type
  365. 0, 0, 0, 0, // memory map entry 0 reserved
  366. 0, 252, 9, 0, // memory map entry 1 base_addr
  367. 0, 0, 0, 0, // memory map entry 1 base_addr
  368. 0, 4, 0, 0, // memory map entry 1 length
  369. 0, 0, 0, 0, // memory map entry 1 length
  370. 2, 0, 0, 0, // memory map entry 1 type
  371. 0, 0, 0, 0, // memory map entry 1 reserved
  372. 0, 0, 15, 0, // memory map entry 2 base_addr
  373. 0, 0, 0, 0, // memory map entry 2 base_addr
  374. 0, 0, 1, 0, // memory map entry 2 length
  375. 0, 0, 0, 0, // memory map entry 2 length
  376. 2, 0, 0, 0, // memory map entry 2 type
  377. 0, 0, 0, 0, // memory map entry 2 reserved
  378. 0, 0, 16, 0, // memory map entry 3 base_addr
  379. 0, 0, 0, 0, // memory map entry 3 base_addr
  380. 0, 0, 238, 7, // memory map entry 3 length
  381. 0, 0, 0, 0, // memory map entry 3 length
  382. 1, 0, 0, 0, // memory map entry 3 type
  383. 0, 0, 0, 0, // memory map entry 3 reserved
  384. 0, 0, 254, 7, // memory map entry 4 base_addr
  385. 0, 0, 0, 0, // memory map entry 4 base_addr
  386. 0, 0, 2, 0, // memory map entry 4 length
  387. 0, 0, 0, 0, // memory map entry 4 length
  388. 2, 0, 0, 0, // memory map entry 4 type
  389. 0, 0, 0, 0, // memory map entry 4 reserved
  390. 0, 0, 252, 255, // memory map entry 5 base_addr
  391. 0, 0, 0, 0, // memory map entry 5 base_addr
  392. 0, 0, 4, 0, // memory map entry 5 length
  393. 0, 0, 0, 0, // memory map entry 5 length
  394. 2, 0, 0, 0, // memory map entry 5 type
  395. 0, 0, 0, 0, // memory map entry 5 reserved
  396. 9, 0, 0, 0, // elf symbols tag type
  397. 84, 2, 0, 0, // elf symbols tag size
  398. 9, 0, 0, 0, // elf symbols num
  399. 64, 0, 0, 0, // elf symbols entsize
  400. 8, 0, 0, 0, // elf symbols shndx
  401. 0, 0, 0, 0, // elf symbols entry 0 name
  402. 0, 0, 0, 0, // elf symbols entry 0 type
  403. 0, 0, 0, 0, // elf symbols entry 0 flags
  404. 0, 0, 0, 0, // elf symbols entry 0 flags
  405. 0, 0, 0, 0, // elf symbols entry 0 addr
  406. 0, 0, 0, 0, // elf symbols entry 0 addr
  407. 0, 0, 0, 0, // elf symbols entry 0 offset
  408. 0, 0, 0, 0, // elf symbols entry 0 offset
  409. 0, 0, 0, 0, // elf symbols entry 0 size
  410. 0, 0, 0, 0, // elf symbols entry 0 size
  411. 0, 0, 0, 0, // elf symbols entry 0 link
  412. 0, 0, 0, 0, // elf symbols entry 0 info
  413. 0, 0, 0, 0, // elf symbols entry 0 addralign
  414. 0, 0, 0, 0, // elf symbols entry 0 addralign
  415. 0, 0, 0, 0, // elf symbols entry 0 entsize
  416. 0, 0, 0, 0, // elf symbols entry 0 entsize
  417. 27, 0, 0, 0, // elf symbols entry 1 name
  418. 1, 0, 0, 0, // elf symbols entry 1 type
  419. 2, 0, 0, 0, // elf symbols entry 1 flags
  420. 0, 0, 0, 0, // elf symbols entry 1 flags
  421. 0, 0, 16, 0, // elf symbols entry 1 addr
  422. 0, 128, 255, 255, // elf symbols entry 1 addr
  423. 0, 16, 0, 0, // elf symbols entry 1 offset
  424. 0, 0, 0, 0, // elf symbols entry 1 offset
  425. 0, 48, 0, 0, // elf symbols entry 1 size
  426. 0, 0, 0, 0, // elf symbols entry 1 size
  427. 0, 0, 0, 0, // elf symbols entry 1 link
  428. 0, 0, 0, 0, // elf symbols entry 1 info
  429. 16, 0, 0, 0, // elf symbols entry 1 addralign
  430. 0, 0, 0, 0, // elf symbols entry 1 addralign
  431. 0, 0, 0, 0, // elf symbols entry 1 entsize
  432. 0, 0, 0, 0, // elf symbols entry 1 entsize
  433. 35, 0, 0, 0, // elf symbols entry 2 name
  434. 1, 0, 0, 0, // elf symbols entry 2 type
  435. 6, 0, 0, 0, // elf symbols entry 2 flags
  436. 0, 0, 0, 0, // elf symbols entry 2 flags
  437. 0, 48, 16, 0, // elf symbols entry 2 addr
  438. 0, 128, 255, 255, // elf symbols entry 2 addr
  439. 0, 64, 0, 0, // elf symbols entry 2 offset
  440. 0, 0, 0, 0, // elf symbols entry 2 offset
  441. 0, 144, 0, 0, // elf symbols entry 2 size
  442. 0, 0, 0, 0, // elf symbols entry 2 size
  443. 0, 0, 0, 0, // elf symbols entry 2 link
  444. 0, 0, 0, 0, // elf symbols entry 2 info
  445. 16, 0, 0, 0, // elf symbols entry 2 addralign
  446. 0, 0, 0, 0, // elf symbols entry 2 addralign
  447. 0, 0, 0, 0, // elf symbols entry 2 entsize
  448. 0, 0, 0, 0, // elf symbols entry 2 entsize
  449. 41, 0, 0, 0, // elf symbols entry 3 name
  450. 1, 0, 0, 0, // elf symbols entry 3 type
  451. 3, 0, 0, 0, // elf symbols entry 3 flags
  452. 0, 0, 0, 0, // elf symbols entry 3 flags
  453. 0, 192, 16, 0, // elf symbols entry 3 addr
  454. 0, 128, 255, 255, // elf symbols entry 3 addr
  455. 0, 208, 0, 0, // elf symbols entry 3 offset
  456. 0, 0, 0, 0, // elf symbols entry 3 offset
  457. 0, 32, 0, 0, // elf symbols entry 3 size
  458. 0, 0, 0, 0, // elf symbols entry 3 size
  459. 0, 0, 0, 0, // elf symbols entry 3 link
  460. 0, 0, 0, 0, // elf symbols entry 3 info
  461. 8, 0, 0, 0, // elf symbols entry 3 addralign
  462. 0, 0, 0, 0, // elf symbols entry 3 addralign
  463. 0, 0, 0, 0, // elf symbols entry 3 entsize
  464. 0, 0, 0, 0, // elf symbols entry 3 entsize
  465. 47, 0, 0, 0, // elf symbols entry 4 name
  466. 8, 0, 0, 0, // elf symbols entry 4 type
  467. 3, 0, 0, 0, // elf symbols entry 4 flags
  468. 0, 0, 0, 0, // elf symbols entry 4 flags
  469. 0, 224, 16, 0, // elf symbols entry 4 addr
  470. 0, 128, 255, 255, // elf symbols entry 4 addr
  471. 0, 240, 0, 0, // elf symbols entry 4 offset
  472. 0, 0, 0, 0, // elf symbols entry 4 offset
  473. 0, 80, 0, 0, // elf symbols entry 4 size
  474. 0, 0, 0, 0, // elf symbols entry 4 size
  475. 0, 0, 0, 0, // elf symbols entry 4 link
  476. 0, 0, 0, 0, // elf symbols entry 4 info
  477. 0, 16, 0, 0, // elf symbols entry 4 addralign
  478. 0, 0, 0, 0, // elf symbols entry 4 addralign
  479. 0, 0, 0, 0, // elf symbols entry 4 entsize
  480. 0, 0, 0, 0, // elf symbols entry 4 entsize
  481. 52, 0, 0, 0, // elf symbols entry 5 name
  482. 1, 0, 0, 0, // elf symbols entry 5 type
  483. 3, 0, 0, 0, // elf symbols entry 5 flags
  484. 0, 0, 0, 0, // elf symbols entry 5 flags
  485. 0, 48, 17, 0, // elf symbols entry 5 addr
  486. 0, 128, 255, 255, // elf symbols entry 5 addr
  487. 0, 240, 0, 0, // elf symbols entry 5 offset
  488. 0, 0, 0, 0, // elf symbols entry 5 offset
  489. 0, 0, 0, 0, // elf symbols entry 5 size
  490. 0, 0, 0, 0, // elf symbols entry 5 size
  491. 0, 0, 0, 0, // elf symbols entry 5 link
  492. 0, 0, 0, 0, // elf symbols entry 5 info
  493. 1, 0, 0, 0, // elf symbols entry 5 addralign
  494. 0, 0, 0, 0, // elf symbols entry 5 addralign
  495. 0, 0, 0, 0, // elf symbols entry 5 entsize
  496. 0, 0, 0, 0, // elf symbols entry 5 entsize
  497. 1, 0, 0, 0, // elf symbols entry 6 name
  498. 2, 0, 0, 0, // elf symbols entry 6 type
  499. 0, 0, 0, 0, // elf symbols entry 6 flags
  500. 0, 0, 0, 0, // elf symbols entry 6 flags
  501. 0, 48, 17, 0, // elf symbols entry 6 addr
  502. 0, 0, 0, 0, // elf symbols entry 6 addr
  503. 0, 240, 0, 0, // elf symbols entry 6 offset
  504. 0, 0, 0, 0, // elf symbols entry 6 offset
  505. 224, 43, 0, 0, // elf symbols entry 6 size
  506. 0, 0, 0, 0, // elf symbols entry 6 size
  507. 7, 0, 0, 0, // elf symbols entry 6 link
  508. 102, 1, 0, 0, // elf symbols entry 6 info
  509. 8, 0, 0, 0, // elf symbols entry 6 addralign
  510. 0, 0, 0, 0, // elf symbols entry 6 addralign
  511. 24, 0, 0, 0, // elf symbols entry 6 entsize
  512. 0, 0, 0, 0, // elf symbols entry 6 entsize
  513. 9, 0, 0, 0, // elf symbols entry 7 name
  514. 3, 0, 0, 0, // elf symbols entry 7 type
  515. 0, 0, 0, 0, // elf symbols entry 7 flags
  516. 0, 0, 0, 0, // elf symbols entry 7 flags
  517. 224, 91, 17, 0, // elf symbols entry 7 addr
  518. 0, 0, 0, 0, // elf symbols entry 7 addr
  519. 224, 27, 1, 0, // elf symbols entry 7 offset
  520. 0, 0, 0, 0, // elf symbols entry 7 offset
  521. 145, 55, 0, 0, // elf symbols entry 7 size
  522. 0, 0, 0, 0, // elf symbols entry 7 size
  523. 0, 0, 0, 0, // elf symbols entry 7 link
  524. 0, 0, 0, 0, // elf symbols entry 7 info
  525. 1, 0, 0, 0, // elf symbols entry 7 addralign
  526. 0, 0, 0, 0, // elf symbols entry 7 addralign
  527. 0, 0, 0, 0, // elf symbols entry 7 entsize
  528. 0, 0, 0, 0, // elf symbols entry 7 entsize
  529. 17, 0, 0, 0, // elf symbols entry 8 name
  530. 3, 0, 0, 0, // elf symbols entry 8 type
  531. 0, 0, 0, 0, // elf symbols entry 8 flags
  532. 0, 0, 0, 0, // elf symbols entry 8 flags
  533. 113, 147, 17, 0, // elf symbols entry 8 addr
  534. 0, 0, 0, 0, // elf symbols entry 8 addr
  535. 113, 83, 1, 0, // elf symbols entry 8 offset
  536. 0, 0, 0, 0, // elf symbols entry 8 offset
  537. 65, 0, 0, 0, // elf symbols entry 8 size
  538. 0, 0, 0, 0, // elf symbols entry 8 size
  539. 0, 0, 0, 0, // elf symbols entry 8 link
  540. 0, 0, 0, 0, // elf symbols entry 8 info
  541. 1, 0, 0, 0, // elf symbols entry 8 addralign
  542. 0, 0, 0, 0, // elf symbols entry 8 addralign
  543. 0, 0, 0, 0, // elf symbols entry 8 entsize
  544. 0, 0, 0, 0, // elf symbols entry 8 entsize
  545. 0, 0, 0, 0, // elf symbols padding
  546. 4, 0, 0, 0, // basic memory tag type
  547. 16, 0, 0, 0, // basic memory tag size
  548. 127, 2, 0, 0, // basic memory mem_lower
  549. 128, 251, 1, 0, // basic memory mem_upper
  550. 5, 0, 0, 0, // BIOS boot device tag type
  551. 20, 0, 0, 0, // BIOS boot device tag size
  552. 224, 0, 0, 0, // BIOS boot device biosdev
  553. 255, 255, 255, 255, // BIOS boot device partition
  554. 255, 255, 255, 255, // BIOS boot device subpartition
  555. 0, 0, 0, 0, // BIOS boot device padding
  556. 8, 0, 0, 0, // framebuffer info tag type
  557. 32, 0, 0, 0, // framebuffer info tag size
  558. 0, 128, 11, 0, // framebuffer info framebuffer_addr
  559. 0, 0, 0, 0, // framebuffer info framebuffer_addr
  560. 160, 0, 0, 0, // framebuffer info framebuffer_pitch
  561. 80, 0, 0, 0, // framebuffer info framebuffer_width
  562. 25, 0, 0, 0, // framebuffer info framebuffer_height
  563. 16, 2, 0, 0, // framebuffer info framebuffer_[bpp,type], reserved, color_info
  564. 14, 0, 0, 0, // ACPI old tag type
  565. 28, 0, 0, 0, // ACPI old tag size
  566. 82, 83, 68, 32, // ACPI old
  567. 80, 84, 82, 32, // ACPI old
  568. 89, 66, 79, 67, // ACPI old
  569. 72, 83, 32, 0, // ACPI old
  570. 220, 24, 254, 7, // ACPI old
  571. 0, 0, 0, 0, // ACPI old padding
  572. 0, 0, 0, 0, // end tag type
  573. 8, 0, 0, 0, // end tag size
  574. ];
  575. let string_bytes: [u8; 65] = [
  576. 0, 46, 115, 121,
  577. 109, 116, 97, 98,
  578. 0, 46, 115, 116,
  579. 114, 116, 97, 98,
  580. 0, 46, 115, 104,
  581. 115, 116, 114, 116,
  582. 97, 98, 0, 46,
  583. 114, 111, 100, 97,
  584. 116, 97, 0, 46,
  585. 116, 101, 120, 116,
  586. 0, 46, 100, 97,
  587. 116, 97, 0, 46,
  588. 98, 115, 115, 0,
  589. 46, 100, 97, 116,
  590. 97, 46, 114, 101,
  591. 108, 46, 114, 111,
  592. 0,
  593. ];
  594. let string_addr = string_bytes.as_ptr() as u64;
  595. for i in 0..8 {
  596. bytes[796 + i] = (string_addr >> (i * 8)) as u8;
  597. }
  598. let addr = bytes.as_ptr() as usize;
  599. let bi = unsafe { load(addr) };
  600. assert_eq!(addr, bi.start_address());
  601. assert_eq!(addr + bytes.len(), bi.end_address());
  602. assert_eq!(bytes.len(), bi.total_size() as usize);
  603. let es = bi.elf_sections_tag().unwrap();
  604. let mut s = es.sections();
  605. let s1 = s.next().unwrap();
  606. assert_eq!(".rodata", s1.name());
  607. assert_eq!(0xFFFF_8000_0010_0000, s1.start_address());
  608. assert_eq!(0xFFFF_8000_0010_3000, s1.end_address());
  609. assert_eq!(0x0000_0000_0000_3000, s1.size());
  610. assert_eq!(ElfSectionFlags::ALLOCATED, s1.flags());
  611. assert_eq!(ElfSectionType::ProgramSection, s1.section_type());
  612. let s2 = s.next().unwrap();
  613. assert_eq!(".text", s2.name());
  614. assert_eq!(0xFFFF_8000_0010_3000, s2.start_address());
  615. assert_eq!(0xFFFF_8000_0010_C000, s2.end_address());
  616. assert_eq!(0x0000_0000_0000_9000, s2.size());
  617. assert_eq!(ElfSectionFlags::EXECUTABLE | ElfSectionFlags::ALLOCATED, s2.flags());
  618. assert_eq!(ElfSectionType::ProgramSection, s2.section_type());
  619. let s3 = s.next().unwrap();
  620. assert_eq!(".data", s3.name());
  621. assert_eq!(0xFFFF_8000_0010_C000, s3.start_address());
  622. assert_eq!(0xFFFF_8000_0010_E000, s3.end_address());
  623. assert_eq!(0x0000_0000_0000_2000, s3.size());
  624. assert_eq!(ElfSectionFlags::ALLOCATED | ElfSectionFlags::WRITABLE, s3.flags());
  625. assert_eq!(ElfSectionType::ProgramSection, s3.section_type());
  626. let s4 = s.next().unwrap();
  627. assert_eq!(".bss", s4.name());
  628. assert_eq!(0xFFFF_8000_0010_E000, s4.start_address());
  629. assert_eq!(0xFFFF_8000_0011_3000, s4.end_address());
  630. assert_eq!(0x0000_0000_0000_5000, s4.size());
  631. assert_eq!(ElfSectionFlags::ALLOCATED | ElfSectionFlags::WRITABLE, s4.flags());
  632. assert_eq!(ElfSectionType::Uninitialized, s4.section_type());
  633. let s5 = s.next().unwrap();
  634. assert_eq!(".data.rel.ro", s5.name());
  635. assert_eq!(0xFFFF_8000_0011_3000, s5.start_address());
  636. assert_eq!(0xFFFF_8000_0011_3000, s5.end_address());
  637. assert_eq!(0x0000_0000_0000_0000, s5.size());
  638. assert_eq!(ElfSectionFlags::ALLOCATED | ElfSectionFlags::WRITABLE, s5.flags());
  639. assert_eq!(ElfSectionType::ProgramSection, s5.section_type());
  640. let s6 = s.next().unwrap();
  641. assert_eq!(".symtab", s6.name());
  642. assert_eq!(0x0000_0000_0011_3000, s6.start_address());
  643. assert_eq!(0x0000_0000_0011_5BE0, s6.end_address());
  644. assert_eq!(0x0000_0000_0000_2BE0, s6.size());
  645. assert_eq!(ElfSectionFlags::empty(), s6.flags());
  646. assert_eq!(ElfSectionType::LinkerSymbolTable, s6.section_type());
  647. let s7 = s.next().unwrap();
  648. assert_eq!(".strtab", s7.name());
  649. assert_eq!(0x0000_0000_0011_5BE0, s7.start_address());
  650. assert_eq!(0x0000_0000_0011_9371, s7.end_address());
  651. assert_eq!(0x0000_0000_0000_3791, s7.size());
  652. assert_eq!(ElfSectionFlags::empty(), s7.flags());
  653. assert_eq!(ElfSectionType::StringTable, s7.section_type());
  654. let s8 = s.next().unwrap();
  655. assert_eq!(".shstrtab", s8.name());
  656. assert_eq!(string_addr, s8.start_address());
  657. assert_eq!(string_addr + string_bytes.len() as u64, s8.end_address());
  658. assert_eq!(string_bytes.len() as u64, s8.size());
  659. assert_eq!(ElfSectionFlags::empty(), s8.flags());
  660. assert_eq!(ElfSectionType::StringTable, s8.section_type());
  661. assert!(s.next().is_none());
  662. let mut mm = bi.memory_map_tag().unwrap().memory_areas();
  663. let mm1 = mm.next().unwrap();
  664. assert_eq!(0x00000000, mm1.start_address());
  665. assert_eq!(0x009_FC00, mm1.end_address());
  666. assert_eq!(0x009_FC00, mm1.size());
  667. let mm2 = mm.next().unwrap();
  668. assert_eq!(0x010_0000, mm2.start_address());
  669. assert_eq!(0x7FE_0000, mm2.end_address());
  670. assert_eq!(0x7EE_0000, mm2.size());
  671. assert!(mm.next().is_none());
  672. // Test the RSDP tag
  673. let rsdp_old = bi.rsdp_v1_tag().unwrap();
  674. assert_eq!("RSD PTR ", rsdp_old.signature().unwrap());
  675. assert_eq!(89, rsdp_old.checksum());
  676. assert_eq!("BOCHS ", rsdp_old.oem_id().unwrap());
  677. assert_eq!(0, rsdp_old.revision());
  678. assert_eq!(0x7FE18DC, rsdp_old.rsdt_address());
  679. assert!(bi.module_tags().next().is_none());
  680. assert_eq!("GRUB 2.02~beta3-5", bi.boot_loader_name_tag().unwrap().name());
  681. assert_eq!("", bi.command_line_tag().unwrap().command_line());
  682. // Test the Framebuffer tag
  683. let fbi = bi.framebuffer_tag().unwrap();
  684. assert_eq!(fbi.address, 753664);
  685. assert_eq!(fbi.pitch, 160);
  686. assert_eq!(fbi.width, 80);
  687. assert_eq!(fbi.height, 25);
  688. assert_eq!(fbi.bpp, 16);
  689. assert_eq!(fbi.buffer_type, FramebufferType::Text);
  690. }
  691. #[test]
  692. fn elf_sections() {
  693. let mut bytes: [u8; 168] = [
  694. 168, 0, 0, 0, // total_size
  695. 0, 0, 0, 0, // reserved
  696. 9, 0, 0, 0, // elf symbols tag type
  697. 20, 2, 0, 0, // elf symbols tag size
  698. 2, 0, 0, 0, // elf symbols num
  699. 64, 0, 0, 0, // elf symbols entsize
  700. 1, 0, 0, 0, // elf symbols shndx
  701. 0, 0, 0, 0, // elf symbols entry 0 name
  702. 0, 0, 0, 0, // elf symbols entry 0 type
  703. 0, 0, 0, 0, // elf symbols entry 0 flags
  704. 0, 0, 0, 0, // elf symbols entry 0 flags
  705. 0, 0, 0, 0, // elf symbols entry 0 addr
  706. 0, 0, 0, 0, // elf symbols entry 0 addr
  707. 0, 0, 0, 0, // elf symbols entry 0 offset
  708. 0, 0, 0, 0, // elf symbols entry 0 offset
  709. 0, 0, 0, 0, // elf symbols entry 0 size
  710. 0, 0, 0, 0, // elf symbols entry 0 size
  711. 0, 0, 0, 0, // elf symbols entry 0 link
  712. 0, 0, 0, 0, // elf symbols entry 0 info
  713. 0, 0, 0, 0, // elf symbols entry 0 addralign
  714. 0, 0, 0, 0, // elf symbols entry 0 addralign
  715. 0, 0, 0, 0, // elf symbols entry 0 entsize
  716. 0, 0, 0, 0, // elf symbols entry 0 entsize
  717. 1, 0, 0, 0, // elf symbols entry 1 name
  718. 3, 0, 0, 0, // elf symbols entry 1 type
  719. 0, 0, 0, 0, // elf symbols entry 1 flags
  720. 0, 0, 0, 0, // elf symbols entry 1 flags
  721. 255, 255, 255, 255, // elf symbols entry 1 addr
  722. 255, 255, 255, 255, // elf symbols entry 1 addr
  723. 113, 83, 1, 0, // elf symbols entry 1 offset
  724. 0, 0, 0, 0, // elf symbols entry 1 offset
  725. 11, 0, 0, 0, // elf symbols entry 1 size
  726. 0, 0, 0, 0, // elf symbols entry 1 size
  727. 0, 0, 0, 0, // elf symbols entry 1 link
  728. 0, 0, 0, 0, // elf symbols entry 1 info
  729. 1, 0, 0, 0, // elf symbols entry 1 addralign
  730. 0, 0, 0, 0, // elf symbols entry 1 addralign
  731. 0, 0, 0, 0, // elf symbols entry 1 entsize
  732. 0, 0, 0, 0, // elf symbols entry 1 entsize
  733. 0, 0, 0, 0, // elf symbols padding
  734. 0, 0, 0, 0, // end tag type
  735. 8, 0, 0, 0, // end tag size
  736. ];
  737. let string_bytes: [u8; 11] = [
  738. 0, 46, 115, 104,
  739. 115, 116, 114, 116,
  740. 97, 98, 0,
  741. ];
  742. let string_addr = string_bytes.as_ptr() as u64;
  743. for i in 0..8 {
  744. let offset = 108;
  745. assert_eq!(255, bytes[offset + i]);
  746. bytes[offset + i] = (string_addr >> (i * 8)) as u8;
  747. }
  748. let addr = bytes.as_ptr() as usize;
  749. let bi = unsafe { load(addr) };
  750. assert_eq!(addr, bi.start_address());
  751. assert_eq!(addr + bytes.len(), bi.end_address());
  752. assert_eq!(bytes.len(), bi.total_size() as usize);
  753. let es = bi.elf_sections_tag().unwrap();
  754. let mut s = es.sections();
  755. let s1 = s.next().unwrap();
  756. assert_eq!(".shstrtab", s1.name());
  757. assert_eq!(string_addr, s1.start_address());
  758. assert_eq!(string_addr + string_bytes.len() as u64, s1.end_address());
  759. assert_eq!(string_bytes.len() as u64, s1.size());
  760. assert_eq!(ElfSectionFlags::empty(), s1.flags());
  761. assert_eq!(ElfSectionType::StringTable, s1.section_type());
  762. assert!(s.next().is_none());
  763. }
  764. }