vga/
lib.rs

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
// Copyright (c) Microsoft Corporation.
// Licensed under the MIT License.

//! This is a port of the PCI VGA adapter from Hyper-V.
//!
//! It, in turn, was originally an emulator of an S3 Trio, but over time it was
//! defeatured to be a generic SVGA card. It still uses the S3 Trio registers
//! for SVGA mode switching, and it has some extra Hyper-V enlightenments, so it
//! must be paired with the proprietary Hyper-V SVGA BIOS.
//!
//! This code needs a lot of cleanup, and various features need to be completed
//! (especially around supporting different graphics modes). Ultimately, the
//! standard core VGA portion should be split out so that alternate SVGA mode
//! switching can be layered on top (e.g. to support the bochs mode switching
//! interface that SeaVGABios uses).

#![expect(missing_docs)]

mod emu;
mod non_linear;
mod render;
mod spec;
mod text_mode;

use chipset_device::ChipsetDevice;
use chipset_device::io::IoResult;
use chipset_device::mmio::MmioIntercept;
use chipset_device::pci::PciConfigSpace;
use chipset_device::pio::PortIoIntercept;
use framebuffer::FramebufferLocalControl;
use guestmem::MapRom;
use inspect::InspectMut;
use render::Renderer;
use thiserror::Error;
use video_core::FramebufferFormat;
use vmcore::device_state::ChangeDeviceState;
use vmcore::save_restore::SaveRestore;
use vmcore::save_restore::SavedStateNotSupported;
use vmcore::vm_task::VmTaskDriver;
use vmcore::vmtime::VmTimeSource;

#[derive(InspectMut)]
pub struct VgaDevice {
    #[inspect(flatten)]
    emu: emu::Emulator,
    renderer: Renderer,
}

#[derive(Debug, Error)]
pub enum Error {
    #[error("failed to map framebuffer")]
    Framebuffer(#[source] std::io::Error),
}

impl VgaDevice {
    pub fn new(
        driver: &VmTaskDriver,
        vmtime: &VmTimeSource,
        mut control: FramebufferLocalControl,
        rom: Option<Box<dyn MapRom>>,
    ) -> Result<Self, Error> {
        control.set_format(FramebufferFormat {
            width: 800,
            height: 600,
            bytes_per_line: 800 * 4,
            offset: 0,
        });

        let vram = control.memory().map_err(Error::Framebuffer)?;
        let renderer = Renderer::new(driver, control.clone(), vram.clone());
        let emu = emu::Emulator::new(control, vram, vmtime, rom, renderer.control());
        Ok(Self { emu, renderer })
    }
}

impl ChangeDeviceState for VgaDevice {
    fn start(&mut self) {
        self.renderer.start();
    }

    async fn stop(&mut self) {
        self.renderer.stop().await;
    }

    async fn reset(&mut self) {
        self.emu.reset();
    }
}

impl ChipsetDevice for VgaDevice {
    fn supports_pio(&mut self) -> Option<&mut dyn PortIoIntercept> {
        Some(self)
    }

    fn supports_mmio(&mut self) -> Option<&mut dyn MmioIntercept> {
        Some(self)
    }

    fn supports_pci(&mut self) -> Option<&mut dyn PciConfigSpace> {
        Some(self)
    }
}

impl PciConfigSpace for VgaDevice {
    fn pci_cfg_read(&mut self, offset: u16, value: &mut u32) -> IoResult {
        self.emu.notify_pci_config_access_read(offset, value)
    }

    fn pci_cfg_write(&mut self, offset: u16, value: u32) -> IoResult {
        self.emu.notify_pci_config_access_write(offset, value)
    }

    fn suggested_bdf(&mut self) -> Option<(u8, u8, u8)> {
        Some((0, 8, 0)) // to match legacy Hyper-V behavior
    }
}

impl MmioIntercept for VgaDevice {
    fn mmio_read(&mut self, addr: u64, data: &mut [u8]) -> IoResult {
        self.emu.notify_mmio_read(addr, data);
        IoResult::Ok
    }

    fn mmio_write(&mut self, addr: u64, data: &[u8]) -> IoResult {
        self.emu.notify_mmio_write(addr, data);
        IoResult::Ok
    }

    fn get_static_regions(&mut self) -> &[(&str, std::ops::RangeInclusive<u64>)] {
        // N.B. The VM's RAM must be configured as unmapped in this region.
        &[("vga", 0xa0000..=0xbffff)]
    }
}

impl PortIoIntercept for VgaDevice {
    fn io_read(&mut self, io_port: u16, data: &mut [u8]) -> IoResult {
        let v = self.emu.io_port_read(io_port, data.len() as u16);
        data.copy_from_slice(&v.to_ne_bytes()[..data.len()]);
        IoResult::Ok
    }

    fn io_write(&mut self, io_port: u16, data: &[u8]) -> IoResult {
        let mut v = [0; 4];
        v[..data.len()].copy_from_slice(data);
        self.emu
            .io_port_write(io_port, data.len() as u16, u32::from_ne_bytes(v));
        IoResult::Ok
    }

    fn get_static_regions(&mut self) -> &[(&str, std::ops::RangeInclusive<u16>)] {
        &[
            ("mda", 0x3b0..=0x3bf),
            ("vga", 0x3c0..=0x3cf),
            ("cga", 0x3d0..=0x3df),
            ("s3", 0x4ae8..=0x4ae8),
        ]
    }
}

impl SaveRestore for VgaDevice {
    type SavedState = SavedStateNotSupported;

    fn save(&mut self) -> Result<Self::SavedState, vmcore::save_restore::SaveError> {
        todo!()
    }

    fn restore(
        &mut self,
        state: Self::SavedState,
    ) -> Result<(), vmcore::save_restore::RestoreError> {
        match state {}
    }
}