information_request.rs 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157
  1. use crate::{HeaderTagFlag, MbiTagType};
  2. use crate::{HeaderTagType, MbiTagTypeId};
  3. use core::fmt;
  4. use core::fmt::{Debug, Formatter};
  5. use core::marker::PhantomData;
  6. use core::mem::size_of;
  7. use multiboot2::TagType;
  8. /// Specifies what specific tag types the bootloader should provide
  9. /// inside the mbi.
  10. #[derive(Copy, Clone, PartialEq, Eq)]
  11. #[repr(C)]
  12. pub struct InformationRequestHeaderTag<const N: usize> {
  13. typ: HeaderTagType,
  14. flags: HeaderTagFlag,
  15. size: u32,
  16. // Length is determined by size.
  17. // Must be parsed during runtime with unsafe pointer magic and the size field.
  18. requests: [MbiTagTypeId; N],
  19. }
  20. impl<const N: usize> InformationRequestHeaderTag<N> {
  21. /// Creates a new object. The size parameter is the value of the size property.
  22. /// It doesn't have to match with `N` necessarily, because during compile time we
  23. /// can't know the size of the tag in all runtime situations.
  24. pub fn new(flags: HeaderTagFlag, requests: [MbiTagTypeId; N], size: Option<u32>) -> Self {
  25. InformationRequestHeaderTag {
  26. typ: HeaderTagType::InformationRequest,
  27. flags,
  28. size: size.unwrap_or(size_of::<Self>() as u32),
  29. requests,
  30. }
  31. }
  32. pub const fn typ(&self) -> HeaderTagType {
  33. self.typ
  34. }
  35. pub const fn flags(&self) -> HeaderTagFlag {
  36. self.flags
  37. }
  38. pub const fn size(&self) -> u32 {
  39. self.size
  40. }
  41. /// Returns the requests as array. Only works if the number of requests
  42. /// is known at compile time. For safety and correctness during runtime,
  43. /// you should use `req_iter()`.
  44. pub const fn requests(&self) -> [MbiTagTypeId; N] {
  45. // cheap to copy, otherwise difficult with lifetime
  46. self.requests
  47. }
  48. /// Returns the number of [`MbiTagType`]-requests derived
  49. /// from the `size`-property. This method is useful
  50. /// because this struct uses a const generic, but during runtime
  51. /// we don't know the value in almost any case.
  52. pub const fn dynamic_requests_size(&self) -> u32 {
  53. let base_struct_size = size_of::<InformationRequestHeaderTag<0>>();
  54. let size_diff = self.size - base_struct_size as u32;
  55. if size_diff > 0 {
  56. size_diff / size_of::<u32>() as u32
  57. } else {
  58. 0
  59. }
  60. }
  61. /// Returns an [`InformationRequestHeaderTagIter`].
  62. pub fn req_iter(&self) -> InformationRequestHeaderTagIter {
  63. let base_struct_size = size_of::<InformationRequestHeaderTag<0>>();
  64. let count = self.dynamic_requests_size();
  65. let base_ptr = self as *const InformationRequestHeaderTag<N>;
  66. let base_ptr = base_ptr as *const u8;
  67. let base_ptr = unsafe { base_ptr.add(base_struct_size) };
  68. let base_ptr = base_ptr as *const MbiTagTypeId;
  69. InformationRequestHeaderTagIter::new(count, base_ptr)
  70. }
  71. }
  72. impl<const N: usize> Debug for InformationRequestHeaderTag<N> {
  73. fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
  74. f.debug_struct("InformationRequestHeaderTag")
  75. .field("type", &{ self.typ })
  76. .field("flags", &{ self.flags })
  77. .field("size", &{ self.size })
  78. .field("requests", &{ self.req_iter() })
  79. .finish()
  80. }
  81. }
  82. /// Iterates the dynamically sized information request structure and finds all MBI tags
  83. /// that are requested.
  84. #[derive(Copy, Clone)]
  85. pub struct InformationRequestHeaderTagIter<'a> {
  86. base_ptr: *const MbiTagTypeId,
  87. i: u32,
  88. count: u32,
  89. _marker: PhantomData<&'a ()>,
  90. }
  91. impl<'a> InformationRequestHeaderTagIter<'a> {
  92. const fn new(count: u32, base_ptr: *const MbiTagTypeId) -> Self {
  93. Self {
  94. i: 0,
  95. count,
  96. base_ptr,
  97. _marker: PhantomData,
  98. }
  99. }
  100. }
  101. impl<'a> Iterator for InformationRequestHeaderTagIter<'a> {
  102. type Item = MbiTagType;
  103. fn next(&mut self) -> Option<Self::Item> {
  104. if self.i < self.count {
  105. let ptr = unsafe { self.base_ptr.offset(self.i as isize) };
  106. self.i += 1;
  107. let tag_type_id = unsafe { *ptr };
  108. Some(TagType::from(tag_type_id))
  109. } else {
  110. None
  111. }
  112. }
  113. }
  114. impl<'a> Debug for InformationRequestHeaderTagIter<'a> {
  115. fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
  116. let mut debug = f.debug_list();
  117. self.for_each(|e| {
  118. debug.entry(&e);
  119. });
  120. debug.finish()
  121. }
  122. }
  123. #[cfg(test)]
  124. mod tests {
  125. use crate::InformationRequestHeaderTag;
  126. #[test]
  127. #[allow(clippy::erasing_op)]
  128. #[allow(clippy::identity_op)]
  129. fn test_assert_size() {
  130. assert_eq!(
  131. core::mem::size_of::<InformationRequestHeaderTag<0>>(),
  132. 2 + 2 + 4 + 0 * 4
  133. );
  134. assert_eq!(
  135. core::mem::size_of::<InformationRequestHeaderTag<1>>(),
  136. 2 + 2 + 4 + 1 * 4
  137. );
  138. assert_eq!(
  139. core::mem::size_of::<InformationRequestHeaderTag<2>>(),
  140. 2 + 2 + 4 + 2 * 4
  141. );
  142. }
  143. }