123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137 |
- use ::Reader;
- use core::slice;
- use header::Tag;
- /// The VBE Framebuffer information Tag.
- #[derive(Debug, PartialEq)]
- pub struct FramebufferTag<'a> {
- /// Contains framebuffer physical address.
- ///
- /// This field is 64-bit wide but bootloader should set it under 4GiB if
- /// possible for compatibility with payloads which aren’t aware of PAE or
- /// amd64.
- pub address: u64,
- /// Contains the pitch in bytes.
- pub pitch: u32,
- /// Contains framebuffer width in pixels.
- pub width: u32,
- /// Contains framebuffer height in pixels.
- pub height: u32,
- /// Contains number of bits per pixel.
- pub bpp: u8,
- /// The type of framebuffer, one of: `Indexed`, `RGB` or `Text`.
- pub buffer_type: FramebufferType<'a>,
- }
- /// The type of framebuffer.
- #[derive(Debug, PartialEq)]
- pub enum FramebufferType<'a> {
- /// Indexed color.
- Indexed {
- #[allow(missing_docs)]
- palette: &'a [FramebufferColor],
- },
- /// Direct RGB color.
- #[allow(missing_docs)]
- RGB {
- red: FramebufferField,
- green: FramebufferField,
- blue: FramebufferField,
- },
- /// EGA Text.
- ///
- /// In this case the framebuffer width and height are expressed in
- /// characters and not in pixels.
- ///
- /// The bpp is equal 16 (16 bits per character) and pitch is expressed in bytes per text line.
- Text,
- }
- /// An RGB color type field.
- #[derive(Debug, PartialEq)]
- pub struct FramebufferField {
- /// Color field position.
- pub position: u8,
- /// Color mask size.
- pub size: u8,
- }
- /// A framebuffer color descriptor in the palette.
- #[derive(Clone, Copy, Debug, PartialEq)]
- #[repr(C, packed)]
- pub struct FramebufferColor {
- /// The Red component of the color.
- pub red: u8,
- /// The Green component of the color.
- pub green: u8,
- /// The Blue component of the color.
- pub blue: u8,
- }
- pub fn framebuffer_tag<'a>(tag: &'a Tag) -> FramebufferTag<'a> {
- let mut reader = Reader::new(tag as *const Tag);
- reader.skip(8);
- let address = reader.read_u64();
- let pitch = reader.read_u32();
- let width = reader.read_u32();
- let height = reader.read_u32();
- let bpp = reader.read_u8();
- let type_no = reader.read_u8();
- reader.skip(2); // In the multiboot spec, it has this listed as a u8 _NOT_ a u16.
- // Reading the GRUB2 source code reveals it is in fact a u16.
- let buffer_type = match type_no {
- 0 => {
- let num_colors = reader.read_u32();
- let palette = unsafe {
- slice::from_raw_parts(
- reader.current_address() as *const FramebufferColor,
- num_colors as usize,
- )
- } as &'static [FramebufferColor];
- FramebufferType::Indexed { palette }
- }
- 1 => {
- let red_pos = reader.read_u8(); // These refer to the bit positions of the LSB of each field
- let red_mask = reader.read_u8(); // And then the length of the field from LSB to MSB
- let green_pos = reader.read_u8();
- let green_mask = reader.read_u8();
- let blue_pos = reader.read_u8();
- let blue_mask = reader.read_u8();
- FramebufferType::RGB {
- red: FramebufferField {
- position: red_pos,
- size: red_mask,
- },
- green: FramebufferField {
- position: green_pos,
- size: green_mask,
- },
- blue: FramebufferField {
- position: blue_pos,
- size: blue_mask,
- },
- }
- }
- 2 => FramebufferType::Text,
- _ => panic!("Unknown framebuffer type: {}", type_no),
- };
- FramebufferTag {
- address,
- pitch,
- width,
- height,
- bpp,
- buffer_type,
- }
- }
|