header.rs 3.0 KB

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