header.rs 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120
  1. use core::fmt::{Debug, Formatter};
  2. use core::marker::PhantomData;
  3. /// Magic number that a multiboot2-compliant boot loader will store in `eax` register
  4. /// right before handoff to the payload (the kernel). This value can be used to check,
  5. /// that the kernel was indeed booted via multiboot2.
  6. ///
  7. /// Caution: You might need some assembly code (e.g. GAS or NASM) first, which
  8. /// moves `eax` to another register, like `edi`. Otherwise it probably happens,
  9. /// that the Rust compiler output changes `eax` before you can access it.
  10. pub const MULTIBOOT2_BOOTLOADER_MAGIC: u32 = 0x36d76289;
  11. /// Possible Types of a [`Tag`]. The names and values are taken from the example C code
  12. /// at the bottom of the Multiboot2 specification.
  13. #[repr(u32)]
  14. #[derive(Copy, Clone, Debug)]
  15. pub enum TagType {
  16. End = 0,
  17. Cmdline = 1,
  18. BootLoaderName = 2,
  19. Module = 3,
  20. BasicMeminfo = 4,
  21. Bootdev = 5,
  22. Mmap = 6,
  23. Vbe = 7,
  24. Framebuffer = 8,
  25. ElfSections = 9,
  26. Apm = 10,
  27. Efi32 = 11,
  28. Efi64 = 12,
  29. Smbios = 13,
  30. /// Also called "AcpiOld" in other multiboot2 implementations.
  31. AcpiV1 = 14,
  32. /// Refers to version 2 and later of Acpi.
  33. /// Also called "AcpiNew" in other multiboot2 implementations.
  34. AcpiV2 = 15,
  35. Network = 16,
  36. EfiMmap = 17,
  37. EfiBs = 18,
  38. Efi32Ih = 19,
  39. Efi64Ih = 20,
  40. LoadBaseAddr = 21,
  41. }
  42. // each compare/equal direction must be implemented manually
  43. impl PartialEq<u32> for TagType {
  44. fn eq(&self, other: &u32) -> bool {
  45. *self as u32 == *other
  46. }
  47. }
  48. // each compare/equal direction must be implemented manually
  49. impl PartialEq<TagType> for u32 {
  50. fn eq(&self, other: &TagType) -> bool {
  51. *self == *other as u32
  52. }
  53. }
  54. impl PartialEq<TagType> for TagType {
  55. fn eq(&self, other: &TagType) -> bool {
  56. *self as u32 == *other as u32
  57. }
  58. }
  59. /// All tags that could passed via the Multiboot2 information structure to a payload/program/kernel.
  60. /// Better not confuse this with the Multiboot2 header tags. They are something different.
  61. #[derive(Clone, Copy)]
  62. #[repr(C)]
  63. pub struct Tag {
  64. // u32 value
  65. pub typ: TagType,
  66. pub size: u32,
  67. // tag specific fields
  68. }
  69. impl Debug for Tag {
  70. fn fmt(&self, f: &mut Formatter<'_>) -> core::fmt::Result {
  71. f.debug_struct("Tag")
  72. .field("typ", &self.typ)
  73. .field("typ (numeric)", &(self.typ as u32))
  74. .field("size", &(self.size))
  75. .finish()
  76. }
  77. }
  78. #[derive(Clone, Debug)]
  79. pub struct TagIter<'a> {
  80. pub current: *const Tag,
  81. phantom: PhantomData<&'a Tag>,
  82. }
  83. impl<'a> TagIter<'a> {
  84. pub fn new(first: *const Tag) -> Self {
  85. TagIter {
  86. current: first,
  87. phantom: PhantomData,
  88. }
  89. }
  90. }
  91. impl<'a> Iterator for TagIter<'a> {
  92. type Item = &'a Tag;
  93. fn next(&mut self) -> Option<&'a Tag> {
  94. match unsafe { &*self.current } {
  95. &Tag {
  96. typ: TagType::End,
  97. size: 8,
  98. } => None, // end tag
  99. tag => {
  100. // go to next tag
  101. let mut tag_addr = self.current as usize;
  102. tag_addr += ((tag.size + 7) & !7) as usize; //align at 8 byte
  103. self.current = tag_addr as *const _;
  104. Some(tag)
  105. }
  106. }
  107. }
  108. }