framebuffer.rs 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137
  1. use ::Reader;
  2. use core::slice;
  3. use header::Tag;
  4. /// The VBE Framebuffer information Tag.
  5. #[derive(Debug, PartialEq)]
  6. pub struct FramebufferTag<'a> {
  7. /// Contains framebuffer physical address.
  8. ///
  9. /// This field is 64-bit wide but bootloader should set it under 4GiB if
  10. /// possible for compatibility with payloads which aren’t aware of PAE or
  11. /// amd64.
  12. pub address: u64,
  13. /// Contains the pitch in bytes.
  14. pub pitch: u32,
  15. /// Contains framebuffer width in pixels.
  16. pub width: u32,
  17. /// Contains framebuffer height in pixels.
  18. pub height: u32,
  19. /// Contains number of bits per pixel.
  20. pub bpp: u8,
  21. /// The type of framebuffer, one of: `Indexed`, `RGB` or `Text`.
  22. pub buffer_type: FramebufferType<'a>,
  23. }
  24. /// The type of framebuffer.
  25. #[derive(Debug, PartialEq)]
  26. pub enum FramebufferType<'a> {
  27. /// Indexed color.
  28. Indexed {
  29. #[allow(missing_docs)]
  30. palette: &'a [FramebufferColor],
  31. },
  32. /// Direct RGB color.
  33. #[allow(missing_docs)]
  34. RGB {
  35. red: FramebufferField,
  36. green: FramebufferField,
  37. blue: FramebufferField,
  38. },
  39. /// EGA Text.
  40. ///
  41. /// In this case the framebuffer width and height are expressed in
  42. /// characters and not in pixels.
  43. ///
  44. /// The bpp is equal 16 (16 bits per character) and pitch is expressed in bytes per text line.
  45. Text,
  46. }
  47. /// An RGB color type field.
  48. #[derive(Debug, PartialEq)]
  49. pub struct FramebufferField {
  50. /// Color field position.
  51. pub position: u8,
  52. /// Color mask size.
  53. pub size: u8,
  54. }
  55. /// A framebuffer color descriptor in the palette.
  56. #[derive(Clone, Copy, Debug, PartialEq)]
  57. #[repr(C, packed)]
  58. pub struct FramebufferColor {
  59. /// The Red component of the color.
  60. pub red: u8,
  61. /// The Green component of the color.
  62. pub green: u8,
  63. /// The Blue component of the color.
  64. pub blue: u8,
  65. }
  66. pub fn framebuffer_tag<'a>(tag: &'a Tag) -> FramebufferTag<'a> {
  67. let mut reader = Reader::new(tag as *const Tag);
  68. reader.skip(8);
  69. let address = reader.read_u64();
  70. let pitch = reader.read_u32();
  71. let width = reader.read_u32();
  72. let height = reader.read_u32();
  73. let bpp = reader.read_u8();
  74. let type_no = reader.read_u8();
  75. reader.skip(2); // In the multiboot spec, it has this listed as a u8 _NOT_ a u16.
  76. // Reading the GRUB2 source code reveals it is in fact a u16.
  77. let buffer_type = match type_no {
  78. 0 => {
  79. let num_colors = reader.read_u32();
  80. let palette = unsafe {
  81. slice::from_raw_parts(
  82. reader.current_address() as *const FramebufferColor,
  83. num_colors as usize,
  84. )
  85. } as &'static [FramebufferColor];
  86. FramebufferType::Indexed { palette }
  87. }
  88. 1 => {
  89. let red_pos = reader.read_u8(); // These refer to the bit positions of the LSB of each field
  90. let red_mask = reader.read_u8(); // And then the length of the field from LSB to MSB
  91. let green_pos = reader.read_u8();
  92. let green_mask = reader.read_u8();
  93. let blue_pos = reader.read_u8();
  94. let blue_mask = reader.read_u8();
  95. FramebufferType::RGB {
  96. red: FramebufferField {
  97. position: red_pos,
  98. size: red_mask,
  99. },
  100. green: FramebufferField {
  101. position: green_pos,
  102. size: green_mask,
  103. },
  104. blue: FramebufferField {
  105. position: blue_pos,
  106. size: blue_mask,
  107. },
  108. }
  109. }
  110. 2 => FramebufferType::Text,
  111. _ => panic!("Unknown framebuffer type: {}", type_no),
  112. };
  113. FramebufferTag {
  114. address,
  115. pitch,
  116. width,
  117. height,
  118. bpp,
  119. buffer_type,
  120. }
  121. }