relocatable.rs 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105
  1. use crate::{HeaderTagFlag, HeaderTagType};
  2. use core::fmt;
  3. use core::fmt::{Debug, Formatter};
  4. use core::mem::size_of;
  5. /// It contains load address placement suggestion for boot loader. Boot loader
  6. /// should follow it. ‘0’ means none, ‘1’ means load image at lowest possible address
  7. /// but not lower than min addr and ‘2’ means load image at highest possible
  8. /// address but not higher than max addr.
  9. #[repr(u32)]
  10. #[derive(Copy, Clone, Debug, PartialEq, Eq, PartialOrd, Ord, Hash)]
  11. pub enum RelocatableHeaderTagPreference {
  12. /// Let boot loader decide.
  13. None = 0,
  14. /// Locate at lower end of possible address space.
  15. Low = 1,
  16. /// Locate at higher end of possible address space.
  17. High = 2,
  18. }
  19. /// This tag indicates that the image is relocatable.
  20. #[derive(Copy, Clone, PartialEq, Eq)]
  21. #[repr(C, align(8))]
  22. pub struct RelocatableHeaderTag {
  23. typ: HeaderTagType,
  24. flags: HeaderTagFlag,
  25. size: u32,
  26. /// Lowest possible physical address at which image should be loaded. The bootloader cannot load any part of image below this address
  27. min_addr: u32,
  28. /// Highest possible physical address at which loaded image should end. The bootloader cannot load any part of image above this address.
  29. max_addr: u32,
  30. /// Image alignment in memory, e.g. 4096.
  31. align: u32,
  32. preference: RelocatableHeaderTagPreference,
  33. }
  34. impl RelocatableHeaderTag {
  35. pub const fn new(
  36. flags: HeaderTagFlag,
  37. min_addr: u32,
  38. max_addr: u32,
  39. align: u32,
  40. preference: RelocatableHeaderTagPreference,
  41. ) -> Self {
  42. RelocatableHeaderTag {
  43. typ: HeaderTagType::Relocatable,
  44. flags,
  45. size: size_of::<Self>() as u32,
  46. min_addr,
  47. max_addr,
  48. align,
  49. preference,
  50. }
  51. }
  52. pub const fn typ(&self) -> HeaderTagType {
  53. self.typ
  54. }
  55. pub const fn flags(&self) -> HeaderTagFlag {
  56. self.flags
  57. }
  58. pub const fn size(&self) -> u32 {
  59. self.size
  60. }
  61. pub const fn min_addr(&self) -> u32 {
  62. self.min_addr
  63. }
  64. pub const fn max_addr(&self) -> u32 {
  65. self.max_addr
  66. }
  67. pub const fn align(&self) -> u32 {
  68. self.align
  69. }
  70. pub const fn preference(&self) -> RelocatableHeaderTagPreference {
  71. self.preference
  72. }
  73. }
  74. impl Debug for RelocatableHeaderTag {
  75. fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
  76. f.debug_struct("RelocatableHeaderTag")
  77. .field("type", &{ self.typ })
  78. .field("flags", &{ self.flags })
  79. .field("size", &{ self.size })
  80. // trick to print this as hexadecimal pointer
  81. .field("min_addr", &(self.min_addr as *const u32))
  82. .field("max_addr", &(self.max_addr as *const u32))
  83. .field("align", &{ self.align })
  84. .field("preference", &{ self.preference })
  85. .finish()
  86. }
  87. }
  88. #[cfg(test)]
  89. mod tests {
  90. use crate::RelocatableHeaderTag;
  91. #[test]
  92. fn test_assert_size() {
  93. assert_eq!(
  94. core::mem::size_of::<RelocatableHeaderTag>(),
  95. 2 + 2 + 4 + 4 + 4 + 4 + 4
  96. );
  97. }
  98. }