framebuffer.rs 10.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357
  1. use crate::{Tag, TagTrait, TagTypeId};
  2. use core::fmt::Debug;
  3. use core::mem::size_of;
  4. use core::slice;
  5. use derive_more::Display;
  6. #[cfg(feature = "builder")]
  7. use {
  8. crate::builder::boxed_dst_tag, crate::builder::traits::StructAsBytes, crate::TagType,
  9. alloc::boxed::Box, alloc::vec::Vec,
  10. };
  11. /// Helper struct to read bytes from a raw pointer and increase the pointer
  12. /// automatically.
  13. struct Reader {
  14. ptr: *const u8,
  15. off: usize,
  16. }
  17. impl Reader {
  18. fn new<T>(ptr: *const T) -> Reader {
  19. Reader {
  20. ptr: ptr as *const u8,
  21. off: 0,
  22. }
  23. }
  24. fn read_u8(&mut self) -> u8 {
  25. self.off += 1;
  26. unsafe { *self.ptr.add(self.off - 1) }
  27. }
  28. fn read_u16(&mut self) -> u16 {
  29. self.read_u8() as u16 | (self.read_u8() as u16) << 8
  30. }
  31. fn read_u32(&mut self) -> u32 {
  32. self.read_u16() as u32 | (self.read_u16() as u32) << 16
  33. }
  34. fn current_address(&self) -> usize {
  35. unsafe { self.ptr.add(self.off) as usize }
  36. }
  37. }
  38. const METADATA_SIZE: usize = size_of::<TagTypeId>()
  39. + 4 * size_of::<u32>()
  40. + size_of::<u64>()
  41. + size_of::<u16>()
  42. + 2 * size_of::<u8>();
  43. /// The VBE Framebuffer information Tag.
  44. #[derive(ptr_meta::Pointee, Eq)]
  45. #[repr(C)]
  46. pub struct FramebufferTag {
  47. typ: TagTypeId,
  48. size: u32,
  49. /// Contains framebuffer physical address.
  50. ///
  51. /// This field is 64-bit wide but bootloader should set it under 4GiB if
  52. /// possible for compatibility with payloads which aren’t aware of PAE or
  53. /// amd64.
  54. address: u64,
  55. /// Contains the pitch in bytes.
  56. pitch: u32,
  57. /// Contains framebuffer width in pixels.
  58. width: u32,
  59. /// Contains framebuffer height in pixels.
  60. height: u32,
  61. /// Contains number of bits per pixel.
  62. bpp: u8,
  63. /// The type of framebuffer, one of: `Indexed`, `RGB` or `Text`.
  64. type_no: u8,
  65. // In the multiboot spec, it has this listed as a u8 _NOT_ a u16.
  66. // Reading the GRUB2 source code reveals it is in fact a u16.
  67. _reserved: u16,
  68. buffer: [u8],
  69. }
  70. impl FramebufferTag {
  71. #[cfg(feature = "builder")]
  72. pub fn new(
  73. address: u64,
  74. pitch: u32,
  75. width: u32,
  76. height: u32,
  77. bpp: u8,
  78. buffer_type: FramebufferType,
  79. ) -> Box<Self> {
  80. let mut bytes: Vec<u8> = address.to_le_bytes().into();
  81. bytes.extend(pitch.to_le_bytes());
  82. bytes.extend(width.to_le_bytes());
  83. bytes.extend(height.to_le_bytes());
  84. bytes.extend(bpp.to_le_bytes());
  85. bytes.extend(buffer_type.to_bytes());
  86. boxed_dst_tag(TagType::Framebuffer, &bytes)
  87. }
  88. /// Contains framebuffer physical address.
  89. ///
  90. /// This field is 64-bit wide but bootloader should set it under 4GiB if
  91. /// possible for compatibility with payloads which aren’t aware of PAE or
  92. /// amd64.
  93. pub fn address(&self) -> u64 {
  94. self.address
  95. }
  96. /// Contains the pitch in bytes.
  97. pub fn pitch(&self) -> u32 {
  98. self.pitch
  99. }
  100. /// Contains framebuffer width in pixels.
  101. pub fn width(&self) -> u32 {
  102. self.width
  103. }
  104. /// Contains framebuffer height in pixels.
  105. pub fn height(&self) -> u32 {
  106. self.height
  107. }
  108. /// Contains number of bits per pixel.
  109. pub fn bpp(&self) -> u8 {
  110. self.bpp
  111. }
  112. /// The type of framebuffer, one of: `Indexed`, `RGB` or `Text`.
  113. pub fn buffer_type(&self) -> Result<FramebufferType, UnknownFramebufferType> {
  114. let mut reader = Reader::new(self.buffer.as_ptr());
  115. let typ = FramebufferTypeId::try_from(self.type_no)?;
  116. match typ {
  117. FramebufferTypeId::Indexed => {
  118. let num_colors = reader.read_u32();
  119. let palette = unsafe {
  120. slice::from_raw_parts(
  121. reader.current_address() as *const FramebufferColor,
  122. num_colors as usize,
  123. )
  124. } as &'static [FramebufferColor];
  125. Ok(FramebufferType::Indexed { palette })
  126. }
  127. FramebufferTypeId::RGB => {
  128. let red_pos = reader.read_u8(); // These refer to the bit positions of the LSB of each field
  129. let red_mask = reader.read_u8(); // And then the length of the field from LSB to MSB
  130. let green_pos = reader.read_u8();
  131. let green_mask = reader.read_u8();
  132. let blue_pos = reader.read_u8();
  133. let blue_mask = reader.read_u8();
  134. Ok(FramebufferType::RGB {
  135. red: FramebufferField {
  136. position: red_pos,
  137. size: red_mask,
  138. },
  139. green: FramebufferField {
  140. position: green_pos,
  141. size: green_mask,
  142. },
  143. blue: FramebufferField {
  144. position: blue_pos,
  145. size: blue_mask,
  146. },
  147. })
  148. }
  149. FramebufferTypeId::Text => Ok(FramebufferType::Text),
  150. }
  151. }
  152. }
  153. impl TagTrait for FramebufferTag {
  154. fn dst_size(base_tag: &Tag) -> usize {
  155. assert!(base_tag.size as usize >= METADATA_SIZE);
  156. base_tag.size as usize - METADATA_SIZE
  157. }
  158. }
  159. #[cfg(feature = "builder")]
  160. impl StructAsBytes for FramebufferTag {
  161. fn byte_size(&self) -> usize {
  162. self.size.try_into().unwrap()
  163. }
  164. }
  165. impl Debug for FramebufferTag {
  166. fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result {
  167. f.debug_struct("FramebufferTag")
  168. .field("typ", &{ self.typ })
  169. .field("size", &{ self.size })
  170. .field("buffer_type", &self.buffer_type())
  171. .field("address", &{ self.address })
  172. .field("pitch", &{ self.pitch })
  173. .field("width", &{ self.width })
  174. .field("height", &{ self.height })
  175. .field("bpp", &self.bpp)
  176. .finish()
  177. }
  178. }
  179. impl PartialEq for FramebufferTag {
  180. fn eq(&self, other: &Self) -> bool {
  181. ({ self.typ } == { other.typ }
  182. && { self.size } == { other.size }
  183. && { self.address } == { other.address }
  184. && { self.pitch } == { other.pitch }
  185. && { self.width } == { other.width }
  186. && { self.height } == { other.height }
  187. && { self.bpp } == { other.bpp }
  188. && { self.type_no } == { other.type_no }
  189. && self.buffer == other.buffer)
  190. }
  191. }
  192. /// Helper struct for [`FramebufferType`].
  193. #[derive(Copy, Clone, Debug, PartialEq, Eq)]
  194. #[repr(u8)]
  195. #[allow(clippy::upper_case_acronyms)]
  196. enum FramebufferTypeId {
  197. Indexed = 0,
  198. RGB = 1,
  199. Text = 2,
  200. // spec says: there may be more variants in the future
  201. }
  202. impl TryFrom<u8> for FramebufferTypeId {
  203. type Error = UnknownFramebufferType;
  204. fn try_from(value: u8) -> Result<Self, Self::Error> {
  205. match value {
  206. 0 => Ok(Self::Indexed),
  207. 1 => Ok(Self::RGB),
  208. 2 => Ok(Self::Text),
  209. val => Err(UnknownFramebufferType(val)),
  210. }
  211. }
  212. }
  213. /// The type of framebuffer.
  214. #[derive(Clone, Debug, PartialEq, Eq, PartialOrd, Ord, Hash)]
  215. pub enum FramebufferType<'a> {
  216. /// Indexed color.
  217. Indexed {
  218. #[allow(missing_docs)]
  219. palette: &'a [FramebufferColor],
  220. },
  221. /// Direct RGB color.
  222. #[allow(missing_docs)]
  223. #[allow(clippy::upper_case_acronyms)]
  224. RGB {
  225. red: FramebufferField,
  226. green: FramebufferField,
  227. blue: FramebufferField,
  228. },
  229. /// EGA Text.
  230. ///
  231. /// In this case the framebuffer width and height are expressed in
  232. /// characters and not in pixels.
  233. ///
  234. /// The bpp is equal 16 (16 bits per character) and pitch is expressed in bytes per text line.
  235. Text,
  236. }
  237. #[cfg(feature = "builder")]
  238. impl<'a> FramebufferType<'a> {
  239. fn to_bytes(&self) -> Vec<u8> {
  240. let mut v = Vec::new();
  241. match self {
  242. FramebufferType::Indexed { palette } => {
  243. v.extend(0u8.to_le_bytes()); // type
  244. v.extend(0u16.to_le_bytes()); // reserved
  245. v.extend((palette.len() as u32).to_le_bytes());
  246. for color in palette.iter() {
  247. v.extend(color.struct_as_bytes());
  248. }
  249. }
  250. FramebufferType::RGB { red, green, blue } => {
  251. v.extend(1u8.to_le_bytes()); // type
  252. v.extend(0u16.to_le_bytes()); // reserved
  253. v.extend(red.struct_as_bytes());
  254. v.extend(green.struct_as_bytes());
  255. v.extend(blue.struct_as_bytes());
  256. }
  257. FramebufferType::Text => {
  258. v.extend(2u8.to_le_bytes()); // type
  259. v.extend(0u16.to_le_bytes()); // reserved
  260. }
  261. }
  262. v
  263. }
  264. }
  265. /// An RGB color type field.
  266. #[derive(Copy, Clone, Debug, PartialEq, Eq, PartialOrd, Ord, Hash)]
  267. pub struct FramebufferField {
  268. /// Color field position.
  269. pub position: u8,
  270. /// Color mask size.
  271. pub size: u8,
  272. }
  273. #[cfg(feature = "builder")]
  274. impl StructAsBytes for FramebufferField {
  275. fn byte_size(&self) -> usize {
  276. size_of::<Self>()
  277. }
  278. }
  279. /// A framebuffer color descriptor in the palette.
  280. #[derive(Copy, Clone, Debug, PartialEq, Eq, PartialOrd, Ord, Hash)]
  281. #[repr(C)] // no align(8) here is correct
  282. pub struct FramebufferColor {
  283. /// The Red component of the color.
  284. pub red: u8,
  285. /// The Green component of the color.
  286. pub green: u8,
  287. /// The Blue component of the color.
  288. pub blue: u8,
  289. }
  290. /// Error when an unknown [`FramebufferTypeId`] is found.
  291. #[derive(Debug, Copy, Clone, Display, PartialEq, Eq)]
  292. #[display(fmt = "Unknown framebuffer type {}", _0)]
  293. pub struct UnknownFramebufferType(u8);
  294. #[cfg(feature = "unstable")]
  295. impl core::error::Error for UnknownFramebufferType {}
  296. #[cfg(feature = "builder")]
  297. impl StructAsBytes for FramebufferColor {
  298. fn byte_size(&self) -> usize {
  299. size_of::<Self>()
  300. }
  301. }
  302. #[cfg(test)]
  303. mod tests {
  304. use super::*;
  305. // Compile time test
  306. #[test]
  307. fn test_size() {
  308. assert_eq!(size_of::<FramebufferColor>(), 3)
  309. }
  310. }